mirror of
https://github.com/snipe/snipe-it.git
synced 2024-12-31 16:37:27 -08:00
Merge remote-tracking branch 'origin/develop'
This commit is contained in:
commit
c536dbc741
|
@ -26,11 +26,19 @@ class StoreAssetRequest extends ImageUploadRequest
|
||||||
|
|
||||||
public function prepareForValidation(): void
|
public function prepareForValidation(): void
|
||||||
{
|
{
|
||||||
|
// Guard against users passing in an array for company_id instead of an integer.
|
||||||
|
// If the company_id is not an integer then we simply use what was
|
||||||
|
// provided to be caught by model level validation later.
|
||||||
|
// The use of is_numeric accounts for 1 and '1'.
|
||||||
|
$idForCurrentUser = is_numeric($this->company_id)
|
||||||
|
? Company::getIdForCurrentUser($this->company_id)
|
||||||
|
: $this->company_id;
|
||||||
|
|
||||||
$this->parseLastAuditDate();
|
$this->parseLastAuditDate();
|
||||||
|
|
||||||
$this->merge([
|
$this->merge([
|
||||||
'asset_tag' => $this->asset_tag ?? Asset::autoincrement_asset(),
|
'asset_tag' => $this->asset_tag ?? Asset::autoincrement_asset(),
|
||||||
'company_id' => Company::getIdForCurrentUser($this->company_id),
|
'company_id' => $idForCurrentUser,
|
||||||
'assigned_to' => $assigned_to ?? null,
|
'assigned_to' => $assigned_to ?? null,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -116,7 +116,7 @@ final class Company extends SnipeModel
|
||||||
if ($current_user->company_id != null) {
|
if ($current_user->company_id != null) {
|
||||||
return $current_user->company_id;
|
return $current_user->company_id;
|
||||||
} else {
|
} else {
|
||||||
return static::getIdFromInput($unescaped_input);
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -179,7 +179,8 @@
|
||||||
@if ($accessory->image!='')
|
@if ($accessory->image!='')
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-12 text-center" style="padding-bottom: 15px;">
|
<div class="col-md-12 text-center" style="padding-bottom: 15px;">
|
||||||
<a href="{{ Storage::disk('public')->url('accessories/'.e($accessory->image)) }}" data-toggle="lightbox"><img src="{{ Storage::disk('public')->url('accessories/'.e($accessory->image)) }}" class="img-responsive img-thumbnail" alt="{{ $accessory->name }}"></a>
|
<a href="{{ Storage::disk('public')->url('accessories/'.e($accessory->image)) }}" data-toggle="lightbox" data-type="image">
|
||||||
|
<img src="{{ Storage::disk('public')->url('accessories/'.e($accessory->image)) }}" class="img-responsive img-thumbnail" alt="{{ $accessory->name }}"></a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@endif
|
@endif
|
||||||
|
|
|
@ -161,7 +161,7 @@
|
||||||
<div class="col-md-3">
|
<div class="col-md-3">
|
||||||
@if ($component->image!='')
|
@if ($component->image!='')
|
||||||
<div class="col-md-12 text-center" style="padding-bottom: 15px;">
|
<div class="col-md-12 text-center" style="padding-bottom: 15px;">
|
||||||
<a href="{{ Storage::disk('public')->url('components/'.e($component->image)) }}" data-toggle="lightbox">
|
<a href="{{ Storage::disk('public')->url('components/'.e($component->image)) }}" data-toggle="lightbox" data-type="image">
|
||||||
<img src="{{ Storage::disk('public')->url('components/'.e($component->image)) }}" class="img-responsive img-thumbnail" alt="{{ $component->name }}"></a>
|
<img src="{{ Storage::disk('public')->url('components/'.e($component->image)) }}" class="img-responsive img-thumbnail" alt="{{ $component->name }}"></a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -89,7 +89,7 @@
|
||||||
|
|
||||||
@if ($consumable->image!='')
|
@if ($consumable->image!='')
|
||||||
<div class="col-md-12 text-center" style="padding-bottom: 20px;">
|
<div class="col-md-12 text-center" style="padding-bottom: 20px;">
|
||||||
<a href="{{ Storage::disk('public')->url('consumables/'.e($consumable->image)) }}" data-toggle="lightbox">
|
<a href="{{ Storage::disk('public')->url('consumables/'.e($consumable->image)) }}" data-toggle="lightbox" data-type="image">
|
||||||
<img src="{{ Storage::disk('public')->url('consumables/'.e($consumable->image)) }}" class="img-responsive img-thumbnail" alt="{{ $consumable->name }}"></a>
|
<img src="{{ Storage::disk('public')->url('consumables/'.e($consumable->image)) }}" class="img-responsive img-thumbnail" alt="{{ $consumable->name }}"></a>
|
||||||
</div>
|
</div>
|
||||||
@endif
|
@endif
|
||||||
|
|
|
@ -168,7 +168,7 @@
|
||||||
<div class="col-md-12 text-center">
|
<div class="col-md-12 text-center">
|
||||||
@if (($asset->image) || (($asset->model) && ($asset->model->image!='')))
|
@if (($asset->image) || (($asset->model) && ($asset->model->image!='')))
|
||||||
<div class="text-center col-md-12" style="padding-bottom: 15px;">
|
<div class="text-center col-md-12" style="padding-bottom: 15px;">
|
||||||
<a href="{{ ($asset->getImageUrl()) ? $asset->getImageUrl() : null }}" data-toggle="lightbox">
|
<a href="{{ ($asset->getImageUrl()) ? $asset->getImageUrl() : null }}" data-toggle="lightbox" data-type="image">
|
||||||
<img src="{{ ($asset->getImageUrl()) ? $asset->getImageUrl() : null }}" class="assetimg img-responsive" alt="{{ $asset->getDisplayNameAttribute() }}">
|
<img src="{{ ($asset->getImageUrl()) ? $asset->getImageUrl() : null }}" class="assetimg img-responsive" alt="{{ $asset->getDisplayNameAttribute() }}">
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Accessories\Ui;
|
||||||
|
|
||||||
|
use App\Models\Accessory;
|
||||||
|
use App\Models\Category;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class CreateAccessoryWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$this->actingAs($actor)
|
||||||
|
->post(route('accessories.store'), [
|
||||||
|
'redirect_option' => 'index',
|
||||||
|
'name' => 'My Cool Accessory',
|
||||||
|
'qty' => '1',
|
||||||
|
'category_id' => Category::factory()->create()->id,
|
||||||
|
'company_id' => $company->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$accessory = Accessory::withoutGlobalScopes()->where([
|
||||||
|
'name' => 'My Cool Accessory',
|
||||||
|
])->sole();
|
||||||
|
|
||||||
|
$assertions($accessory);
|
||||||
|
}
|
||||||
|
}
|
|
@ -560,6 +560,9 @@ class StoreAssetTest extends TestCase
|
||||||
$this->assertTrue($asset->assignedAssets()->find($response['payload']['id'])->is($apiAsset));
|
$this->assertTrue($asset->assignedAssets()->find($response['payload']['id'])->is($apiAsset));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @link https://app.shortcut.com/grokability/story/24475
|
||||||
|
*/
|
||||||
public function testCompanyIdNeedsToBeInteger()
|
public function testCompanyIdNeedsToBeInteger()
|
||||||
{
|
{
|
||||||
$this->actingAsForApi(User::factory()->createAssets()->create())
|
$this->actingAsForApi(User::factory()->createAssets()->create())
|
||||||
|
|
|
@ -0,0 +1,58 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Assets\Api;
|
||||||
|
|
||||||
|
use App\Models\Asset;
|
||||||
|
use App\Models\AssetModel;
|
||||||
|
use App\Models\Statuslabel;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class StoreAssetWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @link https://github.com/snipe/snipe-it/issues/15654
|
||||||
|
*/
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$response = $this->actingAsForApi($actor)
|
||||||
|
->postJson(route('api.assets.store'), [
|
||||||
|
'asset_tag' => 'random_string',
|
||||||
|
'company_id' => $company->id,
|
||||||
|
'model_id' => AssetModel::factory()->create()->id,
|
||||||
|
'status_id' => Statuslabel::factory()->readyToDeploy()->create()->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$asset = Asset::withoutGlobalScopes()->findOrFail($response['payload']['id']);
|
||||||
|
|
||||||
|
$assertions($asset);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testHandlesCompanyIdBeingString($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$response = $this->actingAsForApi($actor)
|
||||||
|
->postJson(route('api.assets.store'), [
|
||||||
|
'asset_tag' => 'random_string',
|
||||||
|
'company_id' => (string) $company->id,
|
||||||
|
'model_id' => AssetModel::factory()->create()->id,
|
||||||
|
'status_id' => Statuslabel::factory()->readyToDeploy()->create()->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$asset = Asset::withoutGlobalScopes()->findOrFail($response['payload']['id']);
|
||||||
|
|
||||||
|
$assertions($asset);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Assets\Ui;
|
||||||
|
|
||||||
|
use App\Models\Asset;
|
||||||
|
use App\Models\AssetModel;
|
||||||
|
use App\Models\Statuslabel;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class StoreAssetWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$this->actingAs($actor)
|
||||||
|
->post(route('hardware.store'), [
|
||||||
|
'asset_tags' => ['1' => '1234'],
|
||||||
|
'model_id' => AssetModel::factory()->create()->id,
|
||||||
|
'status_id' => Statuslabel::factory()->create()->id,
|
||||||
|
'company_id' => $company->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$asset = Asset::where('asset_tag', '1234')->sole();
|
||||||
|
|
||||||
|
$assertions($asset);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Components\Ui;
|
||||||
|
|
||||||
|
use App\Models\Category;
|
||||||
|
use App\Models\Component;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class StoreComponentWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$this->actingAs($actor)
|
||||||
|
->post(route('components.store'), [
|
||||||
|
'name' => 'My Cool Component',
|
||||||
|
'qty' => '1',
|
||||||
|
'category_id' => Category::factory()->create()->id,
|
||||||
|
'company_id' => $company->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$component = Component::where('name', 'My Cool Component')->sole();
|
||||||
|
|
||||||
|
$assertions($component);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,33 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Consumables\Ui;
|
||||||
|
|
||||||
|
use App\Models\Category;
|
||||||
|
use App\Models\Consumable;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class StoreConsumableWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$this->actingAs($actor)
|
||||||
|
->post(route('consumables.store'), [
|
||||||
|
'name' => 'My Cool Consumable',
|
||||||
|
'category_id' => Category::factory()->forConsumables()->create()->id,
|
||||||
|
'company_id' => $company->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$consumable = Consumable::where('name', 'My Cool Consumable')->sole();
|
||||||
|
|
||||||
|
$assertions($consumable);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Licenses\Ui;
|
||||||
|
|
||||||
|
use App\Models\Category;
|
||||||
|
use App\Models\License;
|
||||||
|
use PHPUnit\Framework\Attributes\DataProvider;
|
||||||
|
use Tests\Support\ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
use Tests\TestCase;
|
||||||
|
|
||||||
|
class StoreLicenseWithFullMultipleCompanySupportTest extends TestCase
|
||||||
|
{
|
||||||
|
use ProvidesDataForFullMultipleCompanySupportTesting;
|
||||||
|
|
||||||
|
#[DataProvider('dataForFullMultipleCompanySupportTesting')]
|
||||||
|
public function testAdheresToFullMultipleCompaniesSupportScoping($data)
|
||||||
|
{
|
||||||
|
['actor' => $actor, 'company_attempting_to_associate' => $company, 'assertions' => $assertions] = $data();
|
||||||
|
|
||||||
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
|
$this->actingAs($actor)
|
||||||
|
->post(route('licenses.store'), [
|
||||||
|
'name' => 'My Cool License',
|
||||||
|
'seats' => '1',
|
||||||
|
'category_id' => Category::factory()->forLicenses()->create()->id,
|
||||||
|
'company_id' => $company->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$license = License::where('name', 'My Cool License')->sole();
|
||||||
|
|
||||||
|
$assertions($license);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,70 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Support;
|
||||||
|
|
||||||
|
use App\Models\Company;
|
||||||
|
use App\Models\User;
|
||||||
|
use Generator;
|
||||||
|
|
||||||
|
trait ProvidesDataForFullMultipleCompanySupportTesting
|
||||||
|
{
|
||||||
|
public static function dataForFullMultipleCompanySupportTesting(): Generator
|
||||||
|
{
|
||||||
|
yield "User in a company should result in user's company_id being used" => [
|
||||||
|
function () {
|
||||||
|
$jedi = Company::factory()->create();
|
||||||
|
$sith = Company::factory()->create();
|
||||||
|
$luke = User::factory()->for($jedi)
|
||||||
|
->createAccessories()
|
||||||
|
->createAssets()
|
||||||
|
->createComponents()
|
||||||
|
->createConsumables()
|
||||||
|
->createLicenses()
|
||||||
|
->create();
|
||||||
|
|
||||||
|
return [
|
||||||
|
'actor' => $luke,
|
||||||
|
'company_attempting_to_associate' => $sith,
|
||||||
|
'assertions' => function ($model) use ($jedi) {
|
||||||
|
self::assertEquals($jedi->id, $model->company_id);
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "User without a company should result in company_id being null" => [
|
||||||
|
function () {
|
||||||
|
$userInNoCompany = User::factory()
|
||||||
|
->createAccessories()
|
||||||
|
->createAssets()
|
||||||
|
->createComponents()
|
||||||
|
->createConsumables()
|
||||||
|
->createLicenses()
|
||||||
|
->create(['company_id' => null]);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'actor' => $userInNoCompany,
|
||||||
|
'company_attempting_to_associate' => Company::factory()->create(),
|
||||||
|
'assertions' => function ($model) {
|
||||||
|
self::assertNull($model->company_id);
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Super-User assigning across companies should result in company_id being set to what was provided" => [
|
||||||
|
function () {
|
||||||
|
$superUser = User::factory()->superuser()->create(['company_id' => null]);
|
||||||
|
$company = Company::factory()->create();
|
||||||
|
|
||||||
|
return [
|
||||||
|
'actor' => $superUser,
|
||||||
|
'company_attempting_to_associate' => $company,
|
||||||
|
'assertions' => function ($model) use ($company) {
|
||||||
|
self::assertEquals($model->company_id, $company->id);
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -32,11 +32,11 @@ class GetIdForCurrentUserTest extends TestCase
|
||||||
$this->assertEquals(2000, Company::getIdForCurrentUser(1000));
|
$this->assertEquals(2000, Company::getIdForCurrentUser(1000));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testReturnsProvidedValueForNonSuperUserWithoutCompanyIdWhenFullCompanySupportEnabled()
|
public function testReturnsNullForNonSuperUserWithoutCompanyIdWhenFullCompanySupportEnabled()
|
||||||
{
|
{
|
||||||
$this->settings->enableMultipleFullCompanySupport();
|
$this->settings->enableMultipleFullCompanySupport();
|
||||||
|
|
||||||
$this->actingAs(User::factory()->create(['company_id' => null]));
|
$this->actingAs(User::factory()->create(['company_id' => null]));
|
||||||
$this->assertEquals(1000, Company::getIdForCurrentUser(1000));
|
$this->assertNull(Company::getIdForCurrentUser(1000));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue