mirror of
https://github.com/crater-invoice/crater.git
synced 2025-10-27 11:41:09 -04:00
446 lines
13 KiB
PHP
446 lines
13 KiB
PHP
<?php
|
|
|
|
use Crater\Http\Controllers\V1\Admin\Estimate\EstimatesController;
|
|
use Crater\Http\Controllers\V1\Admin\Estimate\SendEstimateController;
|
|
use Crater\Http\Requests\DeleteEstimatesRequest;
|
|
use Crater\Http\Requests\EstimatesRequest;
|
|
use Crater\Http\Requests\SendEstimatesRequest;
|
|
use Crater\Mail\SendEstimateMail;
|
|
use Crater\Models\Estimate;
|
|
use Crater\Models\EstimateItem;
|
|
use Crater\Models\Tax;
|
|
use Crater\Models\User;
|
|
use Illuminate\Support\Facades\Artisan;
|
|
use Laravel\Sanctum\Sanctum;
|
|
|
|
use function Pest\Laravel\getJson;
|
|
use function Pest\Laravel\postJson;
|
|
use function Pest\Laravel\putJson;
|
|
|
|
beforeEach(function () {
|
|
Artisan::call('db:seed', ['--class' => 'DatabaseSeeder', '--force' => true]);
|
|
Artisan::call('db:seed', ['--class' => 'DemoSeeder', '--force' => true]);
|
|
|
|
$user = User::find(1);
|
|
$this->withHeaders([
|
|
'company' => $user->companies()->first()->id,
|
|
]);
|
|
Sanctum::actingAs(
|
|
$user,
|
|
['*']
|
|
);
|
|
});
|
|
|
|
test('get estimates', function () {
|
|
$response = getJson('api/v1/estimates?page=1');
|
|
|
|
$response->assertOk();
|
|
});
|
|
|
|
test('create estimate', function () {
|
|
$estimate = Estimate::factory()->raw([
|
|
'estimate_number' => 'EST-000006',
|
|
'items' => [
|
|
EstimateItem::factory()->raw(),
|
|
],
|
|
'taxes' => [
|
|
Tax::factory()->raw(),
|
|
],
|
|
]);
|
|
|
|
postJson('api/v1/estimates', $estimate)
|
|
->assertStatus(201);
|
|
|
|
$this->assertDatabaseHas('estimates', [
|
|
'template_name' => $estimate['template_name'],
|
|
'estimate_number' => $estimate['estimate_number'],
|
|
'discount_type' => $estimate['discount_type'],
|
|
'discount_val' => $estimate['discount_val'],
|
|
'sub_total' => $estimate['sub_total'],
|
|
'discount' => $estimate['discount'],
|
|
'customer_id' => $estimate['customer_id'],
|
|
'total' => $estimate['total'],
|
|
'notes' => $estimate['notes'],
|
|
'tax' => $estimate['tax'],
|
|
]);
|
|
});
|
|
|
|
test('store validates using a form request', function () {
|
|
$this->assertActionUsesFormRequest(
|
|
EstimatesController::class,
|
|
'store',
|
|
EstimatesRequest::class
|
|
);
|
|
});
|
|
|
|
test('update estimate', function () {
|
|
$estimate = Estimate::factory()
|
|
->hasItems(1)
|
|
->hasTaxes(1)
|
|
->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$estimate2 = Estimate::factory()->raw([
|
|
'items' => [
|
|
EstimateItem::factory()->raw([
|
|
'estimate_id' => $estimate->id
|
|
]),
|
|
],
|
|
'taxes' => [
|
|
Tax::factory()->raw([
|
|
'tax_type_id' => $estimate->taxes[0]->tax_type_id,
|
|
]),
|
|
],
|
|
]);
|
|
|
|
$response = putJson('api/v1/estimates/'.$estimate->id, $estimate2);
|
|
|
|
$this->assertDatabaseHas('estimates', [
|
|
'template_name' => $estimate2['template_name'],
|
|
'estimate_number' => $estimate2['estimate_number'],
|
|
'discount_type' => $estimate2['discount_type'],
|
|
'discount_val' => $estimate2['discount_val'],
|
|
'sub_total' => $estimate2['sub_total'],
|
|
'discount' => $estimate2['discount'],
|
|
'customer_id' => $estimate2['customer_id'],
|
|
'total' => $estimate2['total'],
|
|
'notes' => $estimate2['notes'],
|
|
'tax' => $estimate2['tax'],
|
|
]);
|
|
|
|
$this->assertDatabaseHas('estimate_items', [
|
|
'estimate_id' => $estimate2['items'][0]['estimate_id'],
|
|
]);
|
|
|
|
$response->assertStatus(200);
|
|
});
|
|
|
|
test('update validates using a form request', function () {
|
|
$this->assertActionUsesFormRequest(
|
|
EstimatesController::class,
|
|
'update',
|
|
EstimatesRequest::class
|
|
);
|
|
});
|
|
|
|
test('search estimates', function () {
|
|
$filters = [
|
|
'page' => 1,
|
|
'limit' => 15,
|
|
'search' => 'doe',
|
|
'from_date' => '2020-07-18',
|
|
'to_date' => '2020-07-20',
|
|
'estimate_number' => '000003',
|
|
];
|
|
|
|
$queryString = http_build_query($filters, '', '&');
|
|
|
|
$response = getJson('api/v1/estimates?'.$queryString);
|
|
|
|
$response->assertStatus(200);
|
|
});
|
|
|
|
test('send estimate using a form request', function () {
|
|
$this->assertActionUsesFormRequest(
|
|
SendEstimateController::class,
|
|
'__invoke',
|
|
SendEstimatesRequest::class
|
|
);
|
|
});
|
|
|
|
test('send estimate to customer', function () {
|
|
Mail::fake();
|
|
|
|
$estimate = Estimate::factory()->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$data = [
|
|
'subject' => 'test',
|
|
'body' => 'test',
|
|
'from' => 'john@example.com',
|
|
'to' => 'doe@example.com',
|
|
];
|
|
|
|
postJson("api/v1/estimates/{$estimate->id}/send", $data)
|
|
->assertStatus(200)
|
|
->assertJson([
|
|
'success' => true,
|
|
]);
|
|
|
|
Mail::assertSent(SendEstimateMail::class);
|
|
});
|
|
|
|
test('estimate mark as accepted', function () {
|
|
$estimate = Estimate::factory()->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$data = [
|
|
'status' => Estimate::STATUS_ACCEPTED,
|
|
];
|
|
|
|
$response = postJson("api/v1/estimates/{$estimate->id}/status", $data);
|
|
|
|
$response
|
|
->assertOk()
|
|
->assertJson([
|
|
'success' => true,
|
|
]);
|
|
|
|
$estimate2 = Estimate::find($estimate->id);
|
|
$this->assertEquals($estimate2->status, Estimate::STATUS_ACCEPTED);
|
|
});
|
|
|
|
test('estimate mark as rejected', function () {
|
|
$estimate = Estimate::factory()->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$data = [
|
|
'status' => Estimate::STATUS_REJECTED,
|
|
];
|
|
|
|
$response = postJson("api/v1/estimates/{$estimate->id}/status", $data);
|
|
|
|
$response
|
|
->assertOk()
|
|
->assertJson([
|
|
'success' => true,
|
|
]);
|
|
|
|
$estimate2 = Estimate::find($estimate->id);
|
|
$this->assertEquals($estimate2->status, Estimate::STATUS_REJECTED);
|
|
});
|
|
|
|
test('create invoice from estimate', function () {
|
|
$estimate = Estimate::factory()->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$response = postJson("api/v1/estimates/{$estimate->id}/convert-to-invoice")
|
|
->assertStatus(200);
|
|
});
|
|
|
|
test('delete multiple estimates using a form request', function () {
|
|
$this->assertActionUsesFormRequest(
|
|
EstimatesController::class,
|
|
'delete',
|
|
DeleteEstimatesRequest::class
|
|
);
|
|
});
|
|
|
|
test('delete multiple estimates', function () {
|
|
$estimates = Estimate::factory()
|
|
->count(3)
|
|
->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$ids = $estimates->pluck('id');
|
|
|
|
$data = [
|
|
'ids' => $ids,
|
|
];
|
|
|
|
$response = postJson('api/v1/estimates/delete', $data);
|
|
|
|
$response
|
|
->assertStatus(200)
|
|
->assertJson([
|
|
'success' => true,
|
|
]);
|
|
|
|
foreach ($estimates as $estimate) {
|
|
$this->assertDeleted($estimate);
|
|
}
|
|
});
|
|
|
|
test('get estimate templates', function () {
|
|
getJson('api/v1/estimates/templates')->assertStatus(200);
|
|
});
|
|
|
|
test('create estimate with tax per item', function () {
|
|
$estimate = Estimate::factory()->raw([
|
|
'estimate_number' => 'EST-000006',
|
|
'tax_per_item' => 'YES',
|
|
'items' => [
|
|
EstimateItem::factory()->raw([
|
|
'taxes' => [Tax::factory()->raw()],
|
|
]),
|
|
EstimateItem::factory()->raw([
|
|
'taxes' => [Tax::factory()->raw()],
|
|
]),
|
|
],
|
|
]);
|
|
|
|
postJson('api/v1/estimates', $estimate)
|
|
->assertStatus(201);
|
|
|
|
$this->assertDatabaseHas('estimates', [
|
|
'template_name' => $estimate['template_name'],
|
|
'estimate_number' => $estimate['estimate_number'],
|
|
'discount_type' => $estimate['discount_type'],
|
|
'discount_val' => $estimate['discount_val'],
|
|
'sub_total' => $estimate['sub_total'],
|
|
'discount' => $estimate['discount'],
|
|
'customer_id' => $estimate['customer_id'],
|
|
'total' => $estimate['total'],
|
|
'notes' => $estimate['notes'],
|
|
'tax' => $estimate['tax'],
|
|
]);
|
|
|
|
$this->assertDatabaseHas('estimate_items', [
|
|
'name' => $estimate['items'][0]['name'],
|
|
]);
|
|
|
|
$this->assertDatabaseHas('taxes', [
|
|
'tax_type_id' => $estimate['items'][0]['taxes'][0]['tax_type_id']
|
|
]);
|
|
});
|
|
|
|
test('create estimate with EUR currency', function () {
|
|
$estimate = Estimate::factory()
|
|
->raw([
|
|
'discount_type' => 'fixed',
|
|
'discount_val' => 20,
|
|
'sub_total' => 200,
|
|
'total' => 189,
|
|
'tax' => 9,
|
|
'exchange_rate' => 86.403538,
|
|
'base_discount_val' => 1728.07,
|
|
'base_sub_total' => 17280.71,
|
|
'base_total' => 16330.27,
|
|
'base_tax' => 777.63,
|
|
'taxes' => [Tax::factory()->raw([
|
|
'amount' => 9,
|
|
'percent' => 5,
|
|
'exchange_rate' => 86.403538,
|
|
'base_amount' => 777.63,
|
|
])],
|
|
'items' => [EstimateItem::factory()->raw([
|
|
'discount_type' => 'fixed',
|
|
'quantity' => 1,
|
|
'discount' => 0,
|
|
'discount_val' => 0,
|
|
'price' => 200,
|
|
'tax' => 0,
|
|
'total' => 200,
|
|
'exchange_rate' => 86.403538,
|
|
'base_discount_val' => 0,
|
|
'base_price' => 17280.71,
|
|
'base_tax' => 777.63,
|
|
'base_total' => 17280.71,
|
|
])],
|
|
]);
|
|
|
|
$response = postJson('api/v1/estimates', $estimate)->assertStatus(201);
|
|
|
|
$this->assertDatabaseHas('estimates', [
|
|
'template_name' => $estimate['template_name'],
|
|
'estimate_number' => $estimate['estimate_number'],
|
|
'discount_type' => $estimate['discount_type'],
|
|
'discount_val' => $estimate['discount_val'],
|
|
'sub_total' => $estimate['sub_total'],
|
|
'discount' => $estimate['discount'],
|
|
'customer_id' => $estimate['customer_id'],
|
|
'total' => $estimate['total'],
|
|
'notes' => $estimate['notes'],
|
|
'tax' => $estimate['tax'],
|
|
]);
|
|
|
|
$this->assertDatabaseHas('taxes', [
|
|
'tax_type_id' => $estimate['taxes'][0]['tax_type_id'],
|
|
'amount' => $estimate['tax']
|
|
]);
|
|
|
|
$this->assertDatabaseHas('estimate_items', [
|
|
'item_id' => $estimate['items'][0]['item_id'],
|
|
'name' => $estimate['items'][0]['name']
|
|
]);
|
|
});
|
|
|
|
test('update estimate with EUR currency', function () {
|
|
$estimate = Estimate::factory()
|
|
->hasItems(1)
|
|
->hasTaxes(1)
|
|
->create([
|
|
'estimate_date' => '1988-07-18',
|
|
'expiry_date' => '1988-08-18',
|
|
]);
|
|
|
|
$estimate2 = Estimate::factory()
|
|
->raw([
|
|
'id' => $estimate->id,
|
|
'discount_type' => 'fixed',
|
|
'discount_val' => 20,
|
|
'sub_total' => 200,
|
|
'total' => 189,
|
|
'tax' => 9,
|
|
'exchange_rate' => 86.403538,
|
|
'base_discount_val' => 1728.07076,
|
|
'base_sub_total' => 17280.7076,
|
|
'base_total' => 16330.268682,
|
|
'base_tax' => 777.631842,
|
|
'taxes' => [Tax::factory()->raw([
|
|
'tax_type_id' => $estimate->taxes[0]->tax_type_id,
|
|
'amount' => 9,
|
|
'percent' => 5,
|
|
'exchange_rate' => 86.403538,
|
|
'base_amount' => 777.631842,
|
|
])],
|
|
'items' => [EstimateItem::factory()->raw([
|
|
'estimate_id' => $estimate->id,
|
|
'discount_type' => 'fixed',
|
|
'quantity' => 1,
|
|
'discount' => 0,
|
|
'discount_val' => 0,
|
|
'price' => 200,
|
|
'tax' => 0,
|
|
'total' => 200,
|
|
'exchange_rate' => 86.403538,
|
|
'base_discount_val' => 0,
|
|
'base_price' => 17280.7076,
|
|
'base_tax' => 777.631842,
|
|
'base_total' => 17280.7076,
|
|
])],
|
|
]);
|
|
|
|
$response = putJson('api/v1/estimates/'.$estimate->id, $estimate2);
|
|
|
|
$estimate_assert = collect($estimate2)
|
|
->only([
|
|
'id',
|
|
'template_name',
|
|
'estimate_number',
|
|
'discount_type',
|
|
'discount_val',
|
|
'sub_total',
|
|
'discount',
|
|
'customer_id',
|
|
'total',
|
|
'tax'
|
|
])
|
|
->toArray();
|
|
|
|
$this->assertDatabaseHas('estimates', $estimate_assert);
|
|
|
|
$estimate_item_assert = collect($estimate2['items'][0])
|
|
->only([
|
|
'estimate_id',
|
|
'amount'
|
|
])
|
|
->toArray();
|
|
|
|
$this->assertDatabaseHas('estimate_items', $estimate_item_assert);
|
|
|
|
$response->assertStatus(200);
|
|
});
|