Skip to content

Commit

Permalink
Merge pull request #15851 from marcusmoore/testing/api-component-checkin
Browse files Browse the repository at this point in the history
Fixed bug in component checkins via api
  • Loading branch information
snipe authored Nov 21, 2024
2 parents 85c1b0e + f932a4f commit 8b38dc8
Show file tree
Hide file tree
Showing 2 changed files with 173 additions and 20 deletions.
29 changes: 9 additions & 20 deletions app/Http/Controllers/Api/ComponentsController.php
Original file line number Diff line number Diff line change
Expand Up @@ -309,27 +309,21 @@ public function checkout(Request $request, $componentId) : JsonResponse
public function checkin(Request $request, $component_asset_id) : JsonResponse
{
if ($component_assets = DB::table('components_assets')->find($component_asset_id)) {

if (is_null($component = Component::find($component_assets->component_id))) {

return response()->json(Helper::formatStandardApiResponse('error', null, trans('admin/components/message.not_found')));
}

$this->authorize('checkin', $component);

$max_to_checkin = $component_assets->assigned_qty;

if ($max_to_checkin > 1) {

$validator = Validator::make($request->all(), [
"checkin_qty" => "required|numeric|between:1,$max_to_checkin"
]);

if ($validator->fails()) {
return response()->json(Helper::formatStandardApiResponse('error', null, 'Checkin quantity must be between 1 and '.$max_to_checkin));
}
$validator = Validator::make($request->all(), [
"checkin_qty" => "required|numeric|between:1,$max_to_checkin"
]);

if ($validator->fails()) {
return response()->json(Helper::formatStandardApiResponse('error', null, 'Checkin quantity must be between 1 and ' . $max_to_checkin));
}


// Validation passed, so let's figure out what we have to do here.
$qty_remaining_in_checkout = ($component_assets->assigned_qty - (int)$request->input('checkin_qty', 1));
Expand All @@ -339,28 +333,23 @@ public function checkin(Request $request, $component_asset_id) : JsonResponse
$component_assets->assigned_qty = $qty_remaining_in_checkout;

Log::debug($component_asset_id.' - '.$qty_remaining_in_checkout.' remaining in record '.$component_assets->id);

DB::table('components_assets')->where('id',
$component_asset_id)->update(['assigned_qty' => $qty_remaining_in_checkout]);

DB::table('components_assets')->where('id', $component_asset_id)->update(['assigned_qty' => $qty_remaining_in_checkout]);

// If the checked-in qty is exactly the same as the assigned_qty,
// we can simply delete the associated components_assets record
if ($qty_remaining_in_checkout == 0) {
if ($qty_remaining_in_checkout === 0) {
DB::table('components_assets')->where('id', '=', $component_asset_id)->delete();
}


$asset = Asset::find($component_assets->asset_id);

event(new CheckoutableCheckedIn($component, $asset, auth()->user(), $request->input('note'), Carbon::now()));

return response()->json(Helper::formatStandardApiResponse('success', null, trans('admin/components/message.checkin.success')));

}

return response()->json(Helper::formatStandardApiResponse('error', null, 'No matching checkouts for that component join record'));


}

}
164 changes: 164 additions & 0 deletions tests/Feature/Checkins/Api/ComponentCheckinTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
<?php

namespace Tests\Feature\Checkins\Api;

use App\Events\CheckoutableCheckedIn;
use App\Models\Asset;
use App\Models\Company;
use App\Models\Component;
use App\Models\User;
use Illuminate\Support\Facades\Event;
use Tests\Concerns\TestsFullMultipleCompaniesSupport;
use Tests\Concerns\TestsPermissionsRequirement;
use Tests\TestCase;

class ComponentCheckinTest extends TestCase implements TestsFullMultipleCompaniesSupport, TestsPermissionsRequirement
{
public function testRequiresPermission()
{
$component = Component::factory()->checkedOutToAsset()->create();

$this->actingAsForApi(User::factory()->create())
->postJson(route('api.components.checkin', $component->assets->first()->pivot->id))
->assertForbidden();
}

public function testHandlesNonExistentPivotId()
{
$this->actingAsForApi(User::factory()->checkinComponents()->create())
->postJson(route('api.components.checkin', 1000), [
'checkin_qty' => 1,
])
->assertOk()
->assertStatusMessageIs('error');
}

public function testHandlesNonExistentComponent()
{
$component = Component::factory()->checkedOutToAsset()->create();
$pivotId = $component->assets->first()->pivot->id;
$component->delete();

$this->actingAsForApi(User::factory()->checkinComponents()->create())
->postJson(route('api.components.checkin', $pivotId), [
'checkin_qty' => 1,
])
->assertOk()
->assertStatusMessageIs('error');
}

public function testCannotCheckinMoreThanCheckedOut()
{
$component = Component::factory()->checkedOutToAsset()->create();

$pivot = $component->assets->first()->pivot;
$pivot->update(['assigned_qty' => 1]);

$this->actingAsForApi(User::factory()->checkinComponents()->create())
->postJson(route('api.components.checkin', $component->assets->first()->pivot->id), [
'checkin_qty' => 3,
])
->assertOk()
->assertStatusMessageIs('error');
}

public function testCanCheckinComponent()
{
Event::fake([CheckoutableCheckedIn::class]);

$user = User::factory()->checkinComponents()->create();

$component = Component::factory()->checkedOutToAsset()->create();
$pivot = $component->assets->first()->pivot;
$pivot->update(['assigned_qty' => 3]);


$this->actingAsForApi($user)
->postJson(route('api.components.checkin', $component->assets->first()->pivot->id), [
'checkin_qty' => 2,
'note' => 'my note',
])
->assertOk()
->assertStatusMessageIs('success');

$this->assertEquals(1, $component->fresh()->assets->first()->pivot->assigned_qty);

Event::assertDispatched(function (CheckoutableCheckedIn $event) use ($user, $component) {
return $event->checkoutable->is($component)
&& $event->checkedOutTo->is($component->assets->first())
&& $event->checkedInBy->is($user)
&& $event->note === 'my note';
});
}

public function testCheckingInEntireAssignedQuantityClearsThePivotRecordFromTheDatabase()
{
Event::fake([CheckoutableCheckedIn::class]);

$user = User::factory()->checkinComponents()->create();

$component = Component::factory()->checkedOutToAsset()->create();
$pivot = $component->assets->first()->pivot;
$pivot->update(['assigned_qty' => 3]);

$this->actingAsForApi($user)
->postJson(route('api.components.checkin', $component->assets->first()->pivot->id), [
'checkin_qty' => 3,
'note' => 'my note',
])
->assertOk()
->assertStatusMessageIs('success');

$this->assertEmpty($component->fresh()->assets);

Event::assertDispatched(function (CheckoutableCheckedIn $event) use ($user, $component) {
return $event->checkoutable->is($component)
&& $event->checkedOutTo->is($component->assets->first())
&& $event->checkedInBy->is($user)
&& $event->note === 'my note';
});
}

public function testAdheresToFullMultipleCompaniesSupportScoping()
{
$this->settings->enableMultipleFullCompanySupport();

[$companyA, $companyB] = Company::factory()->count(2)->create();

$componentInCompanyA = Component::factory()->for($companyA)->checkedOutToAsset()->create();
$userInCompanyB = User::factory()->for($companyB)->create();
$pivotId = $componentInCompanyA->assets->first()->pivot->id;

$this->actingAsForApi($userInCompanyB)
->postJson(route('api.components.checkin', $pivotId), [
'checkin_qty' => 1,
])
->assertOk()
->assertStatusMessageIs('error');
}

public function testCheckinIsLogged()
{
$user = User::factory()->checkinComponents()->create();

$component = Component::factory()->checkedOutToAsset()->create();
$pivot = $component->assets->first()->pivot;
$pivot->update(['assigned_qty' => 3]);

$this->actingAsForApi($user)
->postJson(route('api.components.checkin', $component->assets->first()->pivot->id), [
'checkin_qty' => 3,
'note' => 'my note',
]);

$this->assertDatabaseHas('action_logs', [
'created_by' => $user->id,
'action_type' => 'checkin from',
'target_id' => $component->assets->first()->id,
'target_type' => Asset::class,
'note' => 'my note',
'item_id' => $component->id,
'item_type' => Component::class,
]);
}
}

0 comments on commit 8b38dc8

Please sign in to comment.