# -*- coding: utf8 -*- '''This tool will ftp all the files in a given directory to a given location if the file ftpallcfg.py exists in the directory it will be loaded and the values within it used, with the current directory used as the source directory. ftpallcfg.py file contains the following variables. =========================== server = <server to ftp to> username = <Username for access to given server> remote_dir = <remote server directory> =========================== ''' import ftplib import os import getpass import sys import time import socket __revision__ = 1.11 SLEEP_SECONDS = 1 class FtpAddOns(): PATH_CACHE = [] def __init__(self, ftp_h): self.ftp_h = ftp_h def ftp_exists(self, path): '''path exists check function for ftp handler''' exists = None if path not in self.PATH_CACHE: try: self.ftp_h.cwd(path) exists = True self.PATH_CACHE.append(path) except ftplib.error_perm, e: if str(e.args).count('550'): exists = False else: exists = True return exists def ftp_mkdirs(self, path, sep='/'): '''mkdirs function for ftp handler''' split_path = path.split(sep) new_dir = '' for server_dir in split_path: if server_dir: new_dir += sep + server_dir if not self.ftp_exists(new_dir): try: print 'Attempting to create directory (%s) ...' % (new_dir), self.ftp_h.mkd(new_dir) print 'Done!' except Exception, e: print 'ERROR -- %s' % (str(e.args)) def _get_local_files(local_dir): '''Retrieve local files list result_list == a list of dictionaries with path and mtime keys. ex: {'path':<filepath>,'mtime':<file last modified time>} ignore_dirs == a list of directories to ignore, should not include the base_dir. ignore_files == a list of files to ignore. ignore_file_ext == a list of extentions to ignore. ''' result_list = [] ignore_dirs = ['CVS', '.svn'] ignore_files = ['.project', '.pydevproject'] ignore_file_ext = ['.pyc'] base_dir = os.path.abspath(local_dir) for current_dir, dirs, files in os.walk(base_dir): for this_dir in ignore_dirs: if this_dir in dirs: dirs.remove(this_dir) sub_dir = current_dir.replace(base_dir, '') if sub_dir: break for this_file in files: if this_file not in ignore_files and os.path.splitext(this_file)[-1].lower() not in ignore_file_ext: filepath = os.path.join(current_dir, this_file) file_monitor_dict = { 'path': filepath, 'mtime': os.path.getmtime(filepath) } result_list.append(file_monitor_dict) return result_list def upload_all(server, username, password, base_local_dir, base_remote_dir, files_to_update=None): '''Upload all files in a given directory to the given remote directory''' continue_on = False login_ok = False server_connect_ok = False base_local_dir = os.path.abspath(base_local_dir) base_remote_dir = os.path.normpath(base_remote_dir) if files_to_update: local_files = files_to_update else: local_files = _get_local_files(base_local_dir) if local_files: ftp_h = ftplib.FTP() try: ftp_h.connect(server) server_connect_ok = True except socket.gaierror, e: print 'ERROR -- Could not connect to (%s): %s' % (server, str(e.args)) except IOError, e: print 'ERROR -- File not found: %s' % (str(e.args)) except socket.error, e: print 'ERROR -- Could not connect to (%s): %s' % (server, str(e.args)) ftp_path_tools = FtpAddOns(ftp_h) if server_connect_ok: try: ftp_h.login(username,password) print 'Logged into (%s) as (%s)' % (server, username) login_ok = True except ftplib.error_perm, e: print 'ERROR -- Check Username/Password: %s' % (str(e.args)) if login_ok: for file_info in local_files: filepath = file_info['path'] path, filename = os.path.split(filepath) remote_sub_path = path.replace(base_local_dir, '') remote_path = path.replace(base_local_dir, base_remote_dir) remote_path = remote_path.replace('\\', '/') # Convert to unix style if not ftp_path_tools.ftp_exists(remote_path): ftp_path_tools.ftp_mkdirs(remote_path) # Change to directory try: ftp_h.cwd(remote_path) continue_on = True except ftplib.error_perm, e: print 'ERROR -- %s' % (str(e.args)) if continue_on: if os.path.exists(filepath): f_h = open(filepath,'rb') filename = os.path.split(f_h.name)[-1] display_filename = os.path.join(remote_sub_path, filename) display_filename = display_filename.replace('\\', '/') print 'Sending (%s) ...' % (display_filename), send_cmd = 'STOR %s' % (filename) try: ftp_h.storbinary(send_cmd, f_h) f_h.close() print 'Done!' except Exception, e: print 'ERROR!' print str(e.args) print else: print "WARNING -- File no longer exists, (%s)!" % (filepath) ftp_h.quit() print 'Closing Connection' else: print 'ERROR -- No files found in (%s)' % (base_local_dir) return continue_on if __name__ == '__main__': import optparse default_config_file = u'ftpallcfg.py' # Create parser, and configure command line options to parse parser = optparse.OptionParser() parser.add_option("-l", "--local_dir", dest="local_dir", help="Local Directory (Defaults to CWD)", default='.') parser.add_option("-r", "--remote_dir", dest="remote_dir", help="[REQUIRED] Target Remote directory", default=None) parser.add_option("-u", "--username", dest="username", help="[REQUIRED] username", default=None) parser.add_option("-s","--server", dest="server", help="[REQUIRED] Server Address", default=None) (options,args) = parser.parse_args() if (options.username and options.server and options.remote_dir) or \ os.path.exists(default_config_file): local_dir = options.local_dir if os.path.exists(default_config_file): sys.path.append('.') import ftpallcfg try: server = ftpallcfg.server username = ftpallcfg.username remote_dir = ftpallcfg.remote_dir except AttributeError, e: print "ERROR --", str(e.args) print print 'Value(s) missing in %s file! The following values MUST be included:' % (default_config_file) print '================================' print 'server = <server to ftp to>' print 'username = <Username for access to given server>' print 'remote_dir = <remote server directory>' print '================================' sys.exit() else: server = options.server username = options.username remote_dir = options.remote_dir p = "FTP_PWD" try: upload_all(server, username, p, local_dir, remote_dir, []) except KeyboardInterrupt: print 'Exiting...' else: print 'ERROR -- Required option not given!' print __revision__ print __doc__ print parser.print_help()