Pergunta

Eu estou aprendendo python e tentando escrever um código para sincronizar duas pastas:uma delas é sobre o servidor de ftp, o outro está no meu disco local.Até então, eu escrevi um código de trabalho, mas eu tenho uma pergunta ou duas sobre ele :)

import os
from ftplib import FTP

h_local_files = [] # create local dir list
h_remote_files = [] # create remote dir list

h_local = 'C:\\something\\bla\\' # local dir

ftp = FTP('ftp.server.com')
ftp.login('user', 'pass')

if os.listdir(h_local) == []:
    print 'LOCAL DIR IS EMPTY'
else:
    print 'BUILDING LOCAL DIR FILE LIST...'
    for file_name in os.listdir(h_local):
        h_local_files.append(file_name) # populate local dir list

ftp.sendcmd('CWD /some/ftp/directory')
print 'BUILDING REMOTE DIR FILE LIST...\n'
for rfile in ftp.nlst():
    if rfile.endswith('.jpg'): # i need only .jpg files
        h_remote_files.append(rfile) # populate remote dir list

h_diff = sorted(list(set(h_remote_files) - set(h_local_files))) # difference between two lists

for h in h_diff:
    with open(os.path.join(h_local,h), 'wb') as ftpfile:
        s = ftp.retrbinary('RETR ' + h, ftpfile.write) # retrieve file
        print 'PROCESSING', h
        if str(s).startswith('226'): # comes from ftp status: '226 Transfer complete.'
            print 'OK\n' # print 'OK' if transfer was successful
        else:
            print s # if error, print retrbinary's return

Este pedaço de código deve fazer duas listas de python:uma lista de arquivos no diretório local e uma lista de arquivos no diretório de ftp.Após a remoção de duplicatas, a partir de listas, o script deve fazer o download de 'falta' arquivos para o meu diretório local.

Por agora, esta parte do código é fazer o que eu preciso, mas eu notei que quando eu executar a minha saída não está agindo como eu imagino que seria act :)

Por exemplo, a minha saída de corrente continua:

PROCESSING 2012-01-17_07.05.jpg
OK

# LONG PAUSE HERE

PROCESSING 2012-01-17_07.06.jpg
OK

# LONG PAUSE HERE

PROCESSING 2012-01-17_07.06.jpg
OK

etc...

mas eu imagino que ele deve funcionar como este:

PROCESSING 2012-01-17_07.05.jpg
# LONG PAUSE HERE (WHILE DOWNLOADING)
OK

PROCESSING 2012-01-17_07.06.jpg
# LONG PAUSE HERE (WHILE DOWNLOADING)
OK

PROCESSING 2012-01-17_07.06.jpg
# LONG PAUSE HERE (WHILE DOWNLOADING)
OK

etc...

Como eu disse, eu só comecei a aprender python, e talvez eu vou fazer algumas coisas aqui completamente errado (if str(s).startswith('226')????).Talvez eu não conseguir isso com oftplib só?No fim, minhas perguntas são:

O que estou fazendo de errado aqui?:)
Como produzir 'adequado' de saída e existe um caminho para a impressão de algum tipo de status durante o download de um arquivo (pelo menos uma linha de pontos), por exemplo:

PROCESSING 2012-01-17_07.05.jpg
..........
OK

PROCESSING 2012-01-17_07.06.jpg
......
OK

PROCESSING 2012-01-17_07.06.jpg
...............
OK

etc...

Muito obrigado por ajudar!

Foi útil?

Solução

retrybinary blocos até que seja concluída.É por isso que você vê Processing ZZZ\n OK imediatamente, porque ele ocorre após a chamada para retrbinary foi concluída.

Se você deseja imprimir . para cada chamada, você precisará fornecer uma função de retorno de chamada para fazer isso.aqui é a docstring para retrbinary:

    """Retrieve data in binary mode.  A new port is created for you.

    Args:
      cmd: A RETR command.
      callback: A single parameter callable to be called on each
                block of data read.
      blocksize: The maximum number of bytes to read from the
                 socket at one time.  [default: 8192]
      rest: Passed to transfercmd().  [default: None]

    Returns:
      The response code.
    """

Então, você precisa fornecer diferentes de chamada de retorno que tanto grava o arquivo e imprime '.'

import sys # At the top of your module.

# Modify your retrbinary    
ftp.retrbinary('RETR ' + h, lambda s: ftpfile.write(s) and sys.stdout.write('.'))

Você pode ter que editar fragmento de código, mas ele deve lhe dar uma idéia do que fazer.

Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top