2020-04-15 02:30:52 +02:00
|
|
|
<?php
|
2020-09-14 13:11:46 +02:00
|
|
|
/**
|
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
2021-01-03 22:54:54 +01:00
|
|
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
2020-09-14 13:11:46 +02:00
|
|
|
*
|
2022-06-21 11:57:17 +02:00
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
2020-09-14 13:11:46 +02:00
|
|
|
*/
|
2022-06-21 11:57:17 +02:00
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
namespace Tests\Feature;
|
|
|
|
|
|
|
|
use App\Jobs\Util\ReminderJob;
|
|
|
|
use App\Models\Invoice;
|
|
|
|
use App\Utils\Traits\MakesHash;
|
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
|
|
use Illuminate\Routing\Middleware\ThrottleRequests;
|
|
|
|
use Illuminate\Support\Carbon;
|
|
|
|
use Tests\MockAccountData;
|
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @covers App\Jobs\Util\ReminderJob
|
|
|
|
*/
|
|
|
|
class ReminderTest extends TestCase
|
|
|
|
{
|
|
|
|
use MakesHash;
|
|
|
|
use DatabaseTransactions;
|
|
|
|
use MockAccountData;
|
|
|
|
|
2023-04-29 10:40:50 +02:00
|
|
|
public $faker;
|
|
|
|
|
2022-06-21 12:00:57 +02:00
|
|
|
protected function setUp() :void
|
2020-04-15 02:30:52 +02:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
$this->withoutMiddleware(
|
|
|
|
ThrottleRequests::class
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->faker = \Faker\Factory::create();
|
|
|
|
|
|
|
|
Model::reguard();
|
|
|
|
|
|
|
|
$this->makeTestData();
|
|
|
|
|
|
|
|
$this->withoutExceptionHandling();
|
|
|
|
}
|
|
|
|
|
2023-07-27 09:33:55 +02:00
|
|
|
public function testForReminderFiringCorrectly()
|
|
|
|
{
|
|
|
|
|
|
|
|
$this->invoice->next_send_date = null;
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->last_sent_date = now();
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(5)->format('Y-m-d');
|
|
|
|
$this->invoice->reminder_last_sent = null;
|
|
|
|
$this->invoice->save();
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder1 = 2;
|
|
|
|
$settings->enable_reminder2 = false;
|
|
|
|
$settings->schedule_reminder2 = '';
|
|
|
|
$settings->num_days_reminder2 = 0;
|
|
|
|
$settings->enable_reminder3 = false;
|
|
|
|
$settings->schedule_reminder3 = '';
|
|
|
|
$settings->num_days_reminder3 = 0;
|
|
|
|
$settings->timezone_id = '109';
|
|
|
|
$settings->entity_send_time = 6;
|
|
|
|
$settings->endless_reminder_frequency_id = '';
|
|
|
|
$settings->enable_reminder_endless = false;
|
|
|
|
|
|
|
|
$this->client->company->settings = $settings;
|
|
|
|
$this->client->push();
|
|
|
|
|
|
|
|
$client_settings = $settings;
|
|
|
|
$client_settings->timezone_id = '5';
|
|
|
|
$client_settings->entity_send_time = 8;
|
|
|
|
|
|
|
|
$this->invoice->client->settings = $client_settings;
|
|
|
|
$this->invoice->push();
|
|
|
|
|
|
|
|
$this->invoice = $this->invoice->service()->markSent()->save();
|
|
|
|
$this->invoice->service()->setReminder($client_settings)->save();
|
|
|
|
|
|
|
|
$this->invoice = $this->invoice->fresh();
|
|
|
|
|
|
|
|
//due to UTC server time, we actually send the "day before"
|
|
|
|
$this->assertEquals(now()->addDays(1)->format('Y-m-d'), Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'));
|
|
|
|
|
|
|
|
$this->travelTo(now()->startOfDay());
|
|
|
|
|
|
|
|
for($x=0; $x<46; $x++) {
|
|
|
|
|
|
|
|
// nlog("traveller {$x} ".now()->format('Y-m-d h:i:s'));
|
|
|
|
(new ReminderJob())->handle();
|
|
|
|
$this->invoice = $this->invoice->fresh();
|
|
|
|
$this->assertNull($this->invoice->reminder1_sent);
|
|
|
|
$this->assertNull($this->invoice->reminder_last_sent);
|
|
|
|
|
|
|
|
$this->travelTo(now()->addHours(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
// nlog("traveller ".now()->format('Y-m-d'));
|
|
|
|
(new ReminderJob())->handle();
|
|
|
|
$this->invoice = $this->invoice->fresh();
|
|
|
|
$this->assertNotNull($this->invoice->reminder1_sent);
|
|
|
|
|
|
|
|
}
|
2022-08-10 05:35:47 +02:00
|
|
|
|
2023-04-29 10:40:50 +02:00
|
|
|
public function testForSingleEndlessReminder()
|
|
|
|
{
|
|
|
|
$this->invoice->next_send_date = null;
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->last_sent_date = now();
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(5)->format('Y-m-d');
|
|
|
|
$this->invoice->save();
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = false;
|
|
|
|
$settings->schedule_reminder1 = '';
|
|
|
|
$settings->num_days_reminder1 = 0;
|
|
|
|
$settings->enable_reminder2 = false;
|
|
|
|
$settings->schedule_reminder2 = '';
|
|
|
|
$settings->num_days_reminder2 = 0;
|
|
|
|
$settings->enable_reminder3 = false;
|
|
|
|
$settings->schedule_reminder3 = '';
|
|
|
|
$settings->num_days_reminder3 = 0;
|
|
|
|
$settings->timezone_id = '5';
|
|
|
|
$settings->entity_send_time = 8;
|
|
|
|
$settings->endless_reminder_frequency_id = '5';
|
|
|
|
$settings->enable_reminder_endless = true;
|
|
|
|
|
|
|
|
$this->client->company->settings = $settings;
|
|
|
|
$this->client->push();
|
|
|
|
|
|
|
|
$client_settings = $settings;
|
|
|
|
$client_settings->timezone_id = '5';
|
|
|
|
$client_settings->entity_send_time = 8;
|
|
|
|
|
|
|
|
$this->invoice->client->settings = $client_settings;
|
|
|
|
$this->invoice->push();
|
|
|
|
|
|
|
|
$this->invoice = $this->invoice->service()->markSent()->save();
|
|
|
|
$this->invoice->service()->setReminder($client_settings)->save();
|
|
|
|
|
|
|
|
$this->invoice = $this->invoice->fresh();
|
|
|
|
|
2023-06-01 04:32:40 +02:00
|
|
|
$this->assertEquals(now()->addMonthNoOverflow()->format('Y-m-d'), Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'));
|
2023-04-29 10:40:50 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-08-10 05:35:47 +02:00
|
|
|
public function testForClientTimezoneEdges()
|
|
|
|
{
|
|
|
|
$this->invoice->next_send_date = null;
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(5)->format('Y-m-d');
|
|
|
|
$this->invoice->save();
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'before_due_date';
|
|
|
|
$settings->num_days_reminder1 = 4;
|
|
|
|
$settings->enable_reminder2 = true;
|
|
|
|
$settings->schedule_reminder2 = 'before_due_date';
|
|
|
|
$settings->num_days_reminder2 = 2;
|
|
|
|
$settings->enable_reminder3 = true;
|
|
|
|
$settings->schedule_reminder3 = 'after_due_date';
|
|
|
|
$settings->num_days_reminder3 = 3;
|
|
|
|
$settings->timezone_id = '15';
|
|
|
|
$settings->entity_send_time = 8;
|
|
|
|
|
|
|
|
$this->client->company->settings = $settings;
|
|
|
|
$this->client->push();
|
|
|
|
|
|
|
|
$client_settings = $settings;
|
|
|
|
$client_settings->timezone_id = '15';
|
|
|
|
$client_settings->entity_send_time = 8;
|
|
|
|
|
|
|
|
$this->invoice->client->settings = $client_settings;
|
|
|
|
$this->invoice->push();
|
|
|
|
|
|
|
|
$this->invoice = $this->invoice->service()->markSent()->save();
|
|
|
|
$this->invoice->service()->setReminder($client_settings)->save();
|
|
|
|
|
|
|
|
$next_send_date = Carbon::parse($this->invoice->next_send_date);
|
|
|
|
$calculatedReminderDate = Carbon::parse($this->invoice->due_date)->subDays(4)->addSeconds($this->invoice->client->timezone_offset());
|
|
|
|
|
|
|
|
nlog($next_send_date->format('Y-m-d h:i:s'));
|
|
|
|
nlog($calculatedReminderDate->format('Y-m-d h:i:s'));
|
|
|
|
|
2023-07-27 09:35:47 +02:00
|
|
|
$this->travelTo($calculatedReminderDate);
|
2022-08-10 05:35:47 +02:00
|
|
|
|
|
|
|
$reminder_template = $this->invoice->calculateTemplate('invoice');
|
|
|
|
|
|
|
|
$this->assertEquals('reminder1', $reminder_template);
|
|
|
|
|
|
|
|
$this->assertTrue($next_send_date->eq($calculatedReminderDate));
|
|
|
|
|
|
|
|
$this->invoice->service()->touchReminder($reminder_template)->save();
|
|
|
|
|
|
|
|
$this->assertNotNull($this->invoice->last_sent_date);
|
|
|
|
$this->assertNotNull($this->invoice->reminder1_sent);
|
|
|
|
$this->assertNotNull($this->invoice->reminder_last_sent);
|
|
|
|
|
|
|
|
//calc next send date
|
|
|
|
$this->invoice->service()->setReminder()->save();
|
|
|
|
|
|
|
|
$next_send_date = Carbon::parse($this->invoice->next_send_date);
|
|
|
|
|
|
|
|
nlog($next_send_date->format('Y-m-d h:i:s'));
|
|
|
|
|
|
|
|
$calculatedReminderDate = Carbon::parse($this->invoice->due_date)->subDays(2)->addSeconds($this->invoice->client->timezone_offset());
|
|
|
|
$this->assertTrue($next_send_date->eq($calculatedReminderDate));
|
|
|
|
|
|
|
|
$this->travelTo(now()->addDays(2));
|
|
|
|
|
|
|
|
$reminder_template = $this->invoice->calculateTemplate('invoice');
|
|
|
|
|
|
|
|
$this->assertEquals('reminder2', $reminder_template);
|
|
|
|
$this->invoice->service()->touchReminder($reminder_template)->save();
|
|
|
|
$this->assertNotNull($this->invoice->reminder2_sent);
|
|
|
|
|
|
|
|
$this->invoice->service()->setReminder()->save();
|
|
|
|
|
|
|
|
$next_send_date = Carbon::parse($this->invoice->next_send_date);
|
|
|
|
$calculatedReminderDate = Carbon::parse($this->invoice->due_date)->addDays(3)->addSeconds($this->invoice->client->timezone_offset());
|
|
|
|
$this->assertTrue($next_send_date->eq($calculatedReminderDate));
|
|
|
|
|
|
|
|
nlog($next_send_date->format('Y-m-d h:i:s'));
|
|
|
|
}
|
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
public function testReminderQueryCatchesDate()
|
|
|
|
{
|
|
|
|
$this->invoice->next_send_date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->save();
|
|
|
|
|
|
|
|
$invoices = Invoice::where('next_send_date', Carbon::today())->get();
|
|
|
|
|
|
|
|
$this->assertEquals(1, $invoices->count());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testReminderHits()
|
|
|
|
{
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(30)->format('Y-m-d');
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder1 = 7;
|
|
|
|
$settings->enable_reminder2 = true;
|
|
|
|
$settings->schedule_reminder2 = 'before_due_date';
|
|
|
|
$settings->num_days_reminder2 = 1;
|
|
|
|
$settings->enable_reminder3 = true;
|
|
|
|
$settings->schedule_reminder3 = 'after_due_date';
|
|
|
|
$settings->num_days_reminder3 = 1;
|
|
|
|
|
|
|
|
$this->company->settings = $settings;
|
|
|
|
$this->invoice->service()->markSent();
|
2021-05-26 02:35:39 +02:00
|
|
|
$this->invoice->service()->setReminder($settings)->save();
|
2020-04-15 02:30:52 +02:00
|
|
|
|
2021-06-11 00:20:46 +02:00
|
|
|
$this->assertEquals(Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'), Carbon::now()->addDays(7)->format('Y-m-d'));
|
2020-04-15 02:30:52 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
// ReminderJob::dispatchNow();
|
2021-05-25 23:31:17 +02:00
|
|
|
}
|
|
|
|
|
2021-08-01 00:05:31 +02:00
|
|
|
public function testReminderHitsScenarioH1()
|
|
|
|
{
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(30)->format('Y-m-d');
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'before_due_date';
|
|
|
|
$settings->num_days_reminder1 = 2;
|
|
|
|
$settings->enable_reminder2 = true;
|
|
|
|
$settings->schedule_reminder2 = 'after_due_date';
|
|
|
|
$settings->num_days_reminder2 = 14;
|
|
|
|
$settings->enable_reminder3 = true;
|
|
|
|
$settings->schedule_reminder3 = 'after_due_date';
|
|
|
|
$settings->num_days_reminder3 = 30;
|
|
|
|
|
|
|
|
$this->company->settings = $settings;
|
|
|
|
$this->invoice->service()->markSent();
|
|
|
|
$this->invoice->service()->setReminder($settings)->save();
|
|
|
|
|
|
|
|
$this->assertEquals(Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'), Carbon::now()->addDays(30)->subDays(2)->format('Y-m-d'));
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
// ReminderJob::dispatchNow();
|
2021-08-01 00:05:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-01 00:44:04 +02:00
|
|
|
/* Cant set a reminder in the past so need to skip reminder 2 and go straigh to reminder 3*/
|
2021-05-25 23:31:17 +02:00
|
|
|
public function testReminderNextSendRecalculation()
|
|
|
|
{
|
2021-05-26 01:43:23 +02:00
|
|
|
$this->invoice->date = now()->subDays(2)->format('Y-m-d');
|
2021-07-20 06:49:04 +02:00
|
|
|
$this->invoice->due_date = now()->addDays(30)->format('Y-m-d');
|
2021-05-26 01:43:23 +02:00
|
|
|
$this->invoice->reminder1_sent = now()->subDays(1)->format('Y-m-d');
|
2021-07-20 06:49:04 +02:00
|
|
|
$this->invoice->last_sent_date = now()->subDays(1)->format('Y-m-d');
|
|
|
|
$this->invoice->next_send_date = now()->subDays(1)->format('Y-m-d');
|
|
|
|
$this->invoice->reminder2_sent = null;
|
|
|
|
|
2021-05-25 23:31:17 +02:00
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder1 = 1;
|
|
|
|
$settings->enable_reminder2 = true;
|
2021-05-26 01:43:23 +02:00
|
|
|
$settings->schedule_reminder2 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder2 = 2;
|
2021-05-25 23:31:17 +02:00
|
|
|
$settings->enable_reminder3 = true;
|
2021-05-26 01:43:23 +02:00
|
|
|
$settings->schedule_reminder3 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder3 = 3;
|
2021-05-25 23:31:17 +02:00
|
|
|
|
|
|
|
$this->company->settings = $settings;
|
|
|
|
$this->invoice->service()->markSent();
|
2021-05-26 02:35:39 +02:00
|
|
|
$this->invoice->service()->setReminder($settings)->save();
|
2021-05-26 01:43:23 +02:00
|
|
|
|
|
|
|
$this->invoice->fresh();
|
|
|
|
|
2021-08-01 00:44:04 +02:00
|
|
|
$this->assertEquals(Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'), now()->addDay()->format('Y-m-d'));
|
2020-04-15 02:30:52 +02:00
|
|
|
}
|
2021-05-26 01:43:23 +02:00
|
|
|
|
|
|
|
public function testReminder3NextSendRecalculation()
|
|
|
|
{
|
|
|
|
$this->invoice->date = now()->subDays(3)->format('Y-m-d');
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(30)->format('Y-m-d');
|
|
|
|
$this->invoice->reminder1_sent = now()->subDays(2)->format('Y-m-d');
|
|
|
|
$this->invoice->reminder2_sent = now()->subDays(1)->format('Y-m-d');
|
2022-06-21 11:57:17 +02:00
|
|
|
|
2021-05-26 01:43:23 +02:00
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder1 = 1;
|
|
|
|
$settings->enable_reminder2 = true;
|
|
|
|
$settings->schedule_reminder2 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder2 = 2;
|
|
|
|
$settings->enable_reminder3 = true;
|
|
|
|
$settings->schedule_reminder3 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder3 = 3;
|
|
|
|
|
|
|
|
$this->company->settings = $settings;
|
|
|
|
$this->invoice->service()->markSent();
|
2021-05-26 02:35:39 +02:00
|
|
|
$this->invoice->service()->setReminder($settings)->save();
|
2021-05-26 01:43:23 +02:00
|
|
|
|
|
|
|
$this->invoice->fresh();
|
2022-06-21 11:57:17 +02:00
|
|
|
|
2021-06-11 00:20:46 +02:00
|
|
|
$this->assertEquals(Carbon::parse($this->invoice->next_send_date)->format('Y-m-d'), now()->format('Y-m-d'));
|
2021-05-26 01:43:23 +02:00
|
|
|
}
|
|
|
|
|
2022-05-16 10:53:01 +02:00
|
|
|
public function testReminderIsSet()
|
|
|
|
{
|
|
|
|
$this->invoice->next_send_date = null;
|
|
|
|
$this->invoice->date = now()->format('Y-m-d');
|
|
|
|
$this->invoice->due_date = Carbon::now()->addDays(30)->format('Y-m-d');
|
|
|
|
$this->invoice->save();
|
|
|
|
|
|
|
|
$settings = $this->company->settings;
|
|
|
|
$settings->enable_reminder1 = true;
|
|
|
|
$settings->schedule_reminder1 = 'after_invoice_date';
|
|
|
|
$settings->num_days_reminder1 = 7;
|
|
|
|
$settings->enable_reminder2 = true;
|
|
|
|
$settings->schedule_reminder2 = 'before_due_date';
|
|
|
|
$settings->num_days_reminder2 = 1;
|
|
|
|
$settings->enable_reminder3 = true;
|
|
|
|
$settings->schedule_reminder3 = 'after_due_date';
|
|
|
|
$settings->num_days_reminder3 = 1;
|
|
|
|
|
|
|
|
$this->company->settings = $settings;
|
|
|
|
$this->invoice = $this->invoice->service()->markSent()->save();
|
|
|
|
$this->invoice->service()->setReminder($settings)->save();
|
|
|
|
|
|
|
|
$this->assertNotNull($this->invoice->next_send_date);
|
|
|
|
}
|
2020-04-15 02:30:52 +02:00
|
|
|
}
|