https://www.muzilong.cn/article/117php
開發環境linux
環境:lnmp下進行試驗。laravel
框架:laravel5swoole
問題描述app
這幾天作銀行對賬接口時,踩了一個坑,具體需求大體描述一下。框架
銀行天天凌晨後,會開始準備昨天的交易流水數據,須要咱們這邊請求拿到。測試
由於他們給的是一個base64加密的zip壓縮流,解開之後能夠獲得txt文件,裏面就是咱們須要的數據了。this
業務程序寫好之後,隨手丟了一個定時任務就去睡覺了。加密
哪知道次日上班的時候,檢查。發現並無拿到數據,查詢一下日誌的時候發現,凌晨服務端請求的時候,銀行接口返回了:系統錯誤信息。spa
諮詢銀行那邊後,銀行那邊相關人員建議咱們多請求幾回,可是在屢次請求中,我發現銀行那邊是有頻率限制的,最後得知,此接口只能半個小時才能請求一次。這就比較尷尬了,由於我不知道銀行那邊何時能返回數據給我。
因而這個問題怎麼解決呢?理想的狀況是,服務端請求數據,銀行那邊沒有返回。而後程序等半個小時後,再請求一次,這樣一直到銀行那邊返回正確的數據停止。
問題分析
這個功能換道別的語言也許不難,可是經過php實現的話,那就比較麻煩了。一般的話,咱們能夠搭配linux下的cron來實現,好比咱們能夠在凌晨到6:00之間作一個定時任務,每半個小時掃描一次php腳本,若是發現銀行那邊的狀態依舊爲失敗的話,咱們就執行一次php腳本去請求數據。直到請求到正確的數據,而後把狀態更新爲成功。
這不失爲一種方法,但太傻了。好比說銀行那邊比較正常,凌晨,也就是第一次請求的時候,就已經返回了正確的數據,那麼咱們的cron腳本還傻傻的每一個半個小時執行一次,好蠢!~
或者咱們能夠嘗試使用linux下的at命令,但感受仍是不夠優雅。
解決問題
因而決定給laravel擴展一個swoole插件來解決此問題,swoole的定時任務很完美的解決了咱們目前的問題。
首先咱們須要把swoole擴展安裝好,具體過程略。
裝好之後,咱們寫一個swoole簡易的服務端測試腳本,注意,此腳本是放在app/Console/Commands/下的,筆者是放在了app/Console/Commands/Swoole/swoole.php下,具體代碼爲
<?php namespace App\Console\Commands\Swoole; use Illuminate\Console\Command; class swoole extends Command { /** * The name and signature of the console command. * * @var string */ protected $signature = 'swoole {action}'; /** * The console command description. * * @var string */ protected $description = "Let's use swoole !"; private $serv; /** * Create a new command instance. * * @return void */ public function __construct() { parent::__construct(); } /** * Execute the console command. * * @return mixed */ public function handle() { $arg = $this->argument('action'); switch ($arg) { case 'start': $this->info('swoole server started'); $this->start(); break; case 'stop': $this->info('stoped'); $this->stop(); break; case 'restart': $this->info('restarted'); break; } } private function start() { $this->serv = new \swoole_server("127.0.0.1", 9501); $this->serv->set(array( 'worker_num' => 8, 'daemonize' => false, 'max_request' => 10000, 'dispatch_mode' => 2, 'task_worker_num' => 8, 'task_ipc_mode' => 3, 'log_file' => storage_path('logs/taskqueue.log'), )); $this->serv->on('Receive', array($this, 'onReceive')); $this->serv->on('Task', array($this, 'onTask')); $this->serv->on('Finish', array($this, 'onFinish')); $this->serv->start(); } public function onReceive(\swoole_server $serv, $fd, $from_id, $data) { $serv->task($data); } public function onTask($serv, $task_id, $from_id, $data) { $timeon = (3) * 1000; if ($timeon > 0) { $serv->after($timeon, function () { //業務邏輯處 exec('php /path/to/root/artisan Test:Command'); }); } return date('Y-m-d H:i:s') . "第一次執行"; } public function onFinish($serv, $task_id, $data) { echo "Task finish\n"; } private function stop() { exec('/usr/bin/killall php'); } }
這是服務端,咱們主要用到了after方法,模擬的話,是三秒一執行。實際應該是三十分鐘
而後咱們隨便寫一個客戶端鏈接類
<?php /** * Created by PhpStorm. * User: nosay * Date: 4/13/18 * Time: 9:27 PM */ namespace App\Extension\php\Swoole; class swoole{ private $data; private $client; public function __construct($data){ $this->data = $data; $this->client = new \swoole_client(SWOOLE_SOCK_TCP); } public function connect(){ if( !$this->client->connect("127.0.0.1", 9501 , 1) ) { echo "Error"; } $this->client->send($this->data); } }
因而咱們在銀行腳本中就能夠去執行了
<?php namespace App\Console\Commands\Test; use App\Extension\php\Swoole\swoole; use Illuminate\Console\Command; class TestCommand extends Command { /** * The name and signature of the console command. * * @var string */ protected $signature = 'Test:Command'; /** * The console command description. * * @var string */ protected $description = 'Command Test'; /** * Create a new command instance. * * @return void */ public function __construct() { parent::__construct(); } /** * Execute the console command. * * @return mixed */ public function handle() { //這裏是業務邏輯 //若是銀行那邊返回的爲false的話,那麼咱們把他交給swoole的定時腳本 $status = false; if(!$status) { $swoole = new swoole("hehe"); $swoole->connect(); } } }
https://mp.weixin.qq.com/s/e_ophgURO6qRvfKd3YDJXQ