Extending Python's os.walk function on FTP server

后端 未结 4 1443
[愿得一人]
[愿得一人] 2021-02-15 15:01

How can I make os.walk traverse the directory tree of an FTP database (located on a remote server)? The way the code is structured now is (comments provided):

相关标签:
4条回答
  • 2021-02-15 15:23

    All you need is utilizing the python's ftplib module. Since os.walk() is based on a Breadth-first search algorithm you need to find the directories and file names at each iteration, then continue the traversing recursively from the first directory. I implemented this algorithm about 2 years ago for using as the heart of FTPwalker, which is an optimum package for traversing extremely large directory trees Through FTP.

    from os import path as ospath
    
    
    class FTPWalk:
        """
        This class is contain corresponding functions for traversing the FTP
        servers using BFS algorithm.
        """
        def __init__(self, connection):
            self.connection = connection
    
        def listdir(self, _path):
            """
            return files and directory names within a path (directory)
            """
    
            file_list, dirs, nondirs = [], [], []
            try:
                self.connection.cwd(_path)
            except Exception as exp:
                print ("the current path is : ", self.connection.pwd(), exp.__str__(),_path)
                return [], []
            else:
                self.connection.retrlines('LIST', lambda x: file_list.append(x.split()))
                for info in file_list:
                    ls_type, name = info[0], info[-1]
                    if ls_type.startswith('d'):
                        dirs.append(name)
                    else:
                        nondirs.append(name)
                return dirs, nondirs
    
        def walk(self, path='/'):
            """
            Walk through FTP server's directory tree, based on a BFS algorithm.
            """
            dirs, nondirs = self.listdir(path)
            yield path, dirs, nondirs
            for name in dirs:
                path = ospath.join(path, name)
                yield from self.walk(path)
                # In python2 use:
                # for path, dirs, nondirs in self.walk(path):
                #     yield path, dirs, nondirs
                self.connection.cwd('..')
                path = ospath.dirname(path)
    

    Now for using this class, you can simply create a connection object using ftplib module and pass the the object to FTPWalk object and just loop over the walk() function:

    In [2]: from test import FTPWalk
    
    In [3]: import ftplib
    
    In [4]: connection = ftplib.FTP("ftp.uniprot.org")
    
    In [5]: connection.login()
    Out[5]: '230 Login successful.'
    
    In [6]: ftpwalk = FTPWalk(connection)
    
    In [7]: for i in ftpwalk.walk():
                print(i)
       ...:     
    ('/', ['pub'], [])
    ('/pub', ['databases'], ['robots.txt'])
    ('/pub/databases', ['uniprot'], [])
    ('/pub/databases/uniprot', ['current_release', 'previous_releases'], ['LICENSE', 'current_release/README', 'current_release/knowledgebase/complete', 'previous_releases/', 'current_release/relnotes.txt', 'current_release/uniref'])
    ('/pub/databases/uniprot/current_release', ['decoy', 'knowledgebase', 'rdf', 'uniparc', 'uniref'], ['README', 'RELEASE.metalink', 'changes.html', 'news.html', 'relnotes.txt'])
    ...
    ...
    ...
    
    0 讨论(0)
  • 2021-02-15 15:28

    I wrote a library pip install walk-sftp. Event though it is named walk-sftp I included a WalkFTP class that lets you filter by start_date of files & end_date of files. You can even pass in a processing_function that returns True or False to see whether your process to clean & store data works. It also has a log parameter (pass filename) that uses pickle & keeps track of any progress so you don't overwrite or have to keep track of dates making backfilling easier.

    https://pypi.org/project/walk-sftp/

    0 讨论(0)
  • I needed a function like os.walk on FTP and there where not any so i thought it would be useful to write it , for future references you can find last version here

    by the way here is the code that would do that :

    def FTP_Walker(FTPpath,localpath):
        os.chdir(localpath)
        current_loc = os.getcwd()
        for item in ftp.nlst(FTPpath):
            if not is_file(item):
                yield from FTP_Walker(item,current_loc)
    
            elif is_file(item):
                yield(item)
                current_loc = localpath
            else:
                print('this is a item that i could not process')
        os.chdir(localpath)
        return
    
    
    def is_file(filename):
        current = ftp.pwd()
        try:
            ftp.cwd(filename)
        except Exception as e :
            ftp.cwd(current)
            return True
    
        ftp.cwd(current)
        return False
    

    how to use:

    first connect to your host :

    host_address = "my host address"
    user_name = "my username"
    password = "my password"
    
    
    ftp = FTP(host_address)
    ftp.login(user=user_name,passwd=password)
    

    now you can call the function like this:

    ftpwalk = FTP_Walker("FTP root path","path to local") # I'm not using path to local yet but in future versions I will improve it. so you can just path an '/' to it 
    

    and then to print and download files you can do somthing like this :

    for item in ftpwalk:
    ftp.retrbinary("RETR "+item, open(os.path.join(current_loc,item.split('/')[-1]),"wb").write) #it is downloading the file 
    print(item) # it will print the file address
    

    ( i will write more features for it soon so if you need some specific things or have any idea that can be useful for users i'll be happy to hear that )

    0 讨论(0)
  • 2021-02-15 15:36

    Im going to assume this is what you want ... although really I have no idea

    ssh = paramiko.SSHClient()
    ssh.connect(server, username=username, password=password)
    ssh_stdin, ssh_stdout, ssh_stderr = ssh.exec_command("locate my_file.txt")
    print ssh_stdout
    

    this will require the remote server to have the mlocate package `sudo apt-get install mlocate;sudo updatedb();

    0 讨论(0)
提交回复
热议问题