2024-03-20 13:29:25 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Feature\Checkouts;
|
|
|
|
|
|
|
|
use App\Events\CheckoutableCheckedOut;
|
|
|
|
use App\Models\Asset;
|
2024-04-10 15:47:26 -07:00
|
|
|
use App\Models\Company;
|
2024-03-20 17:46:09 -07:00
|
|
|
use App\Models\LicenseSeat;
|
2024-03-20 14:28:27 -07:00
|
|
|
use App\Models\Location;
|
2024-03-20 13:29:25 -07:00
|
|
|
use App\Models\Statuslabel;
|
|
|
|
use App\Models\User;
|
2024-04-10 13:49:07 -07:00
|
|
|
use Illuminate\Support\Carbon;
|
2024-03-20 13:29:25 -07:00
|
|
|
use Illuminate\Support\Facades\Event;
|
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
class AssetCheckoutTest extends TestCase
|
|
|
|
{
|
|
|
|
public function testCheckingOutAssetRequiresCorrectPermission()
|
|
|
|
{
|
|
|
|
$this->actingAs(User::factory()->create())
|
|
|
|
->post(route('hardware.checkout.store', Asset::factory()->create()), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => User::factory()->create()->id,
|
|
|
|
])
|
|
|
|
->assertForbidden();
|
|
|
|
}
|
|
|
|
|
2024-03-20 14:28:27 -07:00
|
|
|
public function testNonExistentAssetCannotBeCheckedOut()
|
|
|
|
{
|
|
|
|
Event::fake([CheckoutableCheckedOut::class]);
|
|
|
|
|
|
|
|
$this->actingAs(User::factory()->checkoutAssets()->create())
|
|
|
|
->post(route('hardware.checkout.store', 1000), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => User::factory()->create()->id,
|
|
|
|
'name' => 'Changed Name',
|
|
|
|
'status_id' => Statuslabel::factory()->readyToDeploy()->create()->id,
|
|
|
|
'checkout_at' => '2024-03-18',
|
|
|
|
'expected_checkin' => '2024-03-28',
|
|
|
|
'note' => 'An awesome note',
|
|
|
|
])
|
|
|
|
->assertSessionHas('error');
|
|
|
|
|
|
|
|
Event::assertNotDispatched(CheckoutableCheckedOut::class);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAssetNotAvailableForCheckoutCannotBeCheckedOut()
|
|
|
|
{
|
|
|
|
Event::fake([CheckoutableCheckedOut::class]);
|
|
|
|
|
|
|
|
$asset = Asset::factory()->assignedToUser()->create();
|
|
|
|
|
|
|
|
$this->actingAs(User::factory()->checkoutAssets()->create())
|
|
|
|
->post(route('hardware.checkout.store', $asset), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => User::factory()->create()->id,
|
|
|
|
'name' => 'Changed Name',
|
|
|
|
'status_id' => Statuslabel::factory()->readyToDeploy()->create()->id,
|
|
|
|
'checkout_at' => '2024-03-18',
|
|
|
|
'expected_checkin' => '2024-03-28',
|
|
|
|
'note' => 'An awesome note',
|
|
|
|
])
|
|
|
|
->assertSessionHas('error');
|
|
|
|
|
|
|
|
Event::assertNotDispatched(CheckoutableCheckedOut::class);
|
|
|
|
}
|
|
|
|
|
2024-03-20 13:29:25 -07:00
|
|
|
public function testValidationWhenCheckingOutAsset()
|
|
|
|
{
|
|
|
|
$this->actingAs(User::factory()->create())
|
|
|
|
->post(route('hardware.checkout.store', Asset::factory()->create()), [
|
2024-04-10 14:02:25 -07:00
|
|
|
'status_id' => 'does-not-exist',
|
|
|
|
'checkout_at' => 'invalid-date',
|
|
|
|
'expected_checkin' => 'invalid-date',
|
2024-03-20 13:29:25 -07:00
|
|
|
])
|
2024-04-10 14:02:25 -07:00
|
|
|
->assertSessionHasErrors([
|
|
|
|
'assigned_user',
|
|
|
|
'assigned_asset',
|
|
|
|
'assigned_location',
|
|
|
|
'status_id',
|
|
|
|
'checkout_to_type',
|
|
|
|
'checkout_at',
|
|
|
|
'expected_checkin',
|
|
|
|
]);
|
2024-03-20 13:29:25 -07:00
|
|
|
}
|
|
|
|
|
2024-04-10 13:29:31 -07:00
|
|
|
public function checkoutTargets(): array
|
2024-04-10 13:28:52 -07:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
'User' => [function () {
|
|
|
|
$userLocation = Location::factory()->create();
|
|
|
|
$user = User::factory()->for($userLocation)->create();
|
|
|
|
|
|
|
|
return [
|
|
|
|
'checkout_type' => 'user',
|
|
|
|
'target' => $user,
|
|
|
|
'expected_location' => $userLocation,
|
|
|
|
];
|
|
|
|
}],
|
|
|
|
'Asset without location set' => [function () {
|
|
|
|
$rtdLocation = Location::factory()->create();
|
|
|
|
$asset = Asset::factory()->for($rtdLocation, 'defaultLoc')->create(['location_id' => null]);
|
|
|
|
|
|
|
|
return [
|
|
|
|
'checkout_type' => 'asset',
|
|
|
|
'target' => $asset,
|
|
|
|
'expected_location' => $rtdLocation,
|
|
|
|
];
|
|
|
|
}],
|
|
|
|
'Asset with location set' => [function () {
|
|
|
|
$rtdLocation = Location::factory()->create();
|
|
|
|
$location = Location::factory()->create();
|
|
|
|
$asset = Asset::factory()->for($location)->for($rtdLocation, 'defaultLoc')->create();
|
|
|
|
|
|
|
|
return [
|
|
|
|
'checkout_type' => 'asset',
|
|
|
|
'target' => $asset,
|
|
|
|
'expected_location' => $location,
|
|
|
|
];
|
|
|
|
}],
|
|
|
|
'Location' => [function () {
|
|
|
|
$location = Location::factory()->create();
|
|
|
|
|
|
|
|
return [
|
|
|
|
'checkout_type' => 'location',
|
|
|
|
'target' => $location,
|
|
|
|
'expected_location' => $location,
|
|
|
|
];
|
|
|
|
}],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-04-10 13:29:31 -07:00
|
|
|
/** @dataProvider checkoutTargets */
|
|
|
|
public function testAnAssetCanBeCheckedOut($data)
|
2024-03-20 13:29:25 -07:00
|
|
|
{
|
|
|
|
Event::fake([CheckoutableCheckedOut::class]);
|
|
|
|
|
2024-04-10 13:28:52 -07:00
|
|
|
['checkout_type' => $type, 'target' => $target, 'expected_location' => $expectedLocation] = $data();
|
|
|
|
|
2024-03-20 14:28:27 -07:00
|
|
|
$originalStatus = Statuslabel::factory()->readyToDeploy()->create();
|
|
|
|
$updatedStatus = Statuslabel::factory()->readyToDeploy()->create();
|
|
|
|
$asset = Asset::factory()->create(['status_id' => $originalStatus->id]);
|
2024-03-20 13:44:26 -07:00
|
|
|
$admin = User::factory()->checkoutAssets()->create();
|
2024-03-20 13:29:25 -07:00
|
|
|
|
|
|
|
$this->actingAs($admin)
|
|
|
|
->post(route('hardware.checkout.store', $asset), [
|
2024-04-10 13:28:52 -07:00
|
|
|
'checkout_to_type' => $type,
|
|
|
|
'assigned_' . $type => $target->id,
|
2024-03-20 13:29:25 -07:00
|
|
|
'name' => 'Changed Name',
|
2024-03-20 14:28:27 -07:00
|
|
|
'status_id' => $updatedStatus->id,
|
2024-03-20 13:29:25 -07:00
|
|
|
'checkout_at' => '2024-03-18',
|
|
|
|
'expected_checkin' => '2024-03-28',
|
|
|
|
'note' => 'An awesome note',
|
|
|
|
]);
|
|
|
|
|
2024-03-20 14:28:27 -07:00
|
|
|
$asset->refresh();
|
2024-04-10 13:28:52 -07:00
|
|
|
$this->assertTrue($asset->assignedTo()->is($target));
|
|
|
|
$this->assertTrue($asset->location->is($expectedLocation));
|
2024-04-10 11:43:50 -07:00
|
|
|
$this->assertEquals('Changed Name', $asset->name);
|
2024-03-20 14:28:27 -07:00
|
|
|
$this->assertTrue($asset->assetstatus->is($updatedStatus));
|
2024-04-10 11:43:50 -07:00
|
|
|
$this->assertEquals('2024-03-18 00:00:00', $asset->last_checkout);
|
|
|
|
$this->assertEquals('2024-03-28 00:00:00', (string)$asset->expected_checkin);
|
2024-03-20 17:46:09 -07:00
|
|
|
|
2024-04-10 11:52:59 -07:00
|
|
|
Event::assertDispatched(CheckoutableCheckedOut::class, 1);
|
2024-04-10 13:28:52 -07:00
|
|
|
Event::assertDispatched(function (CheckoutableCheckedOut $event) use ($admin, $asset, $target) {
|
2024-03-20 13:44:26 -07:00
|
|
|
return $event->checkoutable->is($asset)
|
2024-04-10 13:28:52 -07:00
|
|
|
&& $event->checkedOutTo->is($target)
|
2024-03-20 13:44:26 -07:00
|
|
|
&& $event->checkedOutBy->is($admin)
|
|
|
|
&& $event->note === 'An awesome note';
|
|
|
|
});
|
2024-03-20 13:29:25 -07:00
|
|
|
}
|
|
|
|
|
2024-04-10 15:47:26 -07:00
|
|
|
public function testCannotCheckoutAcrossCompaniesWhenFullCompanySupportEnabled()
|
|
|
|
{
|
|
|
|
Event::fake([CheckoutableCheckedOut::class]);
|
|
|
|
|
|
|
|
$this->settings->enableMultipleFullCompanySupport();
|
|
|
|
|
|
|
|
$assetCompany = Company::factory()->create();
|
|
|
|
$userCompany = Company::factory()->create();
|
|
|
|
|
|
|
|
$user = User::factory()->for($userCompany)->create();
|
|
|
|
$asset = Asset::factory()->for($assetCompany)->create();
|
|
|
|
|
|
|
|
$this->actingAs(User::factory()->superuser()->create())
|
|
|
|
->post(route('hardware.checkout.store', $asset), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => $user->id,
|
|
|
|
])
|
|
|
|
->assertRedirect(route('hardware.checkout.store', $asset));
|
|
|
|
|
|
|
|
Event::assertNotDispatched(CheckoutableCheckedOut::class);
|
|
|
|
}
|
|
|
|
|
2024-03-20 14:28:27 -07:00
|
|
|
public function testLicenseSeatsAreAssignedToUserUponCheckout()
|
|
|
|
{
|
2024-03-20 17:46:09 -07:00
|
|
|
$asset = Asset::factory()->create();
|
|
|
|
$seat = LicenseSeat::factory()->assignedToAsset($asset)->create();
|
|
|
|
$user = User::factory()->create();
|
|
|
|
|
|
|
|
$this->assertFalse($user->licenses->contains($seat->license));
|
|
|
|
|
|
|
|
$this->actingAs(User::factory()->checkoutAssets()->create())
|
|
|
|
->post(route('hardware.checkout.store', $asset), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => $user->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertTrue($user->fresh()->licenses->contains($seat->license));
|
2024-03-20 14:28:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testLastCheckoutUsesCurrentDateIfNotProvided()
|
|
|
|
{
|
2024-04-10 13:49:07 -07:00
|
|
|
$asset = Asset::factory()->create();
|
|
|
|
|
|
|
|
$this->actingAs(User::factory()->checkoutAssets()->create())
|
|
|
|
->post(route('hardware.checkout.store', $asset), [
|
|
|
|
'checkout_to_type' => 'user',
|
|
|
|
'assigned_user' => User::factory()->create()->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$asset->refresh();
|
|
|
|
|
|
|
|
// It's possible that this can be properly mocked in the future:
|
|
|
|
// https://laravel.com/docs/8.x/mocking#interacting-with-time
|
|
|
|
$this->assertTrue(Carbon::parse($asset->last_checkout)->diffInSeconds(now()) < 2);
|
2024-03-20 14:28:27 -07:00
|
|
|
}
|
2024-03-20 13:29:25 -07:00
|
|
|
}
|