2024-09-30 04:42:41 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Feature\Importing\Api;
|
|
|
|
|
2024-10-17 13:11:39 -07:00
|
|
|
use App\Mail\CheckoutAssetMail;
|
2024-09-30 04:42:41 -07:00
|
|
|
use App\Models\Actionlog as ActionLog;
|
|
|
|
use App\Models\Asset;
|
|
|
|
use App\Models\CustomField;
|
2024-10-03 13:59:58 -07:00
|
|
|
use App\Models\Import;
|
2024-09-30 04:42:41 -07:00
|
|
|
use App\Models\User;
|
|
|
|
use App\Notifications\CheckoutAssetNotification;
|
|
|
|
use Carbon\Carbon;
|
|
|
|
use Illuminate\Foundation\Testing\WithFaker;
|
|
|
|
use Illuminate\Support\Arr;
|
2024-10-17 13:11:39 -07:00
|
|
|
use Illuminate\Support\Facades\Mail;
|
2024-09-30 04:42:41 -07:00
|
|
|
use Illuminate\Support\Facades\Notification;
|
2024-10-03 15:14:07 -07:00
|
|
|
use Illuminate\Support\Str;
|
2024-09-30 04:42:41 -07:00
|
|
|
use Illuminate\Testing\TestResponse;
|
2024-10-03 15:14:07 -07:00
|
|
|
use PHPUnit\Framework\Attributes\Test;
|
2024-10-03 13:39:54 -07:00
|
|
|
use Tests\Concerns\TestsPermissionsRequirement;
|
2024-09-30 04:42:41 -07:00
|
|
|
use Tests\Support\Importing\AssetsImportFileBuilder as ImportFileBuilder;
|
2024-10-03 15:02:03 -07:00
|
|
|
use Tests\Support\Importing\CleansUpImportFiles;
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:39:54 -07:00
|
|
|
class ImportAssetsTest extends ImportDataTestCase implements TestsPermissionsRequirement
|
2024-09-30 04:42:41 -07:00
|
|
|
{
|
2024-10-03 15:02:03 -07:00
|
|
|
use CleansUpImportFiles;
|
2024-09-30 04:42:41 -07:00
|
|
|
use WithFaker;
|
|
|
|
|
|
|
|
protected function importFileResponse(array $parameters = []): TestResponse
|
|
|
|
{
|
|
|
|
if (!array_key_exists('import-type', $parameters)) {
|
|
|
|
$parameters['import-type'] = 'asset';
|
|
|
|
}
|
|
|
|
|
|
|
|
return parent::importFileResponse($parameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
2024-10-03 13:39:54 -07:00
|
|
|
public function testRequiresPermission()
|
2024-09-30 04:42:41 -07:00
|
|
|
{
|
2024-10-03 13:39:54 -07:00
|
|
|
$this->actingAsForApi(User::factory()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$this->importFileResponse(['import' => 44])->assertForbidden();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function userWithImportAssetsPermissionCanImportAssets(): void
|
|
|
|
{
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->canImport()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create();
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function importAsset(): void
|
|
|
|
{
|
2024-10-17 13:11:39 -07:00
|
|
|
Mail::fake();
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$importFileBuilder = ImportFileBuilder::new();
|
|
|
|
$row = $importFileBuilder->firstRow();
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])
|
|
|
|
->assertOk()
|
|
|
|
->assertExactJson([
|
|
|
|
'payload' => null,
|
|
|
|
'status' => 'success',
|
|
|
|
'messages' => ['redirect_url' => route('hardware.index')]
|
|
|
|
]);
|
|
|
|
|
|
|
|
$newAsset = Asset::query()
|
|
|
|
->with(['location', 'supplier', 'company', 'assignedAssets', 'defaultLoc', 'assetStatus', 'model.category', 'model.manufacturer'])
|
|
|
|
->where('serial', $row['serialNumber'])
|
|
|
|
->sole();
|
|
|
|
|
|
|
|
$assignee = User::query()->find($newAsset->assigned_to, ['id', 'first_name', 'last_name', 'email', 'username']);
|
|
|
|
|
|
|
|
$activityLogs = ActionLog::query()
|
|
|
|
->where('item_type', Asset::class)
|
|
|
|
->where('item_id', $newAsset->id)
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(2, $activityLogs);
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals('checkout', $activityLogs[0]->action_type);
|
|
|
|
$this->assertEquals(Asset::class, $activityLogs[0]->item_type);
|
|
|
|
$this->assertEquals($assignee->id, $activityLogs[0]->target_id);
|
|
|
|
$this->assertEquals(User::class, $activityLogs[0]->target_type);
|
|
|
|
$this->assertEquals('Checkout from CSV Importer', $activityLogs[0]->note);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals('create', $activityLogs[1]->action_type);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($activityLogs[1]->target_id);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(Asset::class, $activityLogs[1]->item_type);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($activityLogs[1]->note);
|
|
|
|
$this->assertNull($activityLogs[1]->target_type);
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['assigneeFullName'], "{$assignee->first_name} {$assignee->last_name}");
|
|
|
|
$this->assertEquals($row['assigneeEmail'], $assignee->email);
|
|
|
|
$this->assertEquals($row['assigneeUsername'], $assignee->username);
|
|
|
|
|
|
|
|
$this->assertEquals($row['category'], $newAsset->model->category->name);
|
|
|
|
$this->assertEquals($row['manufacturerName'], $newAsset->model->manufacturer->name);
|
|
|
|
$this->assertEquals($row['itemName'], $newAsset->name);
|
|
|
|
$this->assertEquals($row['tag'], $newAsset->asset_tag);
|
|
|
|
$this->assertEquals($row['model'], $newAsset->model->name);
|
|
|
|
$this->assertEquals($row['modelNumber'], $newAsset->model->model_number);
|
|
|
|
$this->assertEquals($row['purchaseDate'], $newAsset->purchase_date->toDateString());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->asset_eol_date);
|
|
|
|
$this->assertEquals(0, $newAsset->eol_explicit);
|
|
|
|
$this->assertEquals($newAsset->location_id, $newAsset->rtd_location_id);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['purchaseCost'], $newAsset->purchase_cost);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->order_number);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals('', $newAsset->image);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->user_id);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(1, $newAsset->physical);
|
|
|
|
$this->assertEquals($row['status'], $newAsset->assetStatus->name);
|
|
|
|
$this->assertEquals(0, $newAsset->archived);
|
|
|
|
$this->assertEquals($row['warrantyInMonths'], $newAsset->warranty_months);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->deprecate);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['supplierName'], $newAsset->supplier->name);
|
|
|
|
$this->assertEquals(0, $newAsset->requestable);
|
|
|
|
$this->assertEquals($row['location'], $newAsset->defaultLoc->name);
|
|
|
|
$this->assertEquals(null, $newAsset->accepted);
|
|
|
|
$this->assertEquals(now()->toDateString(), Carbon::parse($newAsset->last_checkout)->toDateString());
|
|
|
|
$this->assertEquals(0, $newAsset->last_checkin);
|
|
|
|
$this->assertEquals(0, $newAsset->expected_checkin);
|
|
|
|
$this->assertEquals($row['companyName'], $newAsset->company->name);
|
|
|
|
$this->assertEquals(User::class, $newAsset->assigned_type);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->last_audit_date);
|
|
|
|
$this->assertNull($newAsset->next_audit_date);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['location'], $newAsset->location->name);
|
|
|
|
$this->assertEquals(0, $newAsset->checkin_counter);
|
|
|
|
$this->assertEquals(1, $newAsset->checkout_counter);
|
|
|
|
$this->assertEquals(0, $newAsset->requests_counter);
|
|
|
|
$this->assertEquals(0, $newAsset->byod);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
//Notes is never read.
|
2024-10-03 13:15:02 -07:00
|
|
|
// $this->assertEquals($row['notes'], $newAsset->notes);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-17 13:34:55 -07:00
|
|
|
Mail::assertSent(CheckoutAssetMail::class, function ($mail) use ($assignee) {
|
|
|
|
return $mail->hasTo($assignee->email);
|
|
|
|
});
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willIgnoreUnknownColumnsWhenFileContainsUnknownColumns(): void
|
|
|
|
{
|
|
|
|
$row = ImportFileBuilder::new()->definition();
|
|
|
|
$row['unknownColumnInCsvFile'] = 'foo';
|
|
|
|
|
|
|
|
$importFileBuilder = new ImportFileBuilder([$row]);
|
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewAssetWhenAssetWithSameTagAlreadyExists(): void
|
|
|
|
{
|
2024-10-03 13:59:58 -07:00
|
|
|
$asset = Asset::factory()->create(['asset_tag' => $this->faker->uuid]);
|
2024-09-30 04:42:41 -07:00
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['tag' => $asset->asset_tag]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])
|
|
|
|
->assertInternalServerError()
|
|
|
|
->assertExactJson([
|
|
|
|
'status' => 'import-errors',
|
|
|
|
'payload' => null,
|
|
|
|
'messages' => [
|
|
|
|
'' => [
|
|
|
|
'asset_tag' => [
|
|
|
|
'asset_tag' => [
|
|
|
|
"An asset with the asset tag {$asset->asset_tag} already exists and an update was not requested. No change was made."
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]);
|
|
|
|
|
|
|
|
$assetsWithSameTag = Asset::query()->where('asset_tag', $asset->asset_tag)->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $assetsWithSameTag);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewCompanyWhenCompanyExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['companyName' => Str::random()]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('company_id')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewLocationWhenLocationExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['location' => Str::random()]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('location_id')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewSupplierWhenSupplierExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['supplierName' => $this->faker->company]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get(['supplier_id']);
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('supplier_id')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewManufacturerWhenManufacturerExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['manufacturerName' => $this->faker->company]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->with('model.manufacturer')
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('model.manufacturer_id')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateCategoryWhenCategoryExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['category' => $this->faker->company]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->with('model.category')
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('model.category_id')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willNotCreateNewAssetModelWhenAssetModelExists(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(4)->replace(['model' => Str::random()]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->with('model')
|
|
|
|
->whereIn('serial', $importFileBuilder->pluck('serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(1, $newAssets->pluck('model.name')->unique()->all());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function whenColumnsAreMissingInImportFile(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times()->forget([
|
|
|
|
'purchaseCost',
|
|
|
|
'purchaseDate',
|
|
|
|
'status'
|
|
|
|
]);
|
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAsset = Asset::query()
|
|
|
|
->with(['assetStatus'])
|
|
|
|
->where('serial', $importFileBuilder->firstRow()['serialNumber'])
|
|
|
|
->sole();
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals('Ready to Deploy', $newAsset->assetStatus->name);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($newAsset->purchase_date);
|
|
|
|
$this->assertNull($newAsset->purchase_cost);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willFormatValues(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::new([
|
|
|
|
'warrantyInMonths' => '3 months',
|
|
|
|
'purchaseDate' => '2022/10/10'
|
|
|
|
]);
|
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAsset = Asset::query()
|
|
|
|
->where('serial', $importFileBuilder->firstRow()['serialNumber'])
|
|
|
|
->sole();
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(3, $newAsset->warranty_months);
|
|
|
|
$this->assertEquals('2022-10-10', $newAsset->purchase_date->toDateString());
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function whenRequiredColumnsAreMissingInImportFile(): void
|
|
|
|
{
|
|
|
|
$importFileBuilder = ImportFileBuilder::times(2)
|
|
|
|
->forget(['tag'])
|
|
|
|
->replace(['model' => '']);
|
|
|
|
|
|
|
|
$rows = $importFileBuilder->all();
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])
|
|
|
|
->assertInternalServerError()
|
|
|
|
->assertJson([
|
|
|
|
'status' => 'import-errors',
|
|
|
|
'payload' => null,
|
|
|
|
'messages' => [
|
|
|
|
$rows[0]['itemName'] => [
|
|
|
|
"Asset \"{$rows[0]['itemName']}\"" => [
|
|
|
|
'asset_tag' => [
|
|
|
|
'The asset tag field must be at least 1 characters.',
|
|
|
|
],
|
|
|
|
'model_id' => [
|
|
|
|
'The model id field is required.'
|
|
|
|
]
|
|
|
|
]
|
|
|
|
],
|
|
|
|
$rows[1]['itemName'] => [
|
|
|
|
"Asset \"{$rows[1]['itemName']}\"" => [
|
|
|
|
'asset_tag' => [
|
|
|
|
'The asset tag field must be at least 1 characters.',
|
|
|
|
],
|
|
|
|
'model_id' => [
|
|
|
|
'The model id field is required.'
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]);
|
|
|
|
|
|
|
|
$newAssets = Asset::query()
|
|
|
|
->whereIn('serial', Arr::pluck($rows, 'serialNumber'))
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->assertCount(0, $newAssets);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function updateAssetFromImport(): void
|
|
|
|
{
|
2024-10-03 13:59:58 -07:00
|
|
|
$asset = Asset::factory()->create()->refresh();
|
2024-09-30 04:42:41 -07:00
|
|
|
$importFileBuilder = ImportFileBuilder::times(1)->replace(['tag' => $asset->asset_tag]);
|
|
|
|
$row = $importFileBuilder->firstRow();
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id, 'import-update' => true])->assertOk();
|
|
|
|
|
|
|
|
$updatedAsset = Asset::query()
|
|
|
|
->with(['location', 'supplier', 'company', 'defaultLoc', 'assetStatus', 'model.category', 'model.manufacturer'])
|
|
|
|
->find($asset->id);
|
|
|
|
|
|
|
|
$assignee = User::query()->find($updatedAsset->assigned_to, ['id', 'first_name', 'last_name', 'email', 'username']);
|
|
|
|
|
|
|
|
$updatedAttributes = [
|
|
|
|
'category', 'manufacturer_id', 'name', 'tag', 'model_id',
|
|
|
|
'model_number', 'purchase_date', 'purchase_cost', 'warranty_months', 'supplier_id',
|
|
|
|
'location_id', 'company_id', 'serial', 'assigned_to', 'status_id', 'rtd_location_id',
|
|
|
|
'last_checkout', 'requestable', 'updated_at', 'checkout_counter', 'assigned_type'
|
|
|
|
];
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['assigneeFullName'], "{$assignee->first_name} {$assignee->last_name}");
|
|
|
|
$this->assertEquals($row['assigneeEmail'], $assignee->email);
|
|
|
|
$this->assertEquals($row['assigneeUsername'], $assignee->username);
|
|
|
|
|
|
|
|
$this->assertEquals($row['category'], $updatedAsset->model->category->name);
|
|
|
|
$this->assertEquals($row['manufacturerName'], $updatedAsset->model->manufacturer->name);
|
|
|
|
$this->assertEquals($row['itemName'], $updatedAsset->name);
|
|
|
|
$this->assertEquals($row['tag'], $updatedAsset->asset_tag);
|
|
|
|
$this->assertEquals($row['model'], $updatedAsset->model->name);
|
|
|
|
$this->assertEquals($row['modelNumber'], $updatedAsset->model->model_number);
|
|
|
|
$this->assertEquals($row['purchaseDate'], $updatedAsset->purchase_date->toDateString());
|
|
|
|
$this->assertEquals($row['purchaseCost'], $updatedAsset->purchase_cost);
|
|
|
|
$this->assertEquals($row['status'], $updatedAsset->assetStatus->name);
|
|
|
|
$this->assertEquals($row['warrantyInMonths'], $updatedAsset->warranty_months);
|
|
|
|
$this->assertEquals($row['supplierName'], $updatedAsset->supplier->name);
|
|
|
|
$this->assertEquals($row['location'], $updatedAsset->defaultLoc->name);
|
|
|
|
$this->assertEquals($row['companyName'], $updatedAsset->company->name);
|
|
|
|
$this->assertEquals($row['location'], $updatedAsset->location->name);
|
|
|
|
$this->assertEquals(1, $updatedAsset->checkout_counter);
|
|
|
|
$this->assertEquals(user::class, $updatedAsset->assigned_type);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
//RequestAble is always updated regardless of initial value.
|
2024-10-03 13:15:02 -07:00
|
|
|
// $this->assertEquals($asset->requestable, $updatedAsset->requestable);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$this->assertEquals(
|
|
|
|
Arr::except($asset->attributesToArray(), $updatedAttributes),
|
2024-10-03 13:15:02 -07:00
|
|
|
Arr::except($updatedAsset->attributesToArray(), $updatedAttributes),
|
2024-09-30 04:42:41 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function customColumnMapping(): void
|
|
|
|
{
|
|
|
|
$faker = ImportFileBuilder::new()->definition();
|
|
|
|
$row = [
|
|
|
|
'assigneeFullName' => $faker['supplierName'],
|
|
|
|
'assigneeEmail' => $faker['manufacturerName'],
|
|
|
|
'assigneeUsername' => $faker['serialNumber'],
|
|
|
|
'category' => $faker['location'],
|
|
|
|
'companyName' => $faker['purchaseCost'],
|
|
|
|
'itemName' => $faker['modelNumber'],
|
|
|
|
'location' => $faker['assigneeUsername'],
|
|
|
|
'manufacturerName' => $faker['status'],
|
|
|
|
'model' => $faker['itemName'],
|
|
|
|
'modelNumber' => $faker['category'],
|
|
|
|
'notes' => $faker['notes'],
|
|
|
|
'purchaseCost' => $faker['model'],
|
|
|
|
'purchaseDate' => $faker['companyName'],
|
|
|
|
'serialNumber' => $faker['tag'],
|
|
|
|
'supplierName' => $faker['purchaseDate'],
|
|
|
|
'status' => $faker['warrantyInMonths'],
|
|
|
|
'tag' => $faker['assigneeEmail'],
|
|
|
|
'warrantyInMonths' => $faker['assigneeFullName'],
|
|
|
|
];
|
|
|
|
|
|
|
|
$importFileBuilder = new ImportFileBuilder([$row]);
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
|
|
|
|
$this->importFileResponse([
|
|
|
|
'import' => $import->id,
|
|
|
|
'column-mappings' => [
|
|
|
|
'Asset Tag' => 'email',
|
|
|
|
'Category' => 'location',
|
|
|
|
'Company' => 'purchase_cost',
|
|
|
|
'Email' => 'manufacturer',
|
|
|
|
'Full Name' => 'supplier',
|
|
|
|
'Item Name' => 'model_number',
|
|
|
|
'Location' => 'username',
|
|
|
|
'Manufacturer' => 'status',
|
|
|
|
'Model name' => 'item_name',
|
|
|
|
'Model Number' => 'category',
|
|
|
|
'Notes' => 'asset_notes',
|
|
|
|
'Purchase Cost' => 'asset_model',
|
|
|
|
'Purchase Date' => 'company',
|
|
|
|
'Serial number' => 'asset_tag',
|
|
|
|
'Status' => 'warranty_months',
|
|
|
|
'Supplier' => 'purchase_date',
|
|
|
|
'Username' => 'serial',
|
|
|
|
'Warranty' => 'full_name',
|
|
|
|
]
|
|
|
|
])->assertOk();
|
|
|
|
|
|
|
|
$asset = Asset::query()
|
|
|
|
->with(['location', 'supplier', 'company', 'assignedAssets', 'defaultLoc', 'assetStatus', 'model.category', 'model.manufacturer'])
|
|
|
|
->where('serial', $row['assigneeUsername'])
|
|
|
|
->sole();
|
|
|
|
|
|
|
|
$assignee = User::query()->find($asset->assigned_to, ['id', 'first_name', 'last_name', 'email', 'username']);
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($row['warrantyInMonths'], "{$assignee->first_name} {$assignee->last_name}");
|
|
|
|
$this->assertEquals($row['tag'], $assignee->email);
|
|
|
|
$this->assertEquals($row['location'], $assignee->username);
|
|
|
|
|
|
|
|
$this->assertEquals($row['modelNumber'], $asset->model->category->name);
|
|
|
|
$this->assertEquals($row['assigneeEmail'], $asset->model->manufacturer->name);
|
|
|
|
$this->assertEquals($row['model'], $asset->name);
|
|
|
|
$this->assertEquals($row['serialNumber'], $asset->asset_tag);
|
|
|
|
$this->assertEquals($row['purchaseCost'], $asset->model->name);
|
|
|
|
$this->assertEquals($row['itemName'], $asset->model->model_number);
|
|
|
|
$this->assertEquals($row['supplierName'], $asset->purchase_date->toDateString());
|
|
|
|
$this->assertEquals($row['companyName'], $asset->purchase_cost);
|
|
|
|
$this->assertEquals($row['manufacturerName'], $asset->assetStatus->name);
|
|
|
|
$this->assertEquals($row['status'], $asset->warranty_months);
|
|
|
|
$this->assertEquals($row['assigneeFullName'], $asset->supplier->name);
|
|
|
|
$this->assertEquals($row['category'], $asset->defaultLoc->name);
|
|
|
|
$this->assertEquals($row['purchaseDate'], $asset->company->name);
|
|
|
|
$this->assertEquals($row['category'], $asset->location->name);
|
|
|
|
$this->assertEquals($row['notes'], $asset->notes);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($asset->asset_eol_date);
|
|
|
|
$this->assertEquals(0, $asset->eol_explicit);
|
|
|
|
$this->assertNull($asset->order_number);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals('', $asset->image);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($asset->user_id);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(1, $asset->physical);
|
|
|
|
$this->assertEquals(0, $asset->archived);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($asset->deprecate);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(0, $asset->requestable);
|
|
|
|
$this->assertEquals(null, $asset->accepted);
|
|
|
|
$this->assertEquals(now()->toDateString(), Carbon::parse($asset->last_checkout)->toDateString());
|
|
|
|
$this->assertEquals(0, $asset->last_checkin);
|
|
|
|
$this->assertEquals(0, $asset->expected_checkin);
|
|
|
|
$this->assertEquals(User::class, $asset->assigned_type);
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->assertNull($asset->last_audit_date);
|
|
|
|
$this->assertNull($asset->next_audit_date);
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals(0, $asset->checkin_counter);
|
|
|
|
$this->assertEquals(1, $asset->checkout_counter);
|
|
|
|
$this->assertEquals(0, $asset->requests_counter);
|
|
|
|
$this->assertEquals(0, $asset->byod);
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function customFields(): void
|
|
|
|
{
|
|
|
|
$macAddress = $this->faker->macAddress;
|
|
|
|
|
|
|
|
$row = ImportFileBuilder::new()->definition();
|
|
|
|
$row['Mac Address'] = $macAddress;
|
|
|
|
|
|
|
|
$importFileBuilder = new ImportFileBuilder([$row]);
|
|
|
|
$customField = CustomField::query()->where('name', 'Mac Address')->firstOrNew();
|
|
|
|
|
|
|
|
if (!$customField->exists) {
|
2024-10-03 13:59:58 -07:00
|
|
|
$customField = CustomField::factory()->macAddress()->create(['db_column' => '_snipeit_mac_address_1']);
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($customField->field_encrypted) {
|
|
|
|
$customField->field_encrypted = 0;
|
|
|
|
$customField->save();
|
|
|
|
}
|
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$newAsset = Asset::query()->where('serial', $importFileBuilder->firstRow()['serialNumber'])->sole();
|
|
|
|
|
2024-10-03 13:15:02 -07:00
|
|
|
$this->assertEquals($macAddress, $newAsset->getAttribute($customField->db_column));
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function willEncryptCustomFields(): void
|
|
|
|
{
|
|
|
|
$macAddress = $this->faker->macAddress;
|
|
|
|
$row = ImportFileBuilder::new()->definition();
|
|
|
|
|
|
|
|
$row['Mac Address'] = $macAddress;
|
|
|
|
|
|
|
|
$importFileBuilder = new ImportFileBuilder([$row]);
|
|
|
|
$customField = CustomField::query()->where('name', 'Mac Address')->firstOrNew();
|
|
|
|
|
|
|
|
if (!$customField->exists) {
|
2024-10-03 13:59:58 -07:00
|
|
|
$customField = CustomField::factory()->macAddress()->create();
|
2024-09-30 04:42:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!$customField->field_encrypted) {
|
|
|
|
$customField->field_encrypted = 1;
|
|
|
|
$customField->save();
|
|
|
|
}
|
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$import = Import::factory()->asset()->create(['file_path' => $importFileBuilder->saveToImportsDirectory()]);
|
2024-09-30 04:42:41 -07:00
|
|
|
|
2024-10-03 13:59:58 -07:00
|
|
|
$this->actingAsForApi(User::factory()->superuser()->create());
|
2024-09-30 04:42:41 -07:00
|
|
|
$this->importFileResponse(['import' => $import->id])->assertOk();
|
|
|
|
|
|
|
|
$asset = Asset::query()->where('serial', $importFileBuilder->firstRow()['serialNumber'])->sole();
|
|
|
|
$encryptedMacAddress = $asset->getAttribute($customField->db_column);
|
|
|
|
|
|
|
|
$this->assertNotEquals($encryptedMacAddress, $macAddress);
|
|
|
|
}
|
|
|
|
}
|