使用 Laravel Horizon 优雅的终止进程(2)

简介: 通过调研之后发现,laravel 的 horizon 扩展可以解决这个问题,使用下面的命令可以优雅的结束进程:确保进行中进程不会被kill掉,执行结束后才允许被kill;非进行的任务等候,不会加入到队列中;

抛出问题


我们发布新版本代码时,如果优雅的终止运行中的异步任务,规避异步任务运行到一半被kill掉的情况。


解决办法


通过调研之后发现,laravel 的 horizon 扩展可以解决这个问题,使用下面的命令可以优雅的结束进程:


  1. 确保进行中进程不会被kill掉,执行结束后才允许被kill;
  2. 非进行的任务等候,不会加入到队列中;


优雅的解决了这个问题,思路就是这么的朴实无华。


php artisan horizon:terminate


下面继续介绍 Horizon 进阶知识点


基础知识点可以看这篇:使用 Laravel Horizon 优雅的终止进程(1)


运行 Horizon


当在 config/horizon.php 文件中配置好队列执行进程后,就可以使用 horizon Artisan 命令启动 Horizon。

只需要一条命令语句即可启动所有配置好的队列进程:


php artisan horizon


使用 horizon:pause 和 horizon:continue Artisan 命令来暂停或继续执行队列任务:


php artisan horizon:pause
php artisan horizon:continue


使用 horizon:terminate Artisan 命令优雅的终止 Horizon 主进程。


Horizon 会把正在执行的任务处理完毕后退出:


php artisan horizon:terminate


部署 Horizon


将 Horizon 部署到线上服务器时,则需要配置一个进程监控器来检测 php artisan horizon 命令,在它意外退出时自动重启。

上线新代码时则需要该进程监控器终止 Horizon 进程并以修改后的代码重启 Horizon。


下面的内容比较重要,请集中注意力:


Supervisor 配置


如果使用 Supervisor 进程监控器管理 horizon 进程,那么以下配置文件则可满足需求:


[program:horizon]
process_name=%(program_name)s
command=php /home/forge/app.com/artisan horizon
autostart=true
autorestart=true
user=forge
redirect_stderr=true
stdout_logfile=/home/forge/app.com/horizon.log


标签


Horizon 允许你对任务分配「标签」,包括邮件,事件广播,通知和队列的事件监听器。


优雅如Laravel,Horizon 会智能并且自动根据任务携带 Eloquent 模型给大多数任务标记标签,如下任务示例:


<?php
namespace App\Jobs;
use App\Video;
use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
class RenderVideo implements ShouldQueue
{
    use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
    /**
     * video 实例
     *
     * @var \App\Video
     */
    public $video;
    /**
     * 创建工作实例
     *
     * @param  \App\Video  $video
     * @return void
     */
    public function __construct(Video $video)
    {
        $this->video = $video;
    }
    /**
     * 执行任务
     *
     * @return void
     */
    public function handle()
    {
        //
    }
}


如果该队列任务是一个携带 id 为 1 的 App\Video 实例,那么它将自动被标记上 App\Video:1 标签。


因为 Horizon 会检查任务属性是否具有 Eloquent 模型,如果发现 Eloquent 模型,Horizon 将会智能的用该模型的类名和主键为任务标记上标签:


$video = App\Video::find(1);
App\Jobs\RenderVideo::dispatch($video);


再一次发出感慨:真的优雅!


自定义标签


需要手动的为队列执行的对象定义标签,可以给这个类定义一个 tags 方法:


class RenderVideo implements ShouldQueue
{
    /**
     * 获取分配给这个任务的标签
     *
     * @return array
     */
    public function tags()
    {
        return ['render', 'video:'.$this->video->id];
    }
}


通知


如果需要在队列等待时间过长时发起通知,可以在应用的 HorizonServiceProvider 中调用以下三种方法:


Horizon::routeMailNotificationsTo

Horizon::routeSlackNotificationsTo

Horizon::routeSmsNotificationsTo


Horizon::routeMailNotificationsTo('example@example.com');
//这个不常用,起码我没用过。
Horizon::routeSlackNotificationsTo('slack-webhook-url', '#channel');
Horizon::routeSmsNotificationsTo('138xxxxxxxx');


题外话:Slack是什么?


百度百科的解释:Slack 是聊天群组 + 大规模工具集成 + 文件整合 + 统一搜索。截至2014年底,Slack 已经整合了电子邮件、短信、Google Drives、Twitter、Trello、Asana、GitHub 等 65 种工具和服务,可以把各种碎片化的企业沟通和协作集中到一起。


配置等待时间过长的阈值


可以在 config/horizon.php 配置文件中设置等待时间过长的具体秒数。


waits 配置项可以针对每一个 链接 / 队列 配置阈值:


'waits' => [
    'redis:default' => 60, //单位秒
],


Metrics


Horizon 包含一个 Metrics 仪表盘,它可以提供任务和队列等待时间和吞吐量信息。

为了填充此仪表盘,需要通过应用的任务调度器配置 Horizon 的 snapshot Artisan 命令每五分钟运行一次。


/**
 * 定义应用程序的任务调度
 *
 * @param  \Illuminate\Console\Scheduling\Schedule  $schedule
 * @return void
 */
protected function schedule(Schedule $schedule)
{
    $schedule->command('horizon:snapshot')->everyFiveMinutes();
}


相关文章
|
2月前
|
监控 Linux 应用服务中间件
探索Linux中的`ps`命令:进程监控与分析的利器
探索Linux中的`ps`命令:进程监控与分析的利器
|
28天前
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
|
23天前
|
弹性计算 Linux 区块链
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
30 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
|
18天前
|
算法 Linux 调度
探索进程调度:Linux内核中的完全公平调度器
【8月更文挑战第2天】在操作系统的心脏——内核中,进程调度算法扮演着至关重要的角色。本文将深入探讨Linux内核中的完全公平调度器(Completely Fair Scheduler, CFS),一个旨在提供公平时间分配给所有进程的调度器。我们将通过代码示例,理解CFS如何管理运行队列、选择下一个运行进程以及如何对实时负载进行响应。文章将揭示CFS的设计哲学,并展示其如何在现代多任务计算环境中实现高效的资源分配。
|
1月前
|
存储 缓存 安全
【Linux】冯诺依曼体系结构与操作系统及其进程
【Linux】冯诺依曼体系结构与操作系统及其进程
132 1
|
1月前
|
小程序 Linux
【编程小实验】利用Linux fork()与文件I/O:父进程与子进程协同实现高效cp命令(前半文件与后半文件并行复制)
这个小程序是在文件IO的基础上去结合父子进程的一个使用,利用父子进程相互独立的特点实现对数据不同的操作
|
1月前
|
SQL 自然语言处理 网络协议
【Linux开发实战指南】基于TCP、进程数据结构与SQL数据库:构建在线云词典系统(含注册、登录、查询、历史记录管理功能及源码分享)
TCP(Transmission Control Protocol)连接是互联网上最常用的一种面向连接、可靠的、基于字节流的传输层通信协议。建立TCP连接需要经过著名的“三次握手”过程: 1. SYN(同步序列编号):客户端发送一个SYN包给服务器,并进入SYN_SEND状态,等待服务器确认。 2. SYN-ACK:服务器收到SYN包后,回应一个SYN-ACK(SYN+ACKnowledgment)包,告诉客户端其接收到了请求,并同意建立连接,此时服务器进入SYN_RECV状态。 3. ACK(确认字符):客户端收到服务器的SYN-ACK包后,发送一个ACK包给服务器,确认收到了服务器的确
158 1
|
2月前
|
Web App开发 运维 监控
深入探索Linux命令pwdx:揭秘进程工作目录的秘密
`pwdx`命令在Linux中用于显示指定进程的工作目录,基于`/proc`文件系统获取实时信息。简单易用,如`pwdx 1234`显示PID为1234的进程目录。结合`ps`和`pgrep`等命令可扩展使用,如查看所有进程或特定进程(如Firefox)的目录。使用时注意权限、进程ID的有效性和与其他命令的配合。查阅`man pwdx`获取更多帮助。
|
2月前
|
存储 Shell Linux
Linux进程概念(下)
本文详细的介绍了环境变量和进程空间的概念及其相关的知识。
27 0
Linux进程概念(下)
|
1月前
|
缓存 Linux 编译器
【Linux】多线程——线程概念|进程VS线程|线程控制(下)
【Linux】多线程——线程概念|进程VS线程|线程控制(下)
47 0