zoukankan      html  css  js  c++  java
  • [Perl]FTP自动上传文件的脚本以及配置文件

     原文:http://blog.csdn.net/zhengyun_ustc/archive/2005/01/05/perlftpautoupload.aspx

    本应用程序设计的几个基本理念是:

    l         工具箱再利用:   尽可能利用已有的工具;

    l         简化运行步骤;   不引入过多的业务逻辑,满足的需求越简单越好。

    所以,我们定义了本应用程序依赖于以下几个工具的运行:

    l         ActivePerl-5.8.4.810-MSWin32-x86

    l         Upload.pl

    l         Upload.config

    我们将主要的执行逻辑都放在Perl源文件Upload.pl中了,配置文件为Upload.config。

    这个perl文件将执行的任务是, 按照指定的文件夹目录,自动将该文件夹下的所有文件上传到指定ftp站点的指定目录下。


          这个Perl脚本实际是从Uwe Keim 的《Perl Script for uploading modified files to a FTP-Server》继承下来的,
    只不过增加了容错反应和读取外部配置文件的部分,刨掉了与一般业务逻辑无关的读写access文件的部分。
          

          程序大致的流程是:

          第一步,尝试登陆ftp站点;

           第二步,在指定文件夹A类下寻找符合条件的文件,并将A类文件上传到FTP站点指定目录下;

           第三步,如果A类文件们全部上传成功,那么在指定文件夹B类下寻找指定文件, 并且上传到FTP指定目录下

            第四步,写成功/失败日志。

     

        最后,我们要写的成功/失败日志的格式如下所示: 

             成功: 生成一个名为“Upload_Succ_2005_01_04_17_23.log”的日志文件

           文件格式:输出上传时间,以及所有上传文件名及其大小和耗费的时间。

            失败: 生成一个名为“Upload_Fail_2005_01_04_17_23.log”的日志文件

           文件格式:输出上传时间,以及已经上传的文件名及其大小和耗费的时间,和失败的文件名及原因。

     

    配置perl脚本运行有两个办法:

    u       您可以在Windows计划任务中配置运行“Perl Upload.pl”的时间,这需要在Windows环境中配置ActivePerl 5.8.4.810;

    u       您也可以利用Perl2Exe(p2x-8.40-Win32)来将perl脚本编译为一个exe可执行程序,在计划任务中运行这个exe(这需要PerlCRT.dll在系统路径下)。

    [注意!]在运行之前,您必须修改“Upload.config”文件以配置所需的重要参数。


    外部配置参数

     

    在和perl脚本同一目录下的“Upload.config”配置文件中,是事先配置的六个外部参数: 

    n         参数1: ftp_server:

    n         FTP服务器的IP地址。

    n         参数2: ftp_dir:

    n         指定的FTP上传目录路径;

    n         参数3: ftp_uid:

    n         FTP的登陆用户名;

    n         参数4: ftp_pw:

    n         FTP的登陆密码;

    n         参数5: src_dir_WAVFiles,这是一个数组:

    n         指定A类文件夹,放置所有要上传的语音文件;

    n         注意:这个目录下的子文件夹也会被上传。

    n         参数6: src_dir_NamesListFile,这是一个数组:

    n         指定B类文件夹,放置B类文件.

    n         注意:这个目录下的子文件夹也会被上传。


    附录:
    Upoad.pl内容:
    #!/usr/bin/perl -w

    ####################################################################
    #
    # 工程项目: FTP自动上传两类文件
    #
    # 模块名称: FTPAutoUpload
    #
    # 模块任务: 按照指定的文件夹目录,自动将该文件夹下的所有文件上传到指定ftp站点的指定目录下
    #
    # 程序名称:     Upload.pl

    # 程序员:   Uwe Keim
    #                   郑昀
    # 历史记录:
    # 编号     日期       作者               备注
    # 1       2000          Uwe Keim    创建
    # 2       2005.1.5    郑昀              加入容错处理和读取外部配置文件部分
    #
    ####################################################################

    ##================================================================##
    ## 引用的库声明 2
    #use strict;
    use File::Copy;
    use File::stat;
    use File::Find;
    use Net::FTP;
    use Date::Pcalc qw(Delta_DHMS);
    use Date::Parse;
    use Win32::OLE;
    use Win32::OLE::Variant;
    ##================================================================##


    ##================================================================##
    ## 引用的库声明 1
    #- 读取ini配置文件的库
    use Config::IniFiles;
    my $cfg = Config::IniFiles->new( -file => "Upload.config" );
    ##================================================================##


    ##================================================================##
    ## 从配置文件读取外部参数 ##
    ##
    ## FTP服务器的IP地址 ##
    $ftp_server    = $cfg->val('FTPServer', 'ftp_server') || '';


    ## 指定的FTP上传目录路径 ##
    #! 切记:文件夹最后不要加"/"符号 !#
    $ftp_dir    = $cfg->val('FTPServer', 'ftp_dir') || '';


    ## FTP的登陆用户名 ##
    $ftp_uid   = $cfg->val('FTPServer', 'ftp_uid') || '';


    ## FTP的登陆密码 ##
    $ftp_pw     = $cfg->val('FTPServer', 'ftp_pw') || '';


    ## 指定文件夹“语音文件”,放置所有要上传的语音文件 ##
    #! 切记:文件夹最后不要加"\\"符号 !#
    @src_dir_WAVFiles = $cfg->val('SrcDirectory', 'src_dir_WAVFiles');


    ## 指定文件夹“命名对照列表文件TXT”,放置命名对照列表文件 ##
    #! 切记:文件夹最后不要加"\\"符号 !#
    @src_dir_NamesListFile = $cfg->val('SrcDirectory', 'src_dir_NamesListFile');


    ## 一个字符串集合,表明哪些类型的文件/文件夹将不被上传到服务器 ##
    @wc_exclude   = ("_vti",".mdb","\\bak","\\data","server.inc");
    ##================================================================##


    ##================================================================##
    ## 记录全部过程的日志文件准备
    $logfilename = 'upload.log';


    $log_cnt = 0;


    LOG("");
    LOG("自动上传FTP文件 Version 0.1");
    LOG("");
    LOG("用法: Perl Upload.pl");
    LOG("");
    ##================================================================##


    ##================================================================##
    ##=== 程序执行的第一步:尝试登陆ftp站点 ==========================##
    ##  $total_files是上传文件的数目
    $total_files = 0;


    ##  $processed_files是已上传文件的数目
    $processed_files = 0;


    ##  $skipped_files是跳过文件的数目
    $skipped_files = 0;


    ##  $start_date计算出当前开始的时间
    $start_date = timeString(time);


    ## $g_nUploadSuccess代表是否已经完全上传,-1为不是,1为是:
    my $g_nUploadSuccess = 1;
    ## $g_nIsAllWAVsFile_UploadSuccess代表是否已经完全将A类文件上传,-1为不是,1为是:
    my $g_nIsAllWAVsFile_UploadSuccess = 1;
    ## $g_strLastError代表上次错误原因:
    my $g_strLastError = "";


    LOG("正在链接至指定FTP服务器($ftp_server)...");
    $ftp = Net::FTP->new($ftp_server);
    if($@)
    {
     $g_strLastError = "不能连接到FTP服务器,错误原因:".$@;
     LOG("$g_strLastError@\n");
     $g_nUploadSuccess = -1;
    }
    else
    {
     $ftp->login($ftp_uid, $ftp_pw);
     if($@)
     {
      $g_strLastError = "不能登陆FTP服务器,错误原因:".$@;
      LOG("$g_strLastError\n");
      $g_nUploadSuccess = -1;
     }
     else
     {
      $ftp->binary;
      LOG("链接FTP服务器成功!");
    ##================================================================##


    ##================================================================##
    ##=== 程序执行的第二步,将指定A类文件夹下所有A类文件上传到FTP站点指定目录下 ===##

      my %lookup;
      LOG("准备上传“A类文件”目录(@src_dir_WAVFiles)下的所有文件!");
      find(\&processFiles, @src_dir_WAVFiles);
      LOG("目录(@src_dir_WAVFiles)已经处理完毕,结果是:");


    ##================================================================##
    ##=== 程序执行的第三步,将指定B类文件夹下B类文件上传到FTP站点指定目录下 ===##
      if($g_nIsAllWAVsFile_UploadSuccess > 0)
      {
       LOG("+===============================+");
       LOG("准备上传B类目录(@src_dir_NamesListFile)下的所有文件!");
       find(\&processFiles, @src_dir_NamesListFile);
       LOG("目录(@src_dir_NamesListFile)已经处理完毕,结果是:");
       LOG("-===============================-");
      }
      else
      {
       LOG("-===============================-");
       LOG("由于A类文件目录并没有完全上传,所以本B类文件不上传!");
       LOG("-===============================-");
      }
    ##================================================================##


    ##================================================================##
    # 日志文件的最后是一个统计报告
      $span = calcDeltaSeconds($start_date,timeString(time));
      LOG("上传结果:成功。\n花费:$span 秒, 总共处理了 $total_files 个文件, 其中 $processed_files 上传成功, 跳过了 $skipped_files 个文件。");

      $ftp->quit()         or warn "unable to quit: $@\n";
     }
     closeLogfile();
    }
    ##================================================================##


    ##================================================================##
    ##=== 程序执行的第四步,写成功日志 ===============================##
    if($g_nIsAllWAVsFile_UploadSuccess > 0 && $g_nUploadSuccess > 0)
    {
     $logfilename = 'Upload_Succ_'.shortTimeString(time).'.log';

     $log_cnt = 0;

     LOG("");
     LOG("FTP自动上传文件 Version 0.1");
     LOG("");
     LOG("上传结果:成功。\n花费:$span 秒, 总共处理了 $total_files 个文件, 其中 $processed_files 上传成功, 跳过了 $skipped_files 个文件。");
     LOG("");
     closeLogfile();
    }
    ##================================================================##


    ##================================================================##
    ##=== 程序执行的第四步,写失败日志 ===============================##
    if($g_nIsAllWAVsFile_UploadSuccess < 0 || $g_nUploadSuccess < 0)
    {
     $logfilename = 'Upload_Fail_'.shortTimeString(time).'.log';

     $log_cnt = 0;

     LOG("");
     LOG("FTP自动上传文件 Version 0.1");
     LOG("");
     LOG("上传结果:失败。失败原因:$g_strLastError。\n花费:$span 秒, 总共处理了 $total_files 个文件, 其中 $processed_files 上传成功, 跳过了 $skipped_files 个文件。");
     LOG("");
     closeLogfile();
    }
    ##================================================================##


    ####################################################################


    ## 以下是子函数体的定义


    ##-------------------------------------------------------------##
    ##
    ## 函数名称:processFiles
    ## 功能:   
    ##           得到指定文件夹下的所有文件以及子文件夹,然后依次处理它们。
    ##
    ## 程序员:  Uwe Keim
    ##
    ## 历史记录:
    ## 编号     日期       作者    备注
    ## 1        2000       Uwe Keim
       
    ##
    ##-------------------------------------------------------------##
    sub processFiles
    {
     my $srcdir = fsToBs($File::Find::dir);
     my $srcpath = fsToBs($File::Find::name);
     my $base = fsToBs($File::Find::topdir);

     foreach my $exclude (@wc_exclude) {
      if ( index($srcpath, $exclude)>-1 ) {
       $File::Find::prune = 1 if -d $srcpath;
       return;
      }
     }

     # no DIRECT processing of directories.
     if ( -d $srcpath ) {
      return;
     }

     my $dstdir = $srcdir;
     my $dstpath = $srcpath;
     $dstdir =~ s{\Q$base\E}{$ftp_dir}is;
     $dstpath =~ s{\Q$base\E}{$ftp_dir}is;
     $dstdir = bsToFs($dstdir);
     $dstpath = bsToFs($dstpath);

     processFile($srcpath,$dstpath,$dstdir);
    }

    sub processFile
    {
     my ($src,$dst,$dstdir) = @_;

     $total_files++;
     LOG("正在处理文件 $total_files \"$src\"...");

     # --------------------
     # check time.

     my $need_upload = 0;

     # create time.
     my $t1 = $lookup{$src};
     my $t2 = timeString(stat($src)->mtime);

     if ( not defined $t1 ) {
      $lookup{$src} = $t2;
      $need_upload = 1;
     } else {
      my $delta_sec = calcDeltaSeconds($t1,$t2);
      $need_upload = 1 if $delta_sec>5;     # 5 seconds as tolerance.
     }

     # --------------------

     if ( $need_upload>0 ) {
      $processed_files++;

      LOG("正在上传文件:从源 \"$src\" 到目标 \"$dst\"...");

      $ftp->mkdir($dstdir,1);
      $ftp->put($src, $dst) or  $g_nIsAllWAVsFile_UploadSuccess=-1;
      if($g_nIsAllWAVsFile_UploadSuccess < 0)
      {
       LOG("不能上传文件:从源 \"$src\" 到目标 \"$dst\" (dst-dir: \"$dstdir\")。\n");
       if($@)
       {
        LOG("错误原因是:$@\n");
       }
      }

     } else {
      $skipped_files++;
     }
    }


    ####################################################################

    sub bsToFs {
     my ($s) = @_;
     $s =~ s/\\/\//gis;
     return $s;
    }

    sub fsToBs {
     my ($s) = @_;
     $s =~ s/\//\\/gis;
     return $s;
    }

    sub timeString {
     my ($tm) = @_;
     my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($tm);
     return sprintf("%04d-%02d-%02d %02d:%02d:%02d", $year+1900, $mon+1, $mday, $hour, $min, $sec);
    }

    sub shortTimeString {
     my ($tm) = @_;
     my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($tm);
     return sprintf("%04d_%02d_%02d_%02d_%02d", $year+1900, $mon+1, $mday, $hour, $min);
    }

    # input dates as string "YYYY-MM-DD HH:MM:SS".
    # earlier as first parameter, later as second.
    sub calcDeltaSeconds {
     my ($t1,$t2) = @_;

     my ($year1,$month1,$day1,$hh1,$mm1,$ss1) = scanDate($t1);
     my ($year2,$month2,$day2,$hh2,$mm2,$ss2) = scanDate($t2);

     my ($days, $hours, $minutes, $seconds) = Delta_DHMS(
      $year1, $month1, $day1, $hh1, $mm1, $ss1,     # earlier.
      $year2, $month2, $day2, $hh2, $mm2, $ss2);    # later.

     return $seconds + $minutes*60 + $hours*60*60 + $days*60*60*24.
    }

    sub removeFilename {
     my ($s) = @_;
     my $pos = rindex($s,'\\');
     return substr($s, 0, $pos);
    }

    # format: "2000-09-29 09:09:51".
    sub scanDate {
     my ($date) = @_;
     my ($year, $month, $day, $hour, $minute, $seconds);

     $year   = substr($date, 0, 4);
     $month  = substr($date, 5, 2);
     $day   = substr($date, 8, 2);
     $hour   = substr($date, 11, 2);
     $minute  = substr($date, 14, 2);
     $seconds = substr($date, 17, 2);

     return ($year, $month, $day, $hour, $minute, $seconds);
    }

    ####################################################################

    sub LOG {
     my ($text) = @_;
     my $time = timeString time;

     # log to stdout.
     print "[$time] $text\n";

     # log to logfile.
     my $LOG_STEP = 10;
     flushLogfile() if ($log_cnt % $LOG_STEP)==0 or $log_cnt==0;
     $log_cnt++;
     print HLOG "[$time] $text\n";
    }

    sub openLogfile {
     closeLogfile();
     open(HLOG,">>$logfilename") or die("打开日志文件出错:文件名为 $logfilename ;错误原因为: $!"); 
    };

    sub closeLogfile {
     close HLOG if defined HLOG;
    }

    sub flushLogfile {
     closeLogfile();
     openLogfile();
    }
    ####################################################################

    附录:
    Upoad.config内容:
    ##================================================================##
    ## 配置的外部参数 ##
    ##
    [FTPServer]
    #- FTP服务器的IP地址 -#
    ftp_server   =
    #- 指定的FTP上传目录路径 -#
    #! 切记:文件夹最后不要加"/"符号 !#
    ftp_dir       =
    #- FTP的登陆用户名 -#
    ftp_uid    =
    #- FTP的登陆密码 -#
    ftp_pw    =

    ##================================================================##


    ##================================================================##
    ## 配置的外部参数 ##
    ##
    [SrcDirectory]
    #- 指定文件夹“语音文件”,放置所有要上传的语音文件 -#
    #! 切记:文件夹最后不要加"\"符号 !#
    src_dir_WAVFiles  =


    #- 指定文件夹“命名对照列表文件TXT”,放置命名对照列表文件 -#
    #! 切记:文件夹最后不要加"\"符号 !#
    src_dir_NamesListFile =
    ##================================================================##

  • 相关阅读:
    Effective C# 学习笔记(三十五) 了解PLINQ如何实现并行算法
    Effective C# 学习笔记(三十八)理解Dynamic的得与失
    转单例的分析
    获取系统当前音量 和 监听系统音量 ios
    (转) iphone开发资源汇总
    xcode show line numbers
    为了编程方便的效率宏定义的一些代码
    ios6下cocos2d & ipad 调用摄像头报错问题 (在竖屏情况下调用Camera 会导致转屏)
    转KVC
    不记住的
  • 原文地址:https://www.cnblogs.com/huqingyu/p/108343.html
Copyright © 2011-2022 走看看