在 Laravel 调度器中执行长的多 Bash 命令

Posted

技术标签:

【中文标题】在 Laravel 调度器中执行长的多 Bash 命令【英文标题】:Execute Long Multi Bash Commands in Laravel Scheduler 【发布时间】:2021-03-16 21:54:37 【问题描述】:

我每天都在尝试通过 Laravel 调度程序从我的服务器上清除敏感的历史命令

当我运行这段代码时:

$schedule->exec(`for h in $(history | grep mysql | awk 'print $1-N' | tac) ; do history -d $h ; done; history -d $(history | tail -1 | awk 'print $1') ; history`)->$interval()->emailOutputTo($email)->environments('prod');

我收到此错误:

⚡️  app2020  php artisan schedule:run                                                                                                             
                                                                                                                                                   
   ErrorException                                                                                                                                  
                                                                                                                                                   
  Undefined variable: h                                                                                                                            
                                                                                                                                                   
  at app/Console/Kernel.php:44

注意到这是一个完美的工作命令

for h in $(history | grep mysql | awk 'print $1-N' | tac) ; do history -d $h ; done; history -d $(history | tail -1 | awk 'print $1') ; history

注意: 我知道我可以通过将上面的命令添加到 crontab 中来实现它,但目标是让所有 cron 活动都组织在 Laravel 项目中

既然 bash 和 PHP 使用 $ 作为变量,那么如何使类似的 bash 命令工作?

任何提示,我都会接受。

【问题讨论】:

【参考方案1】:

您将命令包装在反引号中,这将插入变量并将值作为命令执行。这些东西你都不想要。

使用单引号构建命令,然后将其传递给exec()

$cmd =  'for h in $(history | grep mysql | awk \'print $1-N\' | tac) ;';
$cmd .= '    do history -d $h ;';
$cmd .= 'done;';
$cmd .= 'history -d $(history | tail -1 | awk \'print $1\') ;';
$cmd .= 'history';

$schedule
    ->exec($cmd)
    ->$interval()
    ->emailOutputTo($email)
    ->environments('prod');

对于更有效的方法,您可以尝试直接编辑历史文件。

$cmd = 'sed -i /mysql/d "$HOME/.bash_history"';

无需删除文件中的最后一个条目,因为它是交互式 shell 命令的历史记录。

显然最好的办法是不要将“敏感”的东西放在命令行中。对于 MySQL,这可以通过将凭据添加到 .my.cnf 文件来完成。

【讨论】:

以上是关于在 Laravel 调度器中执行长的多 Bash 命令的主要内容,如果未能解决你的问题,请参考以下文章

Java中的多线程调度

laravel 任务调度(定时执行任务)

Laravel 调度器:每秒执行一个命令

bash的多命令执行,shell脚本基础及grep系

Laravel 计划任务(任务调度)的使用

Laravel 计划任务(任务调度)的使用