Laravel测试工作被释放

时间:2016-02-06 18:03:34

标签: laravel testing jobs task-queue

我想测试在某些情况下作业已被释放回队列。

这是我的工作班:

class ChargeOrder extends Job
{
    use InteractsWithQueue, SerializesModels;

    /**
     * The order model which is to be charged
     */
    protected $order;

    /**
     * The token or card_id which allows us to take payment
     */
    protected $source;

    /**
     * Create a new job instance.
     *
     * @param   App\Order       $order;
     * @param   string          $source;
     * @return  array
     */
    public function __construct($order, $source)
    {
        $this->order        = $order;
        $this->source       = $source;
    }

    /**
     * Execute the job.
     *
     * @return void
     */
    public function handle(Charge $charge)
    {
        $result             = $charge->execute($this->source, $this->order->totalInclVat());
        $exception_errors   = config('payment.errors.exception_errors');

        //  If we have an error that isn't caused by the user (anything but a card error)
        //  We're going to notify ourselves via slack so we can investigate.
        if (array_key_exists('error', $result) && in_array($result['error']['code'], array_keys(config('payment.errors.other_error'))))
        {
            $client         = new Client(config('services.slack.channels.payment_errors.url'), config('services.slack.channels.payment_errors.settings'));
            $client->send(app()->environment() . ": " . $result['error']['code']);
        }

        //  If the error is in the list of errors that throw an exception, then throw it.
        if (array_key_exists('error', $result) && (in_array($result['error']['type'], $exception_errors) || in_array($result['error']['code'], $exception_errors)))
        {
            $status_code    = config('payment.errors')[$result['error']['type']][$result['error']['code']]['status_code'];
            $message        = config('payment.errors')[$result['error']['type']][$result['error']['code']]['message'];

            throw new BillingErrorException($status_code, $message);
        }

        //  If we still have an error, then it something out of the user's control.
        //  This could be a network error, or an error with the payment system
        //  Therefore, we're going to throw this job back onto the queue so it can be processed later.
        if (array_key_exists('error', $result) && in_array($result['error']['code'], array_keys(config('payment.errors.other_error'))))
        {
            $this->release(60);
        }
    }
}

我需要测试" $ this-> release(60)"在某些情况下被称为。

我试图在我的测试中嘲笑工作合同:

// Set Up
$this->job  = Mockery::mock('Illuminate\Contracts\Queue\Job');
$this->app->instance('Illuminate\Contracts\Queue\Job', $this->job);

然后

// During Test
$this->job->shouldReceive('release')->once();

但这不起作用。

有人有什么想法吗?

2 个答案:

答案 0 :(得分:1)

在分派作业之前,请尝试在测试中添加以下内容:

Queue::after(function (JobProcessed $event) {
    $this->assertTrue($event->job->isReleased());
});

上述代码将在作业完成后触发,并检查作业是否已被释放。

请务必删除对Queue::fake()$this->expectsJob()的所有来电,因为这会阻止实际作业的执行。

答案 1 :(得分:0)

我通过创建仅在作业释放回队列后触发的事件来解决此问题。然后在我的测试中,我可以使用事件模拟在我发送作业后查看该事件,并知道我们是否将其重新发送回队列。

// In your job
$this->release();
event(new MyJobReleasedEvent()); // Every time you have a release() call

// In your Unit Test
Event::fake([MyJobReleasedEvent::class]);
dispatch(new MyJob();
Event::assertDispatched(MyJobReleasedEvent::class);

如果你想得到想象力,我确定你可以连接你自己的Job类,它在调用release()时自动执行此操作,但我很少需要它,只需根据需要进行内联。< / p>