mirror of
				https://github.com/crater-invoice/crater.git
				synced 2025-11-03 14:03:18 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			466 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			466 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
use Crater\Http\Controllers\V1\Admin\Invoice\InvoicesController;
 | 
						|
use Crater\Http\Requests\InvoicesRequest;
 | 
						|
use Crater\Mail\SendInvoiceMail;
 | 
						|
use Crater\Models\Invoice;
 | 
						|
use Crater\Models\InvoiceItem;
 | 
						|
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('testGetInvoices', function () {
 | 
						|
    $response = getJson('api/v1/invoices?page=1&type=OVERDUE&limit=20');
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
});
 | 
						|
 | 
						|
test('create invoice', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'taxes' => [Tax::factory()->raw()],
 | 
						|
            'items' => [InvoiceItem::factory()->raw()],
 | 
						|
        ]);
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices', $invoice);
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'template_name' => $invoice['template_name'],
 | 
						|
        'invoice_number' => $invoice['invoice_number'],
 | 
						|
        'sub_total' => $invoice['sub_total'],
 | 
						|
        'discount' => $invoice['discount'],
 | 
						|
        'customer_id' => $invoice['customer_id'],
 | 
						|
        'total' => $invoice['total'],
 | 
						|
        'tax' => $invoice['tax'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'item_id' => $invoice['items'][0]['item_id'],
 | 
						|
        'name' => $invoice['items'][0]['name']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('create invoice as sent', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'taxes' => [Tax::factory()->raw()],
 | 
						|
            'items' => [InvoiceItem::factory()->raw()],
 | 
						|
        ]);
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices', $invoice);
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'invoice_number' => $invoice['invoice_number'],
 | 
						|
        'sub_total' => $invoice['sub_total'],
 | 
						|
        'total' => $invoice['total'],
 | 
						|
        'tax' => $invoice['tax'],
 | 
						|
        'discount' => $invoice['discount'],
 | 
						|
        'customer_id' => $invoice['customer_id'],
 | 
						|
        'template_name' => $invoice['template_name'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'item_id' => $invoice['items'][0]['item_id'],
 | 
						|
        'name' => $invoice['items'][0]['name']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('store validates using a form request', function () {
 | 
						|
    $this->assertActionUsesFormRequest(
 | 
						|
        InvoicesController::class,
 | 
						|
        'store',
 | 
						|
        InvoicesRequest::class
 | 
						|
    );
 | 
						|
});
 | 
						|
 | 
						|
test('update invoice', function () {
 | 
						|
    $invoice = Invoice::factory()->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $invoice2 = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'taxes' => [Tax::factory()->raw()],
 | 
						|
            'items' => [InvoiceItem::factory()->raw()],
 | 
						|
        ]);
 | 
						|
 | 
						|
    putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'invoice_number' => $invoice2['invoice_number'],
 | 
						|
        'sub_total' => $invoice2['sub_total'],
 | 
						|
        'total' => $invoice2['total'],
 | 
						|
        'tax' => $invoice2['tax'],
 | 
						|
        'discount' => $invoice2['discount'],
 | 
						|
        'customer_id' => $invoice2['customer_id'],
 | 
						|
        'template_name' => $invoice2['template_name'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'item_id' => $invoice2['items'][0]['item_id'],
 | 
						|
        'name' => $invoice2['items'][0]['name']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('update validates using a form request', function () {
 | 
						|
    $this->assertActionUsesFormRequest(
 | 
						|
        InvoicesController::class,
 | 
						|
        'update',
 | 
						|
        InvoicesRequest::class
 | 
						|
    );
 | 
						|
});
 | 
						|
 | 
						|
test('send invoice to customer', function () {
 | 
						|
    Mail::fake();
 | 
						|
 | 
						|
    $invoices = Invoice::factory()->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $data = [
 | 
						|
        'from' => 'john@example.com',
 | 
						|
        'to' => 'doe@example.com',
 | 
						|
        'subject' => 'email subject',
 | 
						|
        'body' => 'email body',
 | 
						|
    ];
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices/'.$invoices->id.'/send', $data);
 | 
						|
 | 
						|
    $response
 | 
						|
        ->assertOk()
 | 
						|
        ->assertJson([
 | 
						|
            'success' => true,
 | 
						|
        ]);
 | 
						|
 | 
						|
    $invoice2 = Invoice::find($invoices->id);
 | 
						|
 | 
						|
    $this->assertEquals($invoice2->status, Invoice::STATUS_SENT);
 | 
						|
    Mail::assertSent(SendInvoiceMail::class);
 | 
						|
});
 | 
						|
 | 
						|
test('invoice mark as paid', function () {
 | 
						|
    $invoice = Invoice::factory()->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $data = [
 | 
						|
        'status' => Invoice::STATUS_COMPLETED,
 | 
						|
    ];
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices/'.$invoice->id.'/status', $data);
 | 
						|
 | 
						|
    $response
 | 
						|
        ->assertOk()
 | 
						|
        ->assertJson([
 | 
						|
            'success' => true,
 | 
						|
        ]);
 | 
						|
 | 
						|
    $this->assertEquals(Invoice::find($invoice->id)->paid_status, Invoice::STATUS_PAID);
 | 
						|
});
 | 
						|
 | 
						|
test('invoice mark as sent', function () {
 | 
						|
    $invoice = Invoice::factory()->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $data = [
 | 
						|
        'status' => Invoice::STATUS_SENT,
 | 
						|
    ];
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices/'.$invoice->id.'/status', $data);
 | 
						|
 | 
						|
    $response
 | 
						|
        ->assertOk()
 | 
						|
        ->assertJson([
 | 
						|
            'success' => true,
 | 
						|
        ]);
 | 
						|
 | 
						|
    $this->assertEquals(Invoice::find($invoice->id)->status, Invoice::STATUS_SENT);
 | 
						|
});
 | 
						|
 | 
						|
test('search invoices', function () {
 | 
						|
    $filters = [
 | 
						|
        'page' => 1,
 | 
						|
        'limit' => 15,
 | 
						|
        'search' => 'doe',
 | 
						|
        'status' => Invoice::STATUS_DRAFT,
 | 
						|
        'from_date' => '2019-01-20',
 | 
						|
        'to_date' => '2019-01-27',
 | 
						|
        'invoice_number' => '000012',
 | 
						|
    ];
 | 
						|
 | 
						|
    $queryString = http_build_query($filters, '', '&');
 | 
						|
 | 
						|
    $response = getJson('api/v1/invoices?'.$queryString);
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
});
 | 
						|
 | 
						|
test('delete multiple invoices', function () {
 | 
						|
    $invoices = Invoice::factory()->count(3)->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $ids = $invoices->pluck('id');
 | 
						|
 | 
						|
    $data = [
 | 
						|
        'ids' => $ids,
 | 
						|
    ];
 | 
						|
 | 
						|
    postJson('api/v1/invoices/delete', $data)
 | 
						|
        ->assertOk()
 | 
						|
        ->assertJson([
 | 
						|
            'success' => true,
 | 
						|
        ]);
 | 
						|
 | 
						|
    foreach ($invoices as $invoice) {
 | 
						|
        $this->assertDeleted($invoice);
 | 
						|
    }
 | 
						|
});
 | 
						|
 | 
						|
test('clone invoice', function () {
 | 
						|
    $invoices = Invoice::factory()->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    postJson("api/v1/invoices/{$invoices->id}/clone")
 | 
						|
        ->assertStatus(201);
 | 
						|
});
 | 
						|
 | 
						|
test('create invoice with negative tax', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'taxes' => [Tax::factory()->raw([
 | 
						|
                'percent' => -9.99
 | 
						|
            ])],
 | 
						|
            'items' => [InvoiceItem::factory()->raw()],
 | 
						|
        ]);
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices', $invoice);
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'invoice_number' => $invoice['invoice_number'],
 | 
						|
        'sub_total' => $invoice['sub_total'],
 | 
						|
        'total' => $invoice['total'],
 | 
						|
        'tax' => $invoice['tax'],
 | 
						|
        'discount' => $invoice['discount'],
 | 
						|
        'customer_id' => $invoice['customer_id'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'name' => $invoice['items'][0]['name'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('taxes', [
 | 
						|
        'tax_type_id' => $invoice['taxes'][0]['tax_type_id']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('create invoice with tax per item', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
                'tax_per_item' => 'YES',
 | 
						|
                'items' => [
 | 
						|
                    InvoiceItem::factory()->raw([
 | 
						|
                        'taxes' => [Tax::factory()->raw()],
 | 
						|
                    ]),
 | 
						|
                    InvoiceItem::factory()->raw([
 | 
						|
                        'taxes' => [Tax::factory()->raw()],
 | 
						|
                    ]),
 | 
						|
                ],
 | 
						|
            ]);
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices', $invoice);
 | 
						|
 | 
						|
    $response->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'invoice_number' => $invoice['invoice_number'],
 | 
						|
        'sub_total' => $invoice['sub_total'],
 | 
						|
        'total' => $invoice['total'],
 | 
						|
        'tax' => $invoice['tax'],
 | 
						|
        'discount' => $invoice['discount'],
 | 
						|
        'customer_id' => $invoice['customer_id'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'name' => $invoice['items'][0]['name'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('taxes', [
 | 
						|
        'tax_type_id' => $invoice['items'][0]['taxes'][0]['tax_type_id']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('create invoice with EUR currency', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'discount_type' => 'fixed',
 | 
						|
            'discount_val' => 20,
 | 
						|
            'sub_total' => 100,
 | 
						|
            'total' => 84,
 | 
						|
            'tax' => 4,
 | 
						|
            'due_amount' => 84,
 | 
						|
            'exchange_rate' => 86.403538,
 | 
						|
            'base_discount_val' => 1728.07,
 | 
						|
            'base_sub_total' => 8640.35,
 | 
						|
            'base_total' => 7257.90,
 | 
						|
            'base_tax' => 345.61,
 | 
						|
            'base_due_amount' => 7257.90,
 | 
						|
            'taxes' => [Tax::factory()->raw([
 | 
						|
                'amount' => 4,
 | 
						|
                'percent' => 5,
 | 
						|
                'base_amount' => 345.61,
 | 
						|
            ])],
 | 
						|
            'items' => [InvoiceItem::factory()->raw([
 | 
						|
                'discount_type' => 'fixed',
 | 
						|
                'price' => 100,
 | 
						|
                'quantity' => 1,
 | 
						|
                'discount' => 0,
 | 
						|
                'discount_val' => 0,
 | 
						|
                'tax' => 0,
 | 
						|
                'total' => 100,
 | 
						|
                'base_price' => 8640.35,
 | 
						|
                'exchange_rate' => 86.403538,
 | 
						|
                'base_discount_val' => 0,
 | 
						|
                'base_tax' => 0,
 | 
						|
                'base_total' => 8640.35,
 | 
						|
            ])],
 | 
						|
        ]);
 | 
						|
 | 
						|
    $response = postJson('api/v1/invoices', $invoice)->assertOk();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', [
 | 
						|
        'template_name' => $invoice['template_name'],
 | 
						|
        'invoice_number' => $invoice['invoice_number'],
 | 
						|
        'sub_total' => $invoice['sub_total'],
 | 
						|
        'discount' => $invoice['discount'],
 | 
						|
        'customer_id' => $invoice['customer_id'],
 | 
						|
        'total' => $invoice['total'],
 | 
						|
        'tax' => $invoice['tax'],
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('taxes', [
 | 
						|
        'tax_type_id' => $invoice['taxes'][0]['tax_type_id'],
 | 
						|
        'amount' => $invoice['tax']
 | 
						|
    ]);
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', [
 | 
						|
        'item_id' => $invoice['items'][0]['item_id'],
 | 
						|
        'name' => $invoice['items'][0]['name']
 | 
						|
    ]);
 | 
						|
});
 | 
						|
 | 
						|
test('update invoice with EUR currency', function () {
 | 
						|
    $invoice = Invoice::factory()
 | 
						|
        ->hasItems(1)
 | 
						|
        ->hasTaxes(1)
 | 
						|
        ->create([
 | 
						|
        'invoice_date' => '1988-07-18',
 | 
						|
        'due_date' => '1988-08-18',
 | 
						|
    ]);
 | 
						|
 | 
						|
    $invoice2 = Invoice::factory()
 | 
						|
        ->raw([
 | 
						|
            'id' => $invoice['id'],
 | 
						|
            'discount_type' => 'fixed',
 | 
						|
            'discount_val' => 20,
 | 
						|
            'sub_total' => 100,
 | 
						|
            'total' => 84,
 | 
						|
            'tax' => 4,
 | 
						|
            'due_amount' => 84,
 | 
						|
            'exchange_rate' => 86.403538,
 | 
						|
            'base_discount_val' => 1728.07,
 | 
						|
            'base_sub_total' => 8640.35,
 | 
						|
            'base_total' => 7257.897192,
 | 
						|
            'base_tax' => 345.614152,
 | 
						|
            'base_due_amount' => 7257.897192,
 | 
						|
            'taxes' => [Tax::factory()->raw([
 | 
						|
                'tax_type_id' => $invoice->taxes[0]->tax_type_id,
 | 
						|
                'amount' => 4,
 | 
						|
                'percent' => 5,
 | 
						|
                'base_amount' => 345.614152,
 | 
						|
            ])],
 | 
						|
            'items' => [InvoiceItem::factory()->raw([
 | 
						|
                'invoice_id' => $invoice->id,
 | 
						|
                'discount_type' => 'fixed',
 | 
						|
                'price' => 100,
 | 
						|
                'quantity' => 1,
 | 
						|
                'discount' => 0,
 | 
						|
                'discount_val' => 0,
 | 
						|
                'tax' => 0,
 | 
						|
                'total' => 100,
 | 
						|
                'base_price' => 8640.3538,
 | 
						|
                'exchange_rate' => 86.403538,
 | 
						|
                'base_discount_val' => 0,
 | 
						|
                'base_tax' => 0,
 | 
						|
                'base_total' => 8640.3538,
 | 
						|
            ])],
 | 
						|
        ]);
 | 
						|
 | 
						|
    putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk();
 | 
						|
 | 
						|
    $invoice_assert = collect($invoice2)
 | 
						|
        ->only([
 | 
						|
            'invoice_number',
 | 
						|
            'template_name',
 | 
						|
            'sub_total',
 | 
						|
            'total',
 | 
						|
            'tax',
 | 
						|
            'discount',
 | 
						|
            'customer_id',
 | 
						|
        ])
 | 
						|
        ->toArray();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoices', $invoice_assert);
 | 
						|
 | 
						|
    $invoice_item_assert = collect($invoice2['items'][0])
 | 
						|
        ->only([
 | 
						|
            'invoice_id',
 | 
						|
            'item_id',
 | 
						|
            'name',
 | 
						|
        ])
 | 
						|
        ->toArray();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('invoice_items', $invoice_item_assert);
 | 
						|
 | 
						|
    $invoice_tax_assert = collect($invoice2['taxes'][0])
 | 
						|
        ->only([
 | 
						|
            'name',
 | 
						|
            'amount'
 | 
						|
        ])
 | 
						|
        ->toArray();
 | 
						|
 | 
						|
    $this->assertDatabaseHas('taxes', $invoice_tax_assert);
 | 
						|
});
 |