我创建了一个向多个用户发送电子邮件的应用程序,但在处理大量收件人时遇到了一个问题。
错误出现在 failed_jobs
table
Illuminate\Queue\MaxAttemptsExceededException: App\Jobs\ESender has been attempted too many times or run too long. The job may have previously timed out. in D:\EmailSender\vendor\laravel\framework\src\Illuminate\Queue\Worker.php:649
这就是 payload
在 failed_jobs
table
{"uuid":"ff988083-c1da-4d20-a2e3-c2a10e154c79","timeout":9000,"id":"j2Lz0Ro0bkJpqwxKWTxC3Tiii71iE6Cm","data":{"command":"O:16:\"App\\Jobs\\ESender\":13:{s:7:\"timeout\";i:9000;s:12:\"receiver_obj\";O:45:\"Illuminate\\Contracts\\Database\\ModelIdentifier\":4:{s:5:\"class\";s:12:\"App\\Receiver\";s:2:\"id\";i:6;s:9:\"relations\";a:0:{}s:10:\"connection\";s:5:\"mysql\";}s:16:\"sender_all_hosts\";O:45:\"Illuminate\\Contracts\\Database\\ModelIdentifier\":4:{s:5:\"class\";s:15:\"App\\SenderHosts\";s:2:\"id\";a:4:{i:0;i:1;i:1;i:2;i:2;i:3;i:3;i:4;}s:9:\"relations\";a:0:{}s:10:\"connection\";s:5:\"mysql\";}s:11:\"message_obj\";O:45:\"Illuminate\\Contracts\\Database\\ModelIdentifier\":4:{s:5:\"class\";s:12:\"App\\Messages\";s:2:\"id\";i:36;s:9:\"relations\";a:0:{}s:10:\"connection\";s:5:\"mysql\";}s:7:\"counter\";i:1;s:3:\"job\";N;s:10:\"connection\";N;s:5:\"queue\";N;s:15:\"chainConnection\";N;s:10:\"chainQueue\";N;s:5:\"delay\";N;s:10:\"middleware\";a:0:{}s:7:\"chained\";a:0:{}}","commandName":"App\\Jobs\\ESender"},"displayName":"App\\Jobs\\ESender","timeoutAt":1594841911,"maxExceptions":null,"maxTries":null,"job":"Illuminate\\Queue\\CallQueuedHandler@call","delay":null,"attempts":1}
请参见此处的cmd错误。
代码部分:
1
class ESender implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
/**
* The number of times the job may be attempted.
*
* @var int
*/
public $tries = 100;
/**
* The number of seconds the job can run before timing out.
*
* @var int
*/
public $timeout = 9999999;
...more code...
}
2
public function handle(){
Redis::throttle('key')->allow(1)->every(20)->then(function () {
//send email
..... more code .....
}, function () {
// Could not obtain lock...
return $this->release(10);
});
}
这是我的配置: queue.php
:
'redis' => [
'driver' => 'redis',
'connection' => 'default',
'queue' => env('REDIS_QUEUE', 'default'),
'retry_after' => 9000,
'block_for' => null,
],
``` `.env` ```
BROADCAST_DRIVER=log
CACHE_DRIVER=file
QUEUE_CONNECTION=database
SESSION_DRIVER=file
SESSION_LIFETIME=300
REDIS_CLIENT = predis
REDIS_HOST=127.0.0.1
REDIS_PASSWORD=null
REDIS_PORT=6379
QUEUE_DRIVER=database
2条答案
按热度按时间raogr8fs1#
你设置了一个
timeout
但此超时值大于retry_after
在这个配置文件中定义的。看到了吗https://laravel.com/docs/7.x/queues#job-过期和超时
有一个明确的警告:
这个 --超时 值应始终至少比 重试\u之后 配置值。这将确保处理给定作业的worker总是在重试作业之前被终止。如果你的 --超时 选项比您的 重试\u之后 配置值时,您的作业可能会被处理两次。
您可以为长时间运行的作业定义一个新连接,并在作业上设置此连接(分派到特定连接),而不是使用
timeout
.zvokhttg2#
运行队列工作程序的命令需要--tries=和--timeout=设置队列工作程序允许的输出限制。
这可以确保您的命令不能超出定义的worker的限制。
您可以使用作业属性来实现超时或重试,如下所示。并使用队列配置文件设置默认值。