0

我正在使用 Linux 本地计算机,需要定期备份/镜像一些非常大的文件结构。我只能访问 SFTP。

我追求简单的一键式解决方案。我最初尝试在 BASH 中编写这个小脚本,但我以前从未使用过它,并且对语法不熟悉,所以我求助于 PHP。(我知道 PHP 不是为这种工作而设计的,但我的时间很紧,没有时间进入 BASH atm)

<?php
//init
parse_str(implode('&', array_slice($argv, 1)), $_GET);
$error = array();

$lPrefix         = '/home/hozza/Sites/';
$archiveLocation = '/home/hozza/Backups/';

$lDir = isset($_GET['l']) ? $_GET['l'] : $error[] = 'Local Directory Required';
$rDir = isset($_GET['r']) ? $_GET['r'] : $error[] = 'Remote Directory Required';
$bookmark = isset($_GET['b']) ? $_GET['b'] : $error[] = 'lftp Bookmark Required';

//Check for args
if(count($error) == 0) {
    $archiveName = end(explode('/', $lDir)) . '_' . date('Y-m-d_H-i');

    //Validate local dir
    if(is_dir($lPrefix . $lDir)) {

        //preserve Sublime Text 2 config SFTP files
        $ST2_SFTP_conf = false;
        if(file_exists($lPrefix . $lDir . '/sftp-config.json')) {
            $ST2_SFTP_conf = file_get_contents($lPrefix . $lDir . '/sftp-config.json');
            unlink($lPrefix . $lDir . '/sftp-config.json');
        }

        //Start mirror
        $lftOutput = explode("\n", shell_exec('lftp -e "mirror -e -p --parallel=10 --log=' . $archiveLocation . 'logs/' . $archiveName . '.txt ' . $rDir . '/ ' . $lPrefix . $lDir . '/; exit top" ' . $bookmark));

        //Tar regardless of lftp error or success
        $tarOutput = shell_exec('cd ' . $lPrefix . ' && tar -czf ' . $archiveLocation . $archiveName . '.tar.gz ' . $lDir);

        //Output completion or errors
        shell_exec('notify-send -i gnome-network-properties -t 0 "Mirror & Archive Complete" "' . $archiveName . '\n\n' . implode('\n', $lftOutput) . $tarOutput . '"');

        //put back ST2 SFTP conf
        if($ST2_SFTP_conf != false) file_put_contents($lPrefix . $lDir . '/sftp-config.json', $ST2_SFTP_conf);

        exit;
    }
    else shell_exec('notify-send -i error -t 0 "Mirror & Archive Error" "' . date('Y-m-d') . ' ' . date('H-i') . '\n' . $lDir . ' \n Does not exist! D:"');
}
else shell_exec('notify-send -i error -t 0 "Mirror & Archive Error" "' . date('Y-m-d') . ' ' . date('H-i') . '\n' . implode('\n', $error) . '"');
?>

它可以通过这样的快捷方式在许多站点上运行......

terminator -T "Mirror & Archive" -e "php ~/Programs/mirror.php l=local-dir_path r=./ b=lftp-bookmark-name"

如果 LFTP 书签中没有密码(不应该以纯文本形式存储),终端会提示输入密码,在脚本运行后,会给出一个很好的通知,其中包含有关文件/文件夹/速度等的一些信息。

但是,当脚本在终端中运行时,只有“输入密码”位输出到终端,我希望终端中显示所有输出(通常会显示当前正在使用的文件/文件夹等)

有谁知道该怎么做?

4

2 回答 2

3

IIRC 您看到密码提示输出到终端的原因是它正在使用stderr. 您可以尝试重定向stdoutstderr您的命令,这应该向您显示“实时”进度。将其添加到命令的末尾shell_exec()1>&2

IE:

shell_exec('lftp -e "mirror -e -p --parallel=10 --log=' . $archiveLocation . 'logs/' . $archiveName . '.txt ' . $rDir . '/ ' . $lPrefix . $lDir . '/; exit top" ' . $bookmark . ' 1>&2')

但是,这将阻止您shell_exec出于日志记录目的而返回任何内容。我的建议是:

$log_stem = '/tmp/' . time() . '_';  // ie: /tmp/1357581737_
$lfOutfile = $log_stem . 'lftp.log';
$tarOutfile = $log_stem . 'tar.log';

shell_exec('lftp -blah | tee ' . $lfOutfile ' 1>&2' );
shell_exec('tar -blah | tee ' . $tarOutfile ' 1>&2' );

$lfOut = file_get_contents($lfOutfile);
$tarOut = file_get_contetns(tarOutfile);

// remove tmp files
unlink($lfOutfile);
unlink($tarOutfile);

这将在将输出重定向到 stderr 之前将输出的副本捕获到文件中,以便您可以实时观看。

但是,如果您想通过 cron 运行它,我建议不要将任何非错误的内容写入 stderr,否则 cron 将在每次运行时发送警告电子邮件。

于 2013-01-07T18:14:57.723 回答
0

我认为最后一个答案很接近:

要么:

shell_exec('lftp -blah |& tee ' . $lfOutfile );
shell_exec('tar -blah |& tee ' . $tarOutfile );

或者如果这仍然不起作用,试试这个:

shell_exec('lftp -blah 2>&1 | tee ' . $lfOutfile );
shell_exec('tar -blah 2>&1 | tee ' . $tarOutfile );
于 2013-03-10T22:23:16.360 回答