在shell中使用ftp

我经常用ftp从一台电脑传输一个文件到另一台电脑。有时我用命令行交互模式执行操作,但更多时候,我喜欢写一个脚本来做这件苦差事。这个任务着实使我蒙逼了一阵子,但我最终解决了那些坑。我没有见过这方面有关的文档,所以我写了这个供你参考。

问题所在

我在过程中遇到的关键问题是,你要提供一个密码给ftp服务器.典型的ftp客户端(比如unix下的,linux下的,solaris下的,NetBSD下的)都从/dev/tty读取密码.

(non-working) 脚本例子

#!/bin/sh
HOST='ftp.users.qwest.net'
USER='yourid'
PASSWD='yourpw'
FILE='file.txt'

ftp $HOST <

上面的脚本并不会完成工作,无论是执行在前景还是作为定时任务执行在背景,都不会工作.

/dev/tty是一种奇怪的设备.每一个进程(精确的说是每一个进程组)有不同的/dev/tty,你不能天真地让ftp客户端从某个文件中(比如here document)读取密码.当在终端执行时,上面的脚本会挂着,因为它从/dev/tty读取密码,终端构成了脚本的/dev/tty,所以脚本等着键盘输入.

(working) 脚本例子

#!/bin/sh
HOST='ftp.users.qwest.net'
USER='yourid'
PASSWD='yourpw'
FILE='file.txt'

ftp -n $HOST <

诀窍所在

不需要ftp客户端从/dev/tty读取密码需要耍两个把戏:

  1. 使用 -n 选项连接ftp客户端以阻止ftp客户端立即登录.这样,ftp客户端就不要你输入一个用户名和密码了.所以没有用到/dev/tty
  2. 使用ftp客户端命令quote发送用户名和密码.

    你必须在开头和结尾加上END_SCRIPT,尽管命令已经缩进了,还是要加上.

进一步改良

上面的脚本会从ftp客户端涌出许多输出到标准输出,即便它运行良好。运行上面的脚本会立刻显示许多乱七八糟的内容给用户,改良一下的方法是把输出输到其他的地方去,越远越好:

ftp -n $HOST > /tmp/ftp.worked 2> /tmp/ftp.failed <

还可以改良一下处理错误,就是处理ftp客户端的退出状态.

ftp -n $HOST > /tmp/ftp.worked 2> /tmp/ftp.failed <

then里面的代码一般不会执行,大多数时候ftp客户端都会返回状态0.这只会给你没用的信息:文件传输失败了,但脚本也不能查出问题.

下面的方法可以证实文件是否传输成功:

#!/bin/sh
ftp -n << END_SCRIPT
open $1
user $2 $3
put $4
get $4 retrieval.$$
bye
END_SCRIPT
if [ -f retrieval.$$ ]
then
    echo "FTP of $4 to $1 worked"
    rm -f retrieval.$$
else
    echo "FTP of $4 did not work"
fi

不合格的传输会留下文件.

代替脚本

我在usenet 文章上看到另一种方法:

#!/bin/sh
USER=userid
PASSWD=userpw
ftp -n f2dev <
                    
                    

你可能感兴趣的:(在shell中使用ftp)