免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1937 | 回复: 0
打印 上一主题 下一主题

一个ftp同步的perl程序 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2003-12-22 20:56 |只看该作者 |倒序浏览
应该是一个老外写的,感觉很不错,贴上来给大家看看,呵呵

  1. #!/usr/bin/perl
  2. use Net::FTP;
  3. use strict;
  4. use Getopt::Long;

  5. # Variables set in the control file.
  6. #   sync=0|1|fn                        -- Default of the -sync option.
  7. #   host=hostname                -- Host to send to.
  8. #   acct=acctname                -- Account name
  9. #   pwd=password                -- Password to send.
  10. #   cd=initdir                        -- Initial directory.
  11. #   
  12. # The various control options cat be set with lines of the following
  13. # form:
  14. #   name = ...                        Set the value.
  15. #   name += ...                        Add to the value.
  16. # The value is a list of items, "strint" for a literal match, /str/ for a
  17. # pattern match.  The available variables are:
  18. #   send                Matched against local plain files, to transmit.
  19. #   dontsend                Files maching send which should not be sent.
  20. #   preserve                Matched against remote files, and
  21. #                        prevents deletion of remote files not present and sent.
  22. #   descend                Matches local directory names to descend.
  23. #   dontdescend                Local directories matched by descend, which should
  24. #                        not be descended.
  25. #   presdir                Remote directories to preserve, even though not on
  26. #                        the descend list.

  27. # Read the control file.  Info is placed into a hash given as a reference.
  28. # Initial values are preserved for variables not mentioned, and for +=
  29. # assignments.  The option required tells if the control file must exist.
  30. # If required is false, and the file cannot be opened, the function simply
  31. # returns w/o updating the hash.  If required and will not read, the function
  32. # dies.  The contents of hasref define the legal variables,
  33. #   readctl($conn, fn, required, stringargsref, patargsref)
  34. sub readctl {
  35.     my $conn = shift @_;
  36.     my $fn = shift @_;
  37.     my $req = shift @_;
  38.     my $strings = shift @_;
  39.     my $pats = shift @_;

  40.     if(!open(IN, $fn) && $req) {
  41.         &expire($conn, "Cannot read $fn");
  42.     }

  43.     my $ln;
  44.     while($ln = <IN>; || shift @_) {
  45.         # Parse the line and make sure the setting is legal.
  46.         chomp $ln;
  47.         $ln or next;
  48.         $ln =~ /^([a-zA-Z]+)\s*(\=|\+\=)\s*(\S.*)?$/ or
  49.             &expire($conn, "$fn:$.: Cannot parse");
  50.         my ($name, $sym, $data) = ($1, $2, $3);
  51.         exists $strings->;{$name} || exists $pats->;{$name} or
  52.             &expire($conn, "$fn:$.: Unknown control variable $name.");
  53.         if($sym eq '+=' && !exists $pats->;{$name}) {
  54.             &expire($conn, "$fn:$.: Value $name cannot be extended.");
  55.         }

  56.         # Update the hash.
  57.         if(exists $pats->;{$name}) {
  58.             # Pattern.
  59.             if($sym eq '=') { $pats->;{$name} = ''; }
  60.             while($data) {
  61.                 $pats->;{$name} .= '|' if($pats->;{$name});
  62.                 my $p;
  63.                 if($data =~ s|^/(.*?[^\\])/\s*||) {
  64.                     # Just add a pattern to the list.
  65.                     $p = $1;
  66.                 } elsif($data =~ s|^"(.*?[^\\])"\s*||) {
  67.                     # A string specified.  Make a pattern for exact match.
  68.                     $p = $1;
  69.                     $p =~ s/([^a-zA-Z0-9])/\\$1/g;
  70.                     $p = "^($p)\$";
  71.                 } else {
  72.                     &expire($conn, "$fn:$.: Bad match item spec.");
  73.                 }

  74.                 # Check for syntax error in the pattern, then add it
  75.                 # to the whole match pattern.
  76.                 defined eval("'x' =~ /$p/") or
  77.                     &expire($conn, "$fn:$.: Bad pattern $p");
  78.                 $pats->;{$name} .= "($p)";
  79.             }
  80.         } else {
  81.             # Just plain value.
  82.             $strings->;{$name} = $data;
  83.         }
  84.        
  85.     }

  86.     close(IN);
  87. }

  88. # Options.
  89. my $norun = 0;
  90. my $rcfile = ".webserver";
  91. my $csync = undef;
  92. GetOptions("n|norun" =>; \$norun, "r|rmtdir|server=s" =>; \$rcfile,
  93.            "s|sync:s" =>; \$csync);
  94. $rcfile =~ /^\./ or $rcfile = ".$rcfile";

  95. # Read the top-level command file.
  96. my %strings = ( sync =>; 0, host =>; 0, acct =>; 0, pwd =>; 0, cd =>; '');
  97. my %pats = ( send =>; "(\\.html\$)", preserve =>; "", descend =>; "(.)",
  98.              presdir =>; "", dontsend =>; "", dontdescend =>; "" );
  99. readctl(0, $rcfile, 1, \%strings, \%pats, @ARGV);
  100. if(defined $csync) {
  101.     # From command line.
  102.     if($csync eq '') { $csync = 1; }
  103.     $strings{"sync"} = $csync;
  104. }
  105. if($strings{"sync"} == 1) { $strings{"sync"} == 'syncfile'; }

  106. #foreach my $n(keys %strings) { print "strings{$n} = $strings{$n}\n"; }
  107. #print "\n";
  108. #foreach my $n(keys %pats) { print "pats{$n} = $pats{$n}\n"; }
  109. #print "\n";

  110. # Time offset.
  111. my $syncoff = 0;

  112. # Close and maybe exit.
  113. my $was_err = 0;
  114. sub expire {
  115.     my $conn = shift @_;
  116.     my $msg = shift @_;
  117.     my ($fatal) = (@_, 1);

  118.     my $err = '';
  119.     if($conn) {
  120.         $err = $conn->;message();
  121.     }
  122.     chomp $err;
  123.     $err =~ s/\.\s*$//;
  124.     $msg =~ s/\!\!/$err/;
  125.     print "$msg.\n";
  126.     if($fatal) {
  127.         $conn->;quit() if $conn;
  128.         exit(2);
  129.     } else {
  130.         $was_err = 1;
  131.     }
  132. }

  133. # Read the curr directory on the server, and return a pair of hash references.
  134. # The first is to a hash giving the name and mod date of each ordinary file,
  135. # and the second gives the directories, with value 1.
  136. sub rmt_contents {
  137.     my $conn = shift @_;
  138.     my %files = ();
  139.     my %dirs = ();
  140.     my $nttime = '\d\d\-\d\d\-\d\d\s+\d\d\:\d\d[AP]M';

  141.     my @names = $conn->;dir();
  142.     int($conn->;code() / 100) == 2 or
  143.         expire($conn, "Remote list failed: !!");
  144.     foreach my $l(@names) {
  145.         chomp $l;
  146.         my ($name, $isdir);
  147.         if($l =~ /^[\-d]([\-r][\-w][\-xs]){2}[\-r][\-w][\-xt]\s/) {
  148.             # Smells like Unix (inoring specials, etc.)
  149.             my ($mode, $links, $uid, $gid, $size, $d1, $d2, $d3, $n) =
  150.                 split(/\s+/, $l, 9);
  151.             $name = $n;
  152.             $isdir = ($l =~ /^d/);
  153.         } elsif($l =~ /^$nttime\s+(\<(DIR)\>;\s+|\d+\s)(.*)$/) {
  154.             # Smells NT
  155.             $name = $3;
  156.             $isdir = ($2 eq 'DIR');
  157.         } else {
  158.             # Can't figure it out, or its the wrong kind of thing.
  159.             next;
  160.         }
  161.         if($isdir) {
  162.             next if $name eq '.' || $name eq '..';
  163.             $dirs{"$name"} = 1;
  164.         } else {
  165.             my $time = $conn->;mdtm($name);
  166.             $files{"$name"} = $time + $syncoff;
  167.         }
  168.     }
  169.     return (\%files, \%dirs);
  170. }

  171. # Read the curr local directory, and return a pair of references.
  172. # The first is to a hash giving the name and mod dat of each ordinary file,
  173. # and the second is to an array of directory names in the current directory.
  174. # The function ignores any file starting with ., and obeys the send,
  175. # dontsend, descend and dontdescend patterns.  That means that files which
  176. # don't match send, or do match dontsend, are ignored.  Likewise directories
  177. # not matchind descend or matching dontdescend are not reported.  
  178. sub loc_contents {
  179.     my $conn = shift @_; # Only for aborts.
  180.     my $strref = shift @_;
  181.     my $pref = shift @_;

  182.     my %files = ();
  183.     my %dirs = ();

  184.     opendir(CD, '.') or expire($conn, "Directory open failed: $!");
  185.     while(my $name = readdir(CD))
  186.     {
  187.         next if($name =~ /^\./);
  188.         if(-f $name) {
  189.             if(!$pref->;{"send"} || $name !~ /$pref->;{"send"}/) { next; }
  190.             if($pref->;{"dontsend"} && $name =~ /$pref->;{"dontsend"}/) {
  191.                 next;
  192.             }
  193.             my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$mtime)
  194.                 = stat($name);
  195.             $files{"$name"} = $mtime;
  196.         } elsif(-d $name) {
  197.             if(!$pref->;{"descend"} || $name !~ /$pref->;{"descend"}/) {
  198.                 next;
  199.             }
  200.             if($pref->;{"dontdescend"} &&
  201.                $name =~ /$pref->;{"dontdescend"}/) {
  202.                 next;
  203.             }
  204.             $dirs{"$name"} = 1;
  205.         }
  206.     }
  207.     closedir(CD);
  208.     return (\%files, \%dirs);
  209. }

  210. # Delete all contents of the current remote directory, with the given name.
  211. sub delrmt {
  212.     my $conn = shift @_;
  213.     my $name = shift @_;

  214.     # Get the contents.
  215.     my ($files, $dirs) = rmt_contents($conn);

  216.     # Get rid of the files.
  217.     foreach my $fi(keys %$files) {
  218.         print "Deleting $name$fi.\n";
  219.         if(!$norun) {
  220.             $conn->;delete($fi) or
  221.                 expire($conn, "Delete $name$fi failed: !!", 0);
  222.         }
  223.     }

  224.     # Get rid of the directories.
  225.     foreach my $d(keys %$dirs) {
  226.         $conn->;cwd($d) or expire($conn, "cd $name$d failed: !!");
  227.         delrmt($conn, "$name$d/");
  228.         $conn->;cdup() or expire($conn, "cd up failed: !!");
  229.         print "Removing directory $name$d.\n";
  230.         if(!$norun) {
  231.             $conn->;rmdir($d) or expire($conn, "rmdir $name$d failed: !!", 0);
  232.         }
  233.     }
  234. }

  235. # Synchronize.
  236. sub sync {
  237.     my $conn = shift @_;
  238.     my $name = shift @_;
  239.     my $fn = shift @_;
  240.     my $strings = shift @_;
  241.     my $pats = shift @_;

  242.     # Get the contents.
  243.     my ($rfiles, $rdirs) = rmt_contents($conn);
  244.     my ($lfiles, $ldirs) = loc_contents($conn,$strings,$pats);

  245.     #print "FRED $name:\n"; prfl($lfiles);
  246.     #print "\n";
  247.     #prfl($ldirs);
  248.     #print "\n";

  249.     # Go through the local files and put what needs putting.
  250.     foreach my $lf(keys %$lfiles) {
  251.         # See if it must be sent.
  252.         if(!exists $rfiles->;{$lf} || $rfiles->;{$lf} < $lfiles->;{$lf}) {
  253.             # Old or missing....
  254.             print "Sending $name$lf\n";
  255.             if(!$norun) {
  256.                 if(!-r $lf) {
  257.                     expire($conn, "File $lf is not readable", 0);
  258.                 } else {
  259.                     $conn->;put($lf) or
  260.                         expire($conn, "Put $name$lf failed: !!", 0);
  261.                 }
  262.             }
  263.         }
  264.         delete $rfiles->;{$lf};
  265.     }

  266.     # If there are any remote files left, delete them.
  267.     foreach my $rf(keys %$rfiles) {
  268.         next if ($pats->;{"preserve"} && ($rf =~ /$pats->;{"preserve"}/));
  269.         print "Deleting $name$rf.\n";
  270.         if(!$norun) {
  271.             $conn->;delete($rf) or
  272.                 expire($conn, "Delete $name$rf failed: !!", 0);
  273.         }
  274.     }

  275.     # Recur on the local directory names.
  276.     foreach my $ld(keys %$ldirs) {
  277.         # Create if needed.
  278.         if(!$rdirs->;{$ld}) {
  279.             print "Creating $name$ld\n";
  280.             if($norun) {
  281.                 print "Sending $name$ld subtree.\n";
  282.                 next;
  283.             } else {
  284.                 if(!$conn->;mkdir($ld)) {
  285.                     expire($conn, "Create $name$ld failed: !!", 0);
  286.                     next;
  287.                 }
  288.             }
  289.         }

  290.         # Perform the recursion.
  291.         $conn->;cwd($ld) or expire($conn, "cd $name$ld failed: !!");
  292.         chdir($ld) or expire($conn, "local cd $ld failed: $!");
  293.         if(-r $fn) {
  294.             my %nstrings = %$strings;
  295.             my %npats = %$pats;
  296.             readctl($conn, $fn, 0, \%nstrings, \%npats);
  297.             sync($conn, "$name$ld/", $fn, \%nstrings, \%npats);
  298.         } else {
  299.             sync($conn, "$name$ld/", $fn, $strings, $pats);
  300.         }
  301.         $conn->;cdup() or expire($conn, "cd up failed: !!");
  302.         chdir('..') or expire($conn, "local cd .. failed: $!");

  303.         delete $rdirs->;{$ld};
  304.     }

  305.     # Wipe remotes not matched locally.
  306.     foreach my $rd(keys %$rdirs) {
  307.         next if($pats->;{"presdir"} && ($rd =~ /$pats->;{"presdir"}/));
  308.         $conn->;cwd($rd) or expire($conn, "cd $name$rd failed: !!");
  309.         delrmt($conn, "$name$rd/");
  310.         $conn->;cdup() or expire($conn, "cd up failed: !!");
  311.         print "Removing directory $name$rd.\n";
  312.         if(!$norun) {
  313.             $conn->;rmdir($rd) or
  314.                 expire($conn, "Removal of $name$rd failed: !!", 0);
  315.         }
  316.     }
  317. }

  318. # Print the file times map.
  319. sub prfl {
  320.     my $hr = shift @_;

  321.     foreach my $k(sort keys %$hr) {
  322.         my $mod = localtime($hr->;{$k});
  323.         print "$k->;$mod\n";
  324.     }
  325. }

  326. # Synchronize clocks.
  327. sub clocksync {
  328.     my $conn = shift @_;
  329.     my $fn = shift @_;

  330.     if(! -f $fn) {
  331.         open(SF, ">;$fn") or
  332.             expire($conn, "Cannot create $fn for time sync option");
  333.         close(SF);
  334.     }
  335.     -z $fn or
  336.         expire($conn, "File $fn for time sync must be empty.");

  337.     $conn->;put($fn) or
  338.         expire($conn, "$fn send failed: !!");

  339.     my $now_here = time();
  340.     my $now_there = $conn->;mdtm($fn) or
  341.         expire($conn, "Cannot get $fn write time");

  342.     #print "FRED: $now_here $now_there\n";
  343.     #print "FRED: ", localtime($now_here), " ", localtime($now_there), "\n";

  344.     $syncoff = $now_here - $now_there;
  345.     $syncoff -= 5; # Be a bit conservative.

  346.     #print "A: [$now_here] [$now_there] [$syncoff]\n";

  347.     $conn->;delete($fn);
  348.    
  349.     my $hrs = int($syncoff/3600);
  350.     my $mins = int($syncoff/60) - $hrs*60;
  351.     my $secs = $syncoff - $hrs*3600 - $mins*60;
  352.     printf("Clock sync offset: %d:%02d:%02d\n", $hrs, $mins, $secs);
  353. }

  354. # Check for login info.
  355. $strings{"host"} && $strings{"acct"} && $strings{"pwd"} or
  356.     die "Hostname, account name, and password must be specified.\n";

  357. # Do the communications.
  358. my $conn = Net::FTP->;new($strings{"host"}, Passive =>; 1) or
  359.     die "Connect: $@\n";
  360. $conn->;login($strings{"acct"}, $strings{"pwd"}) or
  361.     expire($conn, "Login as $strings{'acct'} failed: !!");
  362. if($strings{"cd"}) {
  363.     $conn->;cwd($strings{"cd"}) or
  364.         expire($conn, "Initial directory change failed: !!");
  365. }
  366. $conn->;binary() or
  367.     expire($conn, "Binary mode failed: !!");

  368. # Optional clock synchronization step.
  369. if($strings{"sync"}) { clocksync($conn, $strings{"sync"}); }

  370. #my ($a, $b) = rmt_contents($conn);
  371. #print "files:\n";
  372. #prfl($a);
  373. #print "dirs:\n";
  374. #prfl($b);

  375. # Now the real work.
  376. sync($conn, '', $rcfile, \%strings, \%pats);

  377. $conn->;quit();

  378. exit $was_err;
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP