2019-04-04 00:40:56 +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
|
|
|
|
2019-04-04 00:40:56 +02:00
|
|
|
namespace Tests\Feature;
|
|
|
|
|
2024-01-09 10:05:18 +01:00
|
|
|
use App\DataMapper\InvoiceItem;
|
|
|
|
use Tests\TestCase;
|
|
|
|
use App\Models\Invoice;
|
2019-04-04 00:40:56 +02:00
|
|
|
use App\Models\Product;
|
2024-01-09 10:05:18 +01:00
|
|
|
use Tests\MockAccountData;
|
2019-04-04 00:40:56 +02:00
|
|
|
use App\Utils\Traits\MakesHash;
|
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
|
|
use Illuminate\Support\Facades\Session;
|
2024-01-09 10:05:18 +01:00
|
|
|
use Illuminate\Routing\Middleware\ThrottleRequests;
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
2019-04-04 00:40:56 +02:00
|
|
|
|
2019-04-20 01:02:49 +02:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @covers App\Http\Controllers\ProductController
|
|
|
|
*/
|
2019-04-04 00:40:56 +02:00
|
|
|
class ProductTest extends TestCase
|
|
|
|
{
|
|
|
|
use MakesHash;
|
2019-04-24 12:01:40 +02:00
|
|
|
use DatabaseTransactions;
|
2020-05-19 00:22:18 +02:00
|
|
|
use MockAccountData;
|
2019-04-04 00:40:56 +02:00
|
|
|
|
2022-06-21 12:00:57 +02:00
|
|
|
protected function setUp() :void
|
2019-04-04 00:40:56 +02:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
Session::start();
|
|
|
|
|
|
|
|
$this->faker = \Faker\Factory::create();
|
|
|
|
|
|
|
|
Model::reguard();
|
|
|
|
|
2020-03-16 11:12:10 +01:00
|
|
|
$this->withoutMiddleware(
|
|
|
|
ThrottleRequests::class
|
|
|
|
);
|
2020-05-19 00:22:18 +02:00
|
|
|
|
|
|
|
$this->makeTestData();
|
2023-04-19 09:31:29 +02:00
|
|
|
$this->withoutExceptionHandling();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:05:18 +01:00
|
|
|
public function testProductCostMigration()
|
|
|
|
{
|
|
|
|
$items = [];
|
|
|
|
|
|
|
|
$item = new InvoiceItem();
|
|
|
|
$item->product_cost = 0;
|
|
|
|
$item->product_key = 'test';
|
|
|
|
$item->quantity = 1;
|
|
|
|
$item->cost = 10;
|
|
|
|
$item->notes = 'product';
|
|
|
|
|
|
|
|
$items[] = $item;
|
|
|
|
|
|
|
|
$p = Product::factory()
|
|
|
|
->create([
|
|
|
|
'user_id' => $this->user->id,
|
|
|
|
'company_id' => $this->company->id,
|
|
|
|
'product_key' => 'test',
|
|
|
|
'cost' => 10,
|
|
|
|
'price' => 20,
|
|
|
|
'quantity' => 1,
|
|
|
|
'notes' => 'product',
|
|
|
|
]);
|
|
|
|
|
|
|
|
$i = Invoice::factory()
|
|
|
|
->create([
|
|
|
|
'client_id' => $this->client->id,
|
|
|
|
'company_id' => $this->company->id,
|
|
|
|
'user_id' => $this->user->id,
|
|
|
|
'line_items' => $items,
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
$line_items = $i->line_items;
|
|
|
|
|
|
|
|
$this->assertEquals(0, $line_items[0]->product_cost);
|
|
|
|
|
|
|
|
Invoice::withTrashed()
|
|
|
|
->where('is_deleted', false)
|
|
|
|
->cursor()
|
|
|
|
->each(function (Invoice $invoice) {
|
|
|
|
|
|
|
|
|
|
|
|
$line_items = $invoice->line_items;
|
|
|
|
|
|
|
|
foreach ($line_items as $key => $item) {
|
|
|
|
|
|
|
|
if($item?->product_cost == 0 && $product = Product::where('company_id', $invoice->company_id)->where('product_key', $item->product_key)->where('cost', '>', 0)->first()) {
|
|
|
|
$line_items[$key]->product_cost = $product->cost;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$invoice->line_items = $line_items;
|
|
|
|
$invoice->saveQuietly();
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
$i = $i->fresh();
|
|
|
|
$line_items = $i->line_items;
|
|
|
|
|
|
|
|
$this->assertEquals(10, $line_items[0]->product_cost);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-04-19 09:31:29 +02:00
|
|
|
public function testSetTaxId()
|
|
|
|
{
|
|
|
|
$p = Product::factory()->create([
|
|
|
|
'user_id' => $this->user->id,
|
|
|
|
'company_id' => $this->company->id
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
$this->assertEquals(1, $p->tax_id);
|
|
|
|
|
|
|
|
$update = [
|
|
|
|
'ids' => [$p->hashed_id],
|
|
|
|
'action' => 'set_tax_id',
|
|
|
|
'tax_id' => 6,
|
|
|
|
];
|
|
|
|
|
|
|
|
$response = false;
|
|
|
|
|
|
|
|
try {
|
|
|
|
$response = $this->withHeaders([
|
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
|
|
'X-API-TOKEN' => $this->token,
|
|
|
|
])->post('/api/v1/products/bulk', $update)
|
|
|
|
->assertStatus(200);
|
2023-11-26 08:41:42 +01:00
|
|
|
} catch(\Exception $e) {
|
2023-04-19 09:31:29 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
$p = $p->fresh();
|
|
|
|
|
|
|
|
$this->assertEquals(6, $p->tax_id);
|
|
|
|
|
2019-04-04 00:40:56 +02:00
|
|
|
}
|
|
|
|
|
2023-01-19 01:24:40 +01:00
|
|
|
public function testProductGetProductKeyFilter()
|
|
|
|
{
|
|
|
|
$response = $this->withHeaders([
|
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
|
|
'X-API-TOKEN' => $this->token,
|
|
|
|
])->get('/api/v1/products?product_key=xx')
|
|
|
|
->assertStatus(200);
|
|
|
|
}
|
|
|
|
|
2019-04-04 00:40:56 +02:00
|
|
|
public function testProductList()
|
|
|
|
{
|
|
|
|
$response = $this->withHeaders([
|
2022-06-21 11:57:17 +02:00
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
|
|
'X-API-TOKEN' => $this->token,
|
|
|
|
])->get('/api/v1/products');
|
2019-04-04 00:40:56 +02:00
|
|
|
|
|
|
|
$response->assertStatus(200);
|
|
|
|
|
|
|
|
$response = $this->withHeaders([
|
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
2020-05-19 00:22:18 +02:00
|
|
|
'X-API-TOKEN' => $this->token,
|
2020-03-21 06:37:30 +01:00
|
|
|
])->post(
|
|
|
|
'/api/v1/products/',
|
2019-04-04 00:40:56 +02:00
|
|
|
[
|
|
|
|
'product_key' => 'a-new-product-key',
|
|
|
|
'notes' => 'Product Notes',
|
|
|
|
'cost' => 10,
|
|
|
|
'qty' => 10,
|
|
|
|
'tax_name1' => 'GST',
|
|
|
|
'tax_rate1' => 10,
|
|
|
|
'tax_name2' => 'VAT',
|
|
|
|
'tax_rate2' => 17.5,
|
|
|
|
'custom_value1' => 'custom',
|
|
|
|
'custom_value2' => 'custom',
|
|
|
|
'custom_value3' => 'custom',
|
|
|
|
'custom_value4' => 'custom',
|
2020-09-06 11:38:10 +02:00
|
|
|
'is_deleted' => 0,
|
2019-04-04 00:40:56 +02:00
|
|
|
]
|
|
|
|
)
|
|
|
|
->assertStatus(200);
|
|
|
|
|
2020-10-06 13:32:07 +02:00
|
|
|
$arr = $response->json();
|
|
|
|
$product = Product::find($this->decodePrimaryKey($arr['data']['id']));
|
|
|
|
|
2019-04-04 00:40:56 +02:00
|
|
|
$product_update = [
|
2020-09-06 11:38:10 +02:00
|
|
|
'notes' => 'CHANGE',
|
2019-04-04 00:40:56 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
$response = $this->withHeaders([
|
2022-06-21 11:57:17 +02:00
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
|
|
|
'X-API-TOKEN' => $this->token,
|
|
|
|
])->put('/api/v1/products/'.$this->encodePrimaryKey($product->id), $product_update)
|
2019-04-04 00:40:56 +02:00
|
|
|
->assertStatus(200);
|
|
|
|
|
|
|
|
$response = $this->withHeaders([
|
|
|
|
'X-API-SECRET' => config('ninja.api_secret'),
|
2020-05-19 00:22:18 +02:00
|
|
|
'X-API-TOKEN' => $this->token,
|
2019-04-04 00:40:56 +02:00
|
|
|
])->delete('/api/v1/products/'.$this->encodePrimaryKey($product->id))
|
|
|
|
->assertStatus(200);
|
|
|
|
}
|
|
|
|
}
|