Paramiko Python:IOError:[Errno 13] 权限被拒绝

10

问题:

我能这样做吗:

self.sftp.put(sourceFilePath, final_destination, use_sudo=True)

我可以创建文件夹,但不能创建文件吗?需要显式调用sudo或设置paramiko中的某些内容吗?应该将文件复制到可许可的空间并更改所有者权限吗?有没有一种方法可以在不使用密钥或处理ssh.exec_command("sudo mv")的情况下给paramikko sudoer?我错过了什么?

代码:

class Ssh(object):

    def __init__(self):
        super(Ssh, self).__init__()

    def setup(self):
        '''Setup connection'''
        try:
            # DEBUG
            paramiko.common.logging.basicConfig(level=paramiko.common.DEBUG)
            #set username & password
            username = 'sgdevbox'
            password = MainFrame.ssh_pass
            host = '192.168.0.170'
            port = 22
            self.transport = paramiko.Transport((host, port))
            self.transport.connect(username = username, password = password)
            self.sftp = paramiko.SFTPClient.from_transport(self.transport)
            print(self.sftp.sock)
        except Exception, e:
            print(traceback.format_exc())

    def putFiles(self, sources, listingSku):
        '''
        Upload images to server along with all currentItemInfo, plus initials and date
        Basically build the auction and put it into the queue for verification
        '''
        print('\n# Ssh.putFiles() #')
        if isinstance(sources, unicode):
            sources = {sources,'True'}
        try:
            self.setup()
            destination = '/var/www'
            cwd = os.getcwd()
            for source in sources:
                filename = os.path.split(source)[-1]
                destinationFolder = listingSku
                final_path = posixpath.join(destination,destinationFolder)

                try:
                    self.sftp.mkdir(final_path, mode=777)
                except:
                    print(traceback.format_exc())
                final_destination = posixpath.join(final_path, filename)
                sourceFilePath = os.path.join(cwd,source)
                print('\n# Source Path: {}\n# Destination Path: {}\n\n'.format(sourceFilePath,final_destination))
                self.sftp.put(sourceFilePath, final_destination)
        except Exception, e:
            print(traceback.format_exc())
        return

回溯信息:

# Source Path: C:\A\Long\Path\622-402_01.JPEG
# Destination Path: /var/www/WOOBLE-WAMBLER-SPAM-1235/622-402_01.JPEG


DEBUG:paramiko.transport.sftp:[chan 1] open('/var/www/WOOBLE-WAMBLER-SPAM-1235/622-402_01_swatch.JPEG', 'wb')
Traceback (most recent call last):
  File "display_image.py", line 67, in putFiles
    self.sftp.put(sourceFilePath, final_destination)
  File "C:\Python27\lib\site-packages\paramiko\sftp_client.py", line 565, in put
    fr = self.file(remotepath, 'wb')
  File "C:\Python27\lib\site-packages\paramiko\sftp_client.py", line 245, in open
    t, msg = self._request(CMD_OPEN, filename, imode, attrblock)
  File "C:\Python27\lib\site-packages\paramiko\sftp_client.py", line 635, in _request
    return self._read_response(num)
  File "C:\Python27\lib\site-packages\paramiko\sftp_client.py", line 682, in _read_response
    self._convert_status(msg)
  File "C:\Python27\lib\site-packages\paramiko\sftp_client.py", line 710, in _convert_status
    raise IOError(errno.EACCES, text)
IOError: [Errno 13] Permission denied

我查看了其他一些帖子:

其中一些帖子有点旧,但似乎表明paramiko没有实现它? Fabric有一个已实现的版本,但我不确定是否要添加更多的依赖项。

(Pdb) import pkg_resources
(Pdb) pkg_resources.get_distribution('paramiko').version
'1.13.0'

使用pysftp会出现相同的错误。不幸的是,我没有使用Fabric或更改为root的选项。这很糟糕,特别是因为我能够使用winscp或fireftp连接。有任何新想法吗? - lukik
@lukik 我在下面更新了我的回答。 - jmunsch
4个回答

8

我在使用sftp.get时遇到了相同的错误。

我尝试过:

sftp.get('/remote_server/readme.txt', localpath='C:\\Users\\user1\\Desktop')

遇到了上述错误:[Errno 13] 拒绝访问。

解决方法是我们需要指定完整的路径,包括文件名。

sftp.get('/remote_server/readme.txt', localpath='C:\\Users\\user1\\Desktop\\readme.txt')

4

1) 关于将文件夹从 /home/user/Desktop 移动到 /var/www 的 Cron 任务

2) 作为 root 用户登录(存在明显的安全问题)

在主机服务器上运行sudo passwd root后,我现在可以使用 root 身份传输文件到 /var/www。

我还将用户添加到了 www-data 中,并递归更改了文件和目录的所有权,但我认为设置 root 密码是解决问题的关键。

不要执行以上操作:

更改权限和/或所有权

在 Linux 上:如果您可以通过 ssh 登录:

ls -ld /path/to/location

以查看谁拥有该目录并具有读/写权限。

bob@bob-p7-1298c:~$ ls -ld /var/www
drwxr-xr-x 3 root root 4096 Sep 24 10:39 /var/www

那么可以考虑使用以下命令:

  • usermod
  • addgroup
  • useradd
  • chown
  • chmod

来给用户授予读写权限。

实现方式有以下几种:

  • 改变目录的所有者
  • 将用户添加到目录所在组中
  • 创建一个新组并将其设置为目录的组
  • 修改目录所有者
  • 更改所有者、组或公共的读/写权限。

参见:


1
但是OP(和我)正在使用Windows...您认为这是相同的权限问题吗? - Francesco Pegoraro

1
我自己在Windows上遇到了“权限被拒绝”的错误。我的代码看起来像这样 -
sftp.put(sftp_local_path + "\\filename.txt", sftp_remote_path)

然后我把它改成了以下这样 -

with pysftp.Connection(host=sftp_host,username=sftp_user_id, password=sftp_pwd, port=sftp_port, cnopts=cnopts ) as sftp:
    with sftp.cd(sftp_remote_path):
        sftp.put(sftp_local_path + "\\filename.txt")

我成功地上传了文件。


-1
当我使用pysftp put()文件时,出现了非常相似的回溯错误。原来我试图将一个与sftp目录中已经存在的文件同名的文件进行put()操作。在put()之前更改文件名可以解决这个问题。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接