Merge branch 'develop' into acceptance-reminder-subject

This commit is contained in:
Marcus Moore 2025-02-24 11:55:50 -08:00
commit e88bba51bb
No known key found for this signature in database
1086 changed files with 13237 additions and 5383 deletions

View file

@ -25,9 +25,9 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
php-version: php-version:
- "8.1"
- "8.2" - "8.2"
- "8.3" - "8.3"
- "8.4"
name: PHP ${{ matrix.php-version }} name: PHP ${{ matrix.php-version }}

View file

@ -21,9 +21,10 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
php-version: php-version:
- "8.1"
- "8.2" - "8.2"
- "8.3" - "8.3"
- "8.4"
name: PHP ${{ matrix.php-version }} name: PHP ${{ matrix.php-version }}

View file

@ -15,7 +15,7 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
php-version: php-version:
- "8.1.1" - "8.3"
name: PHP ${{ matrix.php-version }} name: PHP ${{ matrix.php-version }}

View file

@ -4,7 +4,7 @@
"DOC3": "Please don't rely on these versions for planning upgrades unless you've fetched the most recent version", "DOC3": "Please don't rely on these versions for planning upgrades unless you've fetched the most recent version",
"DOC4": "You should really just ignore it and run upgrade.php. Really", "DOC4": "You should really just ignore it and run upgrade.php. Really",
"php_min_version": "8.1.0", "php_min_version": "8.1.0",
"php_max_major_minor": "8.3", "php_max_major_minor": "8.4",
"php_max_wontwork": "8.4.0", "php_max_wontwork": "8.5.0",
"current_snipeit_version": "7.0" "current_snipeit_version": "8.0"
} }

View file

@ -1,4 +1,4 @@
FROM ubuntu:22.04 FROM ubuntu:24.04
LABEL maintainer="Brady Wetherington <bwetherington@grokability.com>" LABEL maintainer="Brady Wetherington <bwetherington@grokability.com>"
# No need to add `apt-get clean` here, reference: # No need to add `apt-get clean` here, reference:
@ -14,16 +14,16 @@ RUN export DEBIAN_FRONTEND=noninteractive; \
apt-utils \ apt-utils \
apache2 \ apache2 \
apache2-bin \ apache2-bin \
libapache2-mod-php8.1 \ libapache2-mod-php8.3 \
php8.1-curl \ php8.3-curl \
php8.1-ldap \ php8.3-ldap \
php8.1-mysql \ php8.3-mysql \
php8.1-gd \ php8.3-gd \
php8.1-xml \ php8.3-xml \
php8.1-mbstring \ php8.3-mbstring \
php8.1-zip \ php8.3-zip \
php8.1-bcmath \ php8.3-bcmath \
php8.1-redis \ php8.3-redis \
php-memcached \ php-memcached \
patch \ patch \
curl \ curl \
@ -41,7 +41,7 @@ libc-dev \
libldap-common \ libldap-common \
pkg-config \ pkg-config \
libmcrypt-dev \ libmcrypt-dev \
php8.1-dev \ php8.3-dev \
ca-certificates \ ca-certificates \
unzip \ unzip \
dnsutils \ dnsutils \
@ -53,16 +53,16 @@ RUN php go-pear.phar
RUN pecl install mcrypt RUN pecl install mcrypt
RUN bash -c "echo extension=/usr/lib/php/20210902/mcrypt.so > /etc/php/8.1/mods-available/mcrypt.ini" RUN bash -c "echo extension=/usr/lib/php/20210902/mcrypt.so > /etc/php/8.3/mods-available/mcrypt.ini"
RUN phpenmod mcrypt RUN phpenmod mcrypt
RUN phpenmod gd RUN phpenmod gd
RUN phpenmod bcmath RUN phpenmod bcmath
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.1/apache2/php.ini RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.3/apache2/php.ini
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.1/cli/php.ini RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.3/cli/php.ini
RUN useradd -m --uid 1000 --gid 50 docker RUN useradd -m --uid 10000 --gid 50 docker
RUN echo export APACHE_RUN_USER=docker >> /etc/apache2/envvars RUN echo export APACHE_RUN_USER=docker >> /etc/apache2/envvars
RUN echo export APACHE_RUN_GROUP=staff >> /etc/apache2/envvars RUN echo export APACHE_RUN_GROUP=staff >> /etc/apache2/envvars

View file

@ -2,13 +2,13 @@
namespace App\Console\Commands; namespace App\Console\Commands;
use App\Mail\ExpiringAssetsMail;
use App\Mail\ExpiringLicenseMail;
use App\Models\Asset; use App\Models\Asset;
use App\Models\License; use App\Models\License;
use App\Models\Recipients\AlertRecipient;
use App\Models\Setting; use App\Models\Setting;
use App\Notifications\ExpiringAssetsNotification;
use App\Notifications\ExpiringLicenseNotification;
use Illuminate\Console\Command; use Illuminate\Console\Command;
use Illuminate\Support\Facades\Mail;
class SendExpirationAlerts extends Command class SendExpirationAlerts extends Command
{ {
@ -47,22 +47,22 @@ class SendExpirationAlerts extends Command
if (($settings->alert_email != '') && ($settings->alerts_enabled == 1)) { if (($settings->alert_email != '') && ($settings->alerts_enabled == 1)) {
// Send a rollup to the admin, if settings dictate // Send a rollup to the admin, if settings dictate
$recipients = collect(explode(',', $settings->alert_email))->map(function ($item, $key) { $recipients = collect(explode(',', $settings->alert_email))
return new AlertRecipient($item); ->map(fn($item) => trim($item)) // Trim each email
}); ->all();
// Expiring Assets // Expiring Assets
$assets = Asset::getExpiringWarrantee($threshold); $assets = Asset::getExpiringWarrantee($threshold);
if ($assets->count() > 0) { if ($assets->count() > 0) {
$this->info(trans_choice('mail.assets_warrantee_alert', $assets->count(), ['count' => $assets->count(), 'threshold' => $threshold])); $this->info(trans_choice('mail.assets_warrantee_alert', $assets->count(), ['count' => $assets->count(), 'threshold' => $threshold]));
\Notification::send($recipients, new ExpiringAssetsNotification($assets, $threshold)); Mail::to($recipients)->send(new ExpiringAssetsMail($assets, $threshold));
} }
// Expiring licenses // Expiring licenses
$licenses = License::getExpiringLicenses($threshold); $licenses = License::getExpiringLicenses($threshold);
if ($licenses->count() > 0) { if ($licenses->count() > 0) {
$this->info(trans_choice('mail.license_expiring_alert', $licenses->count(), ['count' => $licenses->count(), 'threshold' => $threshold])); $this->info(trans_choice('mail.license_expiring_alert', $licenses->count(), ['count' => $licenses->count(), 'threshold' => $threshold]));
\Notification::send($recipients, new ExpiringLicenseNotification($licenses, $threshold)); Mail::to($recipients)->send(new ExpiringLicenseMail($licenses, $threshold));
} }
} else { } else {
if ($settings->alert_email == '') { if ($settings->alert_email == '') {

View file

@ -1,28 +0,0 @@
<?php
namespace App\Events;
use App\Models\User;
use Illuminate\Foundation\Events\Dispatchable;
use Illuminate\Queue\SerializesModels;
class NoteAdded
{
use Dispatchable, SerializesModels;
public $itemNoteAddedOn;
public $note;
public $noteAddedBy;
/**
* Create a new event instance.
*
* @return void
*/
public function __construct($itemNoteAddedOn, User $noteAddedBy, $note)
{
$this->itemNoteAddedOn = $itemNoteAddedOn;
$this->note = $note;
$this->noteAddedBy = $noteAddedBy;
}
}

View file

@ -122,6 +122,27 @@ class Handler extends ExceptionHandler
} }
// This is traaaaash but it handles models that are not found while using route model binding :(
// The only alternative is to set that at *each* route, which is crazypants
if ($e instanceof \Illuminate\Database\Eloquent\ModelNotFoundException) {
$model_name = last(explode('\\', $e->getModel()));
$route = str_plural(strtolower(last(explode('\\', $e->getModel())))).'.index';
// Sigh.
if ($route == 'assets.index') {
$route = 'hardware.index';
} elseif ($route == 'reporttemplates.index') {
$route = 'reports/custom';
} elseif ($route == 'assetmodels.index') {
$route = 'models.index';
} elseif ($route == 'predefinedkits.index') {
$route = 'kits.index';
}
return redirect()
->route($route)
->withError(trans('general.generic_model_not_found', ['model' => $model_name]));
}
if ($this->isHttpException($e) && (isset($statusCode)) && ($statusCode == '404' )) { if ($this->isHttpException($e) && (isset($statusCode)) && ($statusCode == '404' )) {

View file

@ -1520,11 +1520,11 @@ class Helper
if ($redirect_option == 'target') { if ($redirect_option == 'target') {
switch ($checkout_to_type) { switch ($checkout_to_type) {
case 'user': case 'user':
return route('users.show', ['user' => $request->assigned_user]); return route('users.show', $request->assigned_user);
case 'location': case 'location':
return route('locations.show', ['location' => $request->assigned_location]); return route('locations.show', $request->assigned_location);
case 'asset': case 'asset':
return route('hardware.show', ['hardware' => $request->assigned_asset]); return route('hardware.show', $request->assigned_asset);
} }
} }
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error')); return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'));

View file

@ -59,6 +59,8 @@ class IconHelper
return 'fas fa-cog'; return 'fas fa-cog';
case 'angle-left': case 'angle-left':
return 'fas fa-angle-left'; return 'fas fa-angle-left';
case 'angle-right':
return 'fas fa-angle-right';
case 'warning': case 'warning':
return 'fas fa-exclamation-triangle'; return 'fas fa-exclamation-triangle';
case 'kits': case 'kits':
@ -184,6 +186,8 @@ class IconHelper
return 'fa-regular fa-id-card'; return 'fa-regular fa-id-card';
case 'department' : case 'department' :
return 'fa-solid fa-building-user'; return 'fa-solid fa-building-user';
case 'home' :
return 'fa-solid fa-house';
case 'note': case 'note':
case 'notes': case 'notes':
return 'fas fa-sticky-note'; return 'fas fa-sticky-note';

View file

@ -95,16 +95,10 @@ class AccessoriesController extends Controller
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
* @param int $accessoryId * @param int $accessoryId
*/ */
public function edit($accessoryId = null) : View | RedirectResponse public function edit(Accessory $accessory) : View | RedirectResponse
{ {
$this->authorize('update', Accessory::class);
if ($item = Accessory::find($accessoryId)) { return view('accessories.edit')->with('item', $accessory)->with('category_type', 'accessory');
$this->authorize($item);
return view('accessories.edit', compact('item'))->with('category_type', 'accessory');
}
return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist'));
} }
/** /**
@ -114,19 +108,12 @@ class AccessoriesController extends Controller
* @param int $accessoryId * @param int $accessoryId
* @since [v6.0] * @since [v6.0]
*/ */
public function getClone($accessoryId = null) : View | RedirectResponse public function getClone(Accessory $accessory) : View | RedirectResponse
{ {
$this->authorize('create', Accessory::class); $this->authorize('create', Accessory::class);
// Check if the asset exists $accessory = clone $accessory;
if (is_null($accessory_to_clone = Accessory::find($accessoryId))) {
// Redirect to the asset management page
return redirect()->route('accessories.index')
->with('error', trans('admin/accessories/message.does_not_exist', ['id' => $accessoryId]));
}
$accessory = clone $accessory_to_clone;
$accessory->id = null; $accessory->id = null;
$accessory->location_id = null; $accessory->location_id = null;
@ -142,9 +129,9 @@ class AccessoriesController extends Controller
* @param ImageUploadRequest $request * @param ImageUploadRequest $request
* @param int $accessoryId * @param int $accessoryId
*/ */
public function update(ImageUploadRequest $request, $accessoryId = null) : RedirectResponse public function update(ImageUploadRequest $request, Accessory $accessory) : RedirectResponse
{ {
if ($accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessoryId)) { if ($accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessory->id)) {
$this->authorize($accessory); $this->authorize($accessory);
@ -231,14 +218,10 @@ class AccessoriesController extends Controller
* @see AccessoriesController::getDataView() method that generates the JSON response * @see AccessoriesController::getDataView() method that generates the JSON response
* @since [v1.0] * @since [v1.0]
*/ */
public function show($accessoryID = null) : View | RedirectResponse public function show(Accessory $accessory) : View | RedirectResponse
{ {
$accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessoryID); $accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessory->id);
$this->authorize('view', $accessory); $this->authorize('view', $accessory);
if (isset($accessory->id)) { return view('accessories.view', compact('accessory'));
return view('accessories.view', compact('accessory'));
}
return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist', ['id' => $accessoryID]));
} }
} }

View file

@ -55,11 +55,11 @@ class AccessoriesFilesController extends Controller
} }
return redirect()->route('accessories.show', $accessory->id)->with('error', trans('general.no_files_uploaded')); return redirect()->route('accessories.show', $accessory->id)->withFragment('files')->with('error', trans('general.no_files_uploaded'));
} }
// Prepare the error message // Prepare the error message
return redirect()->route('accessories.index') return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist'));
->with('error', trans('general.file_does_not_exist'));
} }
/** /**
@ -72,29 +72,27 @@ class AccessoriesFilesController extends Controller
*/ */
public function destroy($accessoryId = null, $fileId = null) : RedirectResponse public function destroy($accessoryId = null, $fileId = null) : RedirectResponse
{ {
$accessory = Accessory::find($accessoryId); if ($accessory = Accessory::find($accessoryId)) {
// the asset is valid
if (isset($accessory->id)) {
$this->authorize('update', $accessory); $this->authorize('update', $accessory);
$log = Actionlog::find($fileId);
// Remove the file if one exists if ($log = Actionlog::find($fileId)) {
if (Storage::exists('accessories/'.$log->filename)) {
try { if (Storage::exists('private_uploads/accessories/'.$log->filename)) {
Storage::delete('accessories/'.$log->filename); try {
} catch (\Exception $e) { Storage::delete('private_uploads/accessories/' . $log->filename);
Log::debug($e); $log->delete();
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} catch (\Exception $e) {
Log::debug($e);
return redirect()->route('accessories.index')->with('error', trans('general.file_does_not_exist'));
}
} }
} }
return redirect()->route('accessories.show', ['accessory' => $accessory])->withFragment('files')->with('error', trans('general.log_record_not_found'));
$log->delete();
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} }
// Redirect to the licence management page return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist'));
return redirect()->route('accessories.index')->with('error', trans('general.file_does_not_exist'));
} }
/** /**
@ -124,10 +122,11 @@ class AccessoriesFilesController extends Controller
} }
} }
return redirect()->route('accessories.show', ['accessory' => $accessory])->with('error', trans('general.log_record_not_found')); return redirect()->route('accessories.show', ['accessory' => $accessory])->withFragment('files')->with('error', trans('general.log_record_not_found'));
} }
return redirect()->route('accessories.index')->with('error', trans('general.file_not_found')); return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist'));
} }
} }

View file

@ -765,9 +765,13 @@ class AssetsController extends Controller
} }
if ($problems_updating_encrypted_custom_fields) { if ($problems_updating_encrypted_custom_fields) {
return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.encrypted_warning'))); return response()->json(Helper::formatStandardApiResponse('success', $asset, trans('admin/hardware/message.update.encrypted_warning')));
// Below is the *correct* return since it uses the transformer, but we have to use the old, flat return for now until we can update Jamf2Snipe and Kanji2Snipe
// return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.encrypted_warning')));
} else { } else {
return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.success'))); return response()->json(Helper::formatStandardApiResponse('success', $asset, trans('admin/hardware/message.update.success')));
// Below is the *correct* return since it uses the transformer, but we have to use the old, flat return for now until we can update Jamf2Snipe and Kanji2Snipe
/// return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.success')));
} }
} }
return response()->json(Helper::formatStandardApiResponse('error', null, $asset->getErrors()), 200); return response()->json(Helper::formatStandardApiResponse('error', null, $asset->getErrors()), 200);
@ -1226,7 +1230,10 @@ class AssetsController extends Controller
{ {
$this->authorize('view', Asset::class); $this->authorize('view', Asset::class);
$this->authorize('view', $asset); $this->authorize('view', $asset);
$accessory_checkouts = AccessoryCheckout::AssetsAssigned()->with('adminuser')->with('accessories'); $accessory_checkouts = AccessoryCheckout::AssetsAssigned()
->where('assigned_to', $asset->id)
->with('adminuser')
->with('accessories');
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value'); $offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
$limit = app('api_limit_value'); $limit = app('api_limit_value');
@ -1235,6 +1242,8 @@ class AssetsController extends Controller
$accessory_checkouts = $accessory_checkouts->skip($offset)->take($limit)->get(); $accessory_checkouts = $accessory_checkouts->skip($offset)->take($limit)->get();
return (new AssetsTransformer)->transformCheckedoutAccessories($accessory_checkouts, $total); return (new AssetsTransformer)->transformCheckedoutAccessories($accessory_checkouts, $total);
} }
/** /**
* Generate asset labels by tag * Generate asset labels by tag
* *

View file

@ -39,6 +39,7 @@ class CategoriesController extends Controller
'components_count', 'components_count',
'licenses_count', 'licenses_count',
'image', 'image',
'notes',
]; ];
$categories = Category::select([ $categories = Category::select([
@ -52,6 +53,7 @@ class CategoriesController extends Controller
'require_acceptance', 'require_acceptance',
'checkin_email', 'checkin_email',
'image', 'image',
'notes',
]) ])
->with('adminuser') ->with('adminuser')
->withCount('accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'licenses as licenses_count'); ->withCount('accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'licenses as licenses_count');

View file

@ -38,11 +38,12 @@ class CompaniesController extends Controller
'accessories_count', 'accessories_count',
'consumables_count', 'consumables_count',
'components_count', 'components_count',
'notes',
]; ];
$companies = Company::withCount(['assets as assets_count' => function ($query) { $companies = Company::withCount(['assets as assets_count' => function ($query) {
$query->AssetsForShow(); $query->AssetsForShow();
}])->withCount('assets as assets_count', 'licenses as licenses_count', 'accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'users as users_count'); }])->withCount('licenses as licenses_count', 'accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'users as users_count');
if ($request->filled('search')) { if ($request->filled('search')) {
$companies->TextSearch($request->input('search')); $companies->TextSearch($request->input('search'));

View file

@ -48,7 +48,8 @@ class ComponentsController extends Controller
]; ];
$components = Component::select('components.*') $components = Component::select('components.*')
->with('company', 'location', 'category', 'assets', 'supplier', 'adminuser', 'manufacturer'); ->with('company', 'location', 'category', 'assets', 'supplier', 'adminuser', 'manufacturer', 'uncontrainedAssets')
->withSum('uncontrainedAssets', 'components_assets.assigned_qty');
if ($request->filled('search')) { if ($request->filled('search')) {
$components = $components->TextSearch($request->input('search')); $components = $components->TextSearch($request->input('search'));
@ -197,6 +198,11 @@ class ComponentsController extends Controller
$this->authorize('delete', Component::class); $this->authorize('delete', Component::class);
$component = Component::findOrFail($id); $component = Component::findOrFail($id);
$this->authorize('delete', $component); $this->authorize('delete', $component);
if ($component->numCheckedOut() > 0) {
return response()->json(Helper::formatStandardApiResponse('error', null, trans('admin/components/message.delete.error_qty')));
}
$component->delete(); $component->delete();
return response()->json(Helper::formatStandardApiResponse('success', null, trans('admin/components/message.delete.success'))); return response()->json(Helper::formatStandardApiResponse('success', null, trans('admin/components/message.delete.success')));

View file

@ -23,7 +23,7 @@ class DepartmentsController extends Controller
public function index(Request $request) : JsonResponse | array public function index(Request $request) : JsonResponse | array
{ {
$this->authorize('view', Department::class); $this->authorize('view', Department::class);
$allowed_columns = ['id', 'name', 'image', 'users_count']; $allowed_columns = ['id', 'name', 'image', 'users_count', 'notes'];
$departments = Department::select( $departments = Department::select(
'departments.id', 'departments.id',
@ -35,7 +35,8 @@ class DepartmentsController extends Controller
'departments.manager_id', 'departments.manager_id',
'departments.created_at', 'departments.created_at',
'departments.updated_at', 'departments.updated_at',
'departments.image' 'departments.image',
'departments.notes',
)->with('users')->with('location')->with('manager')->with('company')->withCount('users as users_count'); )->with('users')->with('location')->with('manager')->with('company')->withCount('users as users_count');
if ($request->filled('search')) { if ($request->filled('search')) {

View file

@ -24,7 +24,7 @@ class GroupsController extends Controller
$this->authorize('view', Group::class); $this->authorize('view', Group::class);
$groups = Group::select('id', 'name', 'permissions', 'created_at', 'updated_at', 'created_by')->with('adminuser')->withCount('users as users_count'); $groups = Group::select('id', 'name', 'permissions', 'notes', 'created_at', 'updated_at', 'created_by')->with('adminuser')->withCount('users as users_count');
if ($request->filled('search')) { if ($request->filled('search')) {
$groups = $groups->TextSearch($request->input('search')); $groups = $groups->TextSearch($request->input('search'));
@ -81,6 +81,7 @@ class GroupsController extends Controller
$group->name = $request->input('name'); $group->name = $request->input('name');
$group->created_by = auth()->id(); $group->created_by = auth()->id();
$group->notes = $request->input('notes');
$group->permissions = json_encode($request->input('permissions', $groupPermissions)); $group->permissions = json_encode($request->input('permissions', $groupPermissions));
if ($group->save()) { if ($group->save()) {
@ -118,6 +119,7 @@ class GroupsController extends Controller
$group = Group::findOrFail($id); $group = Group::findOrFail($id);
$group->name = $request->input('name'); $group->name = $request->input('name');
$group->notes = $request->input('notes');
$group->permissions = $request->input('permissions'); // Todo - some JSON validation stuff here $group->permissions = $request->input('permissions'); // Todo - some JSON validation stuff here
if ($group->save()) { if ($group->save()) {

View file

@ -53,6 +53,7 @@ class LocationsController extends Controller
'updated_at', 'updated_at',
'users_count', 'users_count',
'zip', 'zip',
'notes',
]; ];
$locations = Location::with('parent', 'manager', 'children')->select([ $locations = Location::with('parent', 'manager', 'children')->select([
@ -73,6 +74,7 @@ class LocationsController extends Controller
'locations.image', 'locations.image',
'locations.ldap_ou', 'locations.ldap_ou',
'locations.currency', 'locations.currency',
'locations.notes',
]) ])
->withCount('assignedAssets as assigned_assets_count') ->withCount('assignedAssets as assigned_assets_count')
->withCount('assets as assets_count') ->withCount('assets as assets_count')
@ -190,6 +192,7 @@ class LocationsController extends Controller
'locations.updated_at', 'locations.updated_at',
'locations.image', 'locations.image',
'locations.currency', 'locations.currency',
'locations.notes',
]) ])
->withCount('assignedAssets as assigned_assets_count') ->withCount('assignedAssets as assigned_assets_count')
->withCount('assets as assets_count') ->withCount('assets as assets_count')
@ -255,7 +258,7 @@ class LocationsController extends Controller
{ {
$this->authorize('view', Accessory::class); $this->authorize('view', Accessory::class);
$this->authorize('view', $location); $this->authorize('view', $location);
$accessory_checkouts = AccessoryCheckout::LocationAssigned()->with('adminuser')->with('accessories'); $accessory_checkouts = AccessoryCheckout::LocationAssigned()->where('assigned_to', $location->id)->with('adminuser')->with('accessories');
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value'); $offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
$limit = app('api_limit_value'); $limit = app('api_limit_value');

View file

@ -39,7 +39,8 @@ class ManufacturersController extends Controller
'assets_count', 'assets_count',
'consumables_count', 'consumables_count',
'components_count', 'components_count',
'licenses_count' 'licenses_count',
'notes',
]; ];
$manufacturers = Manufacturer::select([ $manufacturers = Manufacturer::select([
@ -55,6 +56,7 @@ class ManufacturersController extends Controller
'updated_at', 'updated_at',
'image', 'image',
'deleted_at', 'deleted_at',
'notes',
]) ])
->with('adminuser') ->with('adminuser')
->withCount('assets as assets_count') ->withCount('assets as assets_count')

View file

@ -1,43 +0,0 @@
<?php
namespace App\Http\Controllers\Api;
use App\Events\NoteAdded;
use App\Helpers\Helper;
use App\Http\Controllers\Controller;
use App\Models\Asset;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Validation\Rule;
class NotesController extends Controller
{
public function store(Request $request)
{
$validated = $request->validate([
'note' => 'required|string|max:500',
'type' => [
'required',
Rule::in(['asset']),
],
]);
// This can be made dynamic by using $request->input('type') to determine which model type to add the note to.
// For now, we are only placing this on Assets
$item = Asset::findOrFail($request->input("id"));
$this->authorize('update', $item);
event(new NoteAdded($item, Auth::user(), $validated['note']));
return response()->json(Helper::formatStandardApiResponse('success'));
}
public function update(Request $request)
{
}
public function destroy(Request $request)
{
}
}

View file

@ -139,19 +139,12 @@ class AssetMaintenancesController extends Controller
* @version v1.0 * @version v1.0
* @since [v1.8] * @since [v1.8]
*/ */
public function edit($assetMaintenanceId = null) : View | RedirectResponse public function edit(AssetMaintenance $maintenance) : View | RedirectResponse
{ {
$this->authorize('update', Asset::class); $this->authorize('update', Asset::class);
// Check if the asset maintenance exists if ((!$maintenance->asset) || ($maintenance->asset->deleted_at!='')) {
$this->authorize('update', Asset::class);
// Check if the asset maintenance exists
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) {
// Redirect to the asset maintenance management page
return redirect()->route('maintenances.index')->with('error', trans('admin/asset_maintenances/message.not_found'));
} elseif ((!$assetMaintenance->asset) || ($assetMaintenance->asset->deleted_at!='')) {
// Redirect to the asset maintenance management page
return redirect()->route('maintenances.index')->with('error', 'asset does not exist'); return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) { } elseif (! Company::isCurrentUserHasAccess($maintenance->asset)) {
return static::getInsufficientPermissionsRedirect(); return static::getInsufficientPermissionsRedirect();
} }
@ -161,7 +154,7 @@ class AssetMaintenancesController extends Controller
return view('asset_maintenances/edit') return view('asset_maintenances/edit')
->with('selectedAsset', null) ->with('selectedAsset', null)
->with('assetMaintenanceType', $assetMaintenanceType) ->with('assetMaintenanceType', $assetMaintenanceType)
->with('item', $assetMaintenance); ->with('item', $maintenance);
} }
/** /**
@ -174,24 +167,20 @@ class AssetMaintenancesController extends Controller
* @version v1.0 * @version v1.0
* @since [v1.8] * @since [v1.8]
*/ */
public function update(Request $request, $assetMaintenanceId = null) : View | RedirectResponse public function update(Request $request, AssetMaintenance $maintenance) : View | RedirectResponse
{ {
$this->authorize('update', Asset::class); $this->authorize('update', Asset::class);
// Check if the asset maintenance exists
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) { if ((!$maintenance->asset) || ($maintenance->asset->deleted_at!='')) {
// Redirect to the asset maintenance management page
return redirect()->route('maintenances.index')->with('error', trans('admin/asset_maintenances/message.not_found'));
} elseif ((!$assetMaintenance->asset) || ($assetMaintenance->asset->deleted_at!='')) {
// Redirect to the asset maintenance management page
return redirect()->route('maintenances.index')->with('error', 'asset does not exist'); return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) { } elseif (! Company::isCurrentUserHasAccess($maintenance->asset)) {
return static::getInsufficientPermissionsRedirect(); return static::getInsufficientPermissionsRedirect();
} }
$assetMaintenance->supplier_id = $request->input('supplier_id'); $maintenance->supplier_id = $request->input('supplier_id');
$assetMaintenance->is_warranty = $request->input('is_warranty'); $maintenance->is_warranty = $request->input('is_warranty');
$assetMaintenance->cost = $request->input('cost'); $maintenance->cost = $request->input('cost');
$assetMaintenance->notes = $request->input('notes'); $maintenance->notes = $request->input('notes');
$asset = Asset::find(request('asset_id')); $asset = Asset::find(request('asset_id'));
@ -200,39 +189,39 @@ class AssetMaintenancesController extends Controller
} }
// Save the asset maintenance data // Save the asset maintenance data
$assetMaintenance->asset_id = $request->input('asset_id'); $maintenance->asset_id = $request->input('asset_id');
$assetMaintenance->asset_maintenance_type = $request->input('asset_maintenance_type'); $maintenance->asset_maintenance_type = $request->input('asset_maintenance_type');
$assetMaintenance->title = $request->input('title'); $maintenance->title = $request->input('title');
$assetMaintenance->start_date = $request->input('start_date'); $maintenance->start_date = $request->input('start_date');
$assetMaintenance->completion_date = $request->input('completion_date'); $maintenance->completion_date = $request->input('completion_date');
if (($assetMaintenance->completion_date == null) if (($maintenance->completion_date == null)
) { ) {
if (($assetMaintenance->asset_maintenance_time !== 0) if (($maintenance->asset_maintenance_time !== 0)
|| (! is_null($assetMaintenance->asset_maintenance_time)) || (! is_null($maintenance->asset_maintenance_time))
) { ) {
$assetMaintenance->asset_maintenance_time = null; $maintenance->asset_maintenance_time = null;
} }
} }
if (($assetMaintenance->completion_date !== null) if (($maintenance->completion_date !== null)
&& ($assetMaintenance->start_date !== '') && ($maintenance->start_date !== '')
&& ($assetMaintenance->start_date !== '0000-00-00') && ($maintenance->start_date !== '0000-00-00')
) { ) {
$startDate = Carbon::parse($assetMaintenance->start_date); $startDate = Carbon::parse($maintenance->start_date);
$completionDate = Carbon::parse($assetMaintenance->completion_date); $completionDate = Carbon::parse($maintenance->completion_date);
$assetMaintenance->asset_maintenance_time = $completionDate->diffInDays($startDate); $maintenance->asset_maintenance_time = $completionDate->diffInDays($startDate);
} }
// Was the asset maintenance created? // Was the asset maintenance created?
if ($assetMaintenance->save()) { if ($maintenance->save()) {
// Redirect to the new asset maintenance page // Redirect to the new asset maintenance page
return redirect()->route('maintenances.index') return redirect()->route('maintenances.index')
->with('success', trans('admin/asset_maintenances/message.edit.success')); ->with('success', trans('admin/asset_maintenances/message.edit.success'));
} }
return redirect()->back()->withInput()->withErrors($assetMaintenance->getErrors()); return redirect()->back()->withInput()->withErrors($maintenance->getErrors());
} }
/** /**
@ -271,19 +260,13 @@ class AssetMaintenancesController extends Controller
* @version v1.0 * @version v1.0
* @since [v1.8] * @since [v1.8]
*/ */
public function show($assetMaintenanceId) : View | RedirectResponse public function show(AssetMaintenance $maintenance) : View | RedirectResponse
{ {
$this->authorize('view', Asset::class); $this->authorize('view', Asset::class);
if (! Company::isCurrentUserHasAccess($maintenance->asset)) {
// Check if the asset maintenance exists
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) {
// Redirect to the asset maintenance management page
return redirect()->route('maintenances.index')
->with('error', trans('admin/asset_maintenances/message.not_found'));
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) {
return static::getInsufficientPermissionsRedirect(); return static::getInsufficientPermissionsRedirect();
} }
return view('asset_maintenances/view')->with('assetMaintenance', $assetMaintenance); return view('asset_maintenances/view')->with('assetMaintenance', $maintenance);
} }
} }

View file

@ -109,16 +109,11 @@ class AssetModelsController extends Controller
* @since [v1.0] * @since [v1.0]
* @param int $modelId * @param int $modelId
*/ */
public function edit($modelId = null) : View | RedirectResponse public function edit(AssetModel $model) : View | RedirectResponse
{ {
$this->authorize('update', AssetModel::class); $this->authorize('update', AssetModel::class);
if ($item = AssetModel::find($modelId)) { $category_type = 'asset';
$category_type = 'asset'; return view('models/edit', compact('category_type'))->with('item', $model)->with('depreciation_list', Helper::depreciationList());
return view('models/edit', compact('item', 'category_type'))->with('depreciation_list', Helper::depreciationList());
}
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
} }
@ -133,16 +128,11 @@ class AssetModelsController extends Controller
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function update(StoreAssetModelRequest $request, $modelId) : RedirectResponse public function update(StoreAssetModelRequest $request, AssetModel $model) : RedirectResponse
{ {
$this->authorize('update', AssetModel::class); $this->authorize('update', AssetModel::class);
if (is_null($model = AssetModel::find($modelId))) {
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
}
$model = $request->handleImages($model); $model = $request->handleImages($model);
$model->depreciation_id = $request->input('depreciation_id'); $model->depreciation_id = $request->input('depreciation_id');
$model->eol = $request->input('eol'); $model->eol = $request->input('eol');
$model->name = $request->input('name'); $model->name = $request->input('name');
@ -188,28 +178,16 @@ class AssetModelsController extends Controller
* @since [v1.0] * @since [v1.0]
* @param int $modelId * @param int $modelId
*/ */
public function destroy($modelId) : RedirectResponse public function destroy(AssetModel $model) : RedirectResponse
{ {
$this->authorize('delete', AssetModel::class); $this->authorize('delete', AssetModel::class);
// Check if the model exists
if (is_null($model = AssetModel::find($modelId))) {
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
}
if ($model->assets()->count() > 0) { if ($model->assets()->count() > 0) {
// Throw an error that this model is associated with assets // Throw an error that this model is associated with assets
return redirect()->route('models.index')->with('error', trans('admin/models/message.assoc_users')); return redirect()->route('models.index')->with('error', trans('admin/models/message.assoc_users'));
} }
if ($model->image) {
try {
Storage::disk('public')->delete('models/'.$model->image);
$model->update(['image' => null]);
} catch (\Exception $e) {
Log::info($e);
}
}
// Delete the model // Delete the model
$model->delete(); $model->delete();
@ -267,16 +245,10 @@ class AssetModelsController extends Controller
* @since [v1.0] * @since [v1.0]
* @param int $modelId * @param int $modelId
*/ */
public function show($modelId = null) : View | RedirectResponse public function show(AssetModel $model) : View | RedirectResponse
{ {
$this->authorize('view', AssetModel::class); $this->authorize('view', AssetModel::class);
$model = AssetModel::withTrashed()->find($modelId); return view('models/view', compact('model'));
if (isset($model->id)) {
return view('models/view', compact('model'));
}
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
} }
/** /**
@ -286,23 +258,20 @@ class AssetModelsController extends Controller
* @since [v1.0] * @since [v1.0]
* @param int $modelId * @param int $modelId
*/ */
public function getClone($modelId = null) : View | RedirectResponse public function getClone(AssetModel $model) : View | RedirectResponse
{ {
$this->authorize('create', AssetModel::class); $this->authorize('create', AssetModel::class);
// Check if the model exists
if (is_null($model_to_clone = AssetModel::find($modelId))) {
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
}
$model = clone $model_to_clone; $cloned_model = clone $model;
$model->id = null; $model->id = null;
$model->deleted_at = null;
// Show the page // Show the page
return view('models/edit') return view('models/edit')
->with('depreciation_list', Helper::depreciationList()) ->with('depreciation_list', Helper::depreciationList())
->with('item', $model) ->with('item', $model)
->with('model_id', $model_to_clone->id) ->with('model_id', $model->id)
->with('clone_model', $model_to_clone); ->with('clone_model', $cloned_model);
} }
@ -321,7 +290,7 @@ class AssetModelsController extends Controller
/** /**
* Returns a view that allows the user to bulk edit model attrbutes * Returns a view that allows the user to bulk edit model attributes
* *
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
* @since [v1.7] * @since [v1.7]

View file

@ -58,11 +58,9 @@ class AssetModelsFilesController extends Controller
* @param int $fileId * @param int $fileId
* @since [v1.0] * @since [v1.0]
*/ */
public function show($modelId = null, $fileId = null) : StreamedResponse | Response | RedirectResponse | BinaryFileResponse public function show(AssetModel $model, $fileId = null) : StreamedResponse | Response | RedirectResponse | BinaryFileResponse
{ {
$model = AssetModel::find($modelId);
// the asset is valid
if (isset($model->id)) {
$this->authorize('view', $model); $this->authorize('view', $model);
if (! $log = Actionlog::find($fileId)) { if (! $log = Actionlog::find($fileId)) {
@ -87,12 +85,6 @@ class AssetModelsFilesController extends Controller
} }
return StorageHelper::downloader($file); return StorageHelper::downloader($file);
}
// Prepare the error message
$error = trans('admin/hardware/message.does_not_exist', ['id' => $fileId]);
// Redirect to the hardware management page
return redirect()->route('hardware.index')->with('error', $error);
} }
/** /**
@ -103,29 +95,21 @@ class AssetModelsFilesController extends Controller
* @param int $fileId * @param int $fileId
* @since [v1.0] * @since [v1.0]
*/ */
public function destroy($modelId = null, $fileId = null) : RedirectResponse public function destroy(AssetModel $model, $fileId = null) : RedirectResponse
{ {
$model = AssetModel::find($modelId);
$this->authorize('update', $model);
$rel_path = 'private_uploads/assetmodels'; $rel_path = 'private_uploads/assetmodels';
$this->authorize('update', $model);
// the asset is valid $log = Actionlog::find($fileId);
if (isset($model->id)) { if ($log) {
$this->authorize('update', $model); if (Storage::exists($rel_path.'/'.$log->filename)) {
$log = Actionlog::find($fileId); Storage::delete($rel_path.'/'.$log->filename);
if ($log) {
if (Storage::exists($rel_path.'/'.$log->filename)) {
Storage::delete($rel_path.'/'.$log->filename);
}
$log->delete();
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} }
$log->delete();
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success')); return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} }
// Redirect to the hardware management page return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
} }
} }

View file

@ -27,18 +27,12 @@ class AssetCheckinController extends Controller
* @param string $backto * @param string $backto
* @since [v1.0] * @since [v1.0]
*/ */
public function create($assetId, $backto = null) : View | RedirectResponse public function create(Asset $asset, $backto = null) : View | RedirectResponse
{ {
// Check if the asset exists
if (is_null($asset = Asset::find($assetId))) {
// Redirect to the asset management page with error
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
}
$this->authorize('checkin', $asset); $this->authorize('checkin', $asset);
// This asset is already checked in, redirect // This asset is already checked in, redirect
if (is_null($asset->assignedTo)) { if (is_null($asset->assignedTo)) {
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkin.already_checked_in')); return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkin.already_checked_in'));
} }
@ -47,7 +41,11 @@ class AssetCheckinController extends Controller
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix')); return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix'));
} }
return view('hardware/checkin', compact('asset'))->with('statusLabel_list', Helper::statusLabelList())->with('backto', $backto)->with('table_name', 'Assets'); return view('hardware/checkin', compact('asset'))
->with('item', $asset)
->with('statusLabel_list', Helper::statusLabelList())
->with('backto', $backto)
->with('table_name', 'Assets');
} }
/** /**
@ -91,6 +89,9 @@ class AssetCheckinController extends Controller
$asset->status_id = e($request->get('status_id')); $asset->status_id = e($request->get('status_id'));
} }
// Add any custom fields that should be included in the checkout
$asset->customFieldsForCheckinCheckout('display_checkin');
$this->migrateLegacyLocations($asset); $this->migrateLegacyLocations($asset);
$asset->location_id = $asset->rtd_location_id; $asset->location_id = $asset->rtd_location_id;
@ -128,6 +129,9 @@ class AssetCheckinController extends Controller
session()->put('redirect_option', $request->get('redirect_option')); session()->put('redirect_option', $request->get('redirect_option'));
// Add any custom fields that should be included in the checkout
$asset->customFieldsForCheckinCheckout('display_checkin');
if ($asset->save()) { if ($asset->save()) {
event(new CheckoutableCheckedIn($asset, $target, auth()->user(), $request->input('note'), $checkin_at, $originalValues)); event(new CheckoutableCheckedIn($asset, $target, auth()->user(), $request->input('note'), $checkin_at, $originalValues));

View file

@ -26,27 +26,25 @@ class AssetCheckoutController extends Controller
* @since [v1.0] * @since [v1.0]
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
*/ */
public function create($assetId) : View | RedirectResponse public function create(Asset $asset) : View | RedirectResponse
{ {
// Check if the asset exists
if (is_null($asset = Asset::with('company')->find(e($assetId)))) {
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
}
$this->authorize('checkout', $asset); $this->authorize('checkout', $asset);
if (!$asset->model) { if (!$asset->model) {
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix')); return redirect()->route('hardware.show', $asset)
->with('error', trans('admin/hardware/general.model_invalid_fix'));
} }
if ($asset->availableForCheckout()) { if ($asset->availableForCheckout()) {
return view('hardware/checkout', compact('asset')) return view('hardware/checkout', compact('asset'))
->with('statusLabel_list', Helper::deployableStatusLabelList()) ->with('statusLabel_list', Helper::deployableStatusLabelList())
->with('table_name', 'Assets'); ->with('table_name', 'Assets')
->with('item', $asset);
} }
return redirect()->route('hardware.index')
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkout.not_available')); ->with('error', trans('admin/hardware/message.checkout.not_available'));
} }
/** /**
@ -68,7 +66,7 @@ class AssetCheckoutController extends Controller
$this->authorize('checkout', $asset); $this->authorize('checkout', $asset);
if (!$asset->model) { if (!$asset->model) {
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix')); return redirect()->route('hardware.show', $asset)->with('error', trans('admin/hardware/general.model_invalid_fix'));
} }
$admin = auth()->user(); $admin = auth()->user();
@ -91,6 +89,7 @@ class AssetCheckoutController extends Controller
$asset->status_id = $request->get('status_id'); $asset->status_id = $request->get('status_id');
} }
if(!empty($asset->licenseseats->all())){ if(!empty($asset->licenseseats->all())){
if(request('checkout_to_type') == 'user') { if(request('checkout_to_type') == 'user') {
foreach ($asset->licenseseats as $seat){ foreach ($asset->licenseseats as $seat){
@ -100,23 +99,26 @@ class AssetCheckoutController extends Controller
} }
} }
// Add any custom fields that should be included in the checkout
$asset->customFieldsForCheckinCheckout('display_checkout');
$settings = \App\Models\Setting::getSettings(); $settings = \App\Models\Setting::getSettings();
// We have to check whether $target->company_id is null here since locations don't have a company yet // We have to check whether $target->company_id is null here since locations don't have a company yet
if (($settings->full_multiple_companies_support) && ((!is_null($target->company_id)) && (!is_null($asset->company_id)))) { if (($settings->full_multiple_companies_support) && ((!is_null($target->company_id)) && (!is_null($asset->company_id)))) {
if ($target->company_id != $asset->company_id){ if ($target->company_id != $asset->company_id){
return redirect()->to("hardware/$assetId/checkout")->with('error', trans('general.error_user_company')); return redirect()->route('hardware.checkout.create', $asset)->with('error', trans('general.error_user_company'));
} }
} }
session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]); session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]);
if ($asset->checkOut($target, $admin, $checkout_at, $expected_checkin, $request->get('note'), $request->get('name'))) { if ($asset->checkOut($target, $admin, $checkout_at, $expected_checkin, $request->get('note'), $request->get('name'))) {
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets')) return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
->with('success', trans('admin/hardware/message.checkout.success')); ->with('success', trans('admin/hardware/message.checkout.success'));
} }
// Redirect to the asset management page with error // Redirect to the asset management page with error
return redirect()->to("hardware/$assetId/checkout")->with('error', trans('admin/hardware/message.checkout.error').$asset->getErrors()); return redirect()->route("hardware.checkout.create", $asset)->with('error', trans('admin/hardware/message.checkout.error').$asset->getErrors());
} catch (ModelNotFoundException $e) { } catch (ModelNotFoundException $e) {
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'))->withErrors($asset->getErrors()); return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'))->withErrors($asset->getErrors());
} catch (CheckoutNotAllowed $e) { } catch (CheckoutNotAllowed $e) {

View file

@ -26,11 +26,8 @@ class AssetFilesController extends Controller
*@since [v1.0] *@since [v1.0]
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
*/ */
public function store(UploadFileRequest $request, $assetId = null) : RedirectResponse public function store(UploadFileRequest $request, Asset $asset) : RedirectResponse
{ {
if (! $asset = Asset::find($assetId)) {
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
}
$this->authorize('update', $asset); $this->authorize('update', $asset);
@ -59,31 +56,28 @@ class AssetFilesController extends Controller
* @param int $fileId * @param int $fileId
* @since [v1.0] * @since [v1.0]
*/ */
public function show($assetId = null, $fileId = null) : View | RedirectResponse | Response | StreamedResponse | BinaryFileResponse public function show(Asset $asset, $fileId = null) : View | RedirectResponse | Response | StreamedResponse | BinaryFileResponse
{ {
if ($asset = Asset::find($assetId)) {
$this->authorize('view', $asset); $this->authorize('view', $asset);
if ($log = Actionlog::whereNotNull('filename')->where('item_id', $asset->id)->find($fileId)) { if ($log = Actionlog::whereNotNull('filename')->where('item_id', $asset->id)->find($fileId)) {
$file = 'private_uploads/assets/'.$log->filename; $file = 'private_uploads/assets/'.$log->filename;
if ($log->action_type == 'audit') {
$file = 'private_uploads/audits/'.$log->filename;
}
try {
return StorageHelper::showOrDownloadFile($file, $log->filename);
} catch (\Exception $e) {
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.file_not_found'));
}
if ($log->action_type == 'audit') {
$file = 'private_uploads/audits/'.$log->filename;
}
try {
return StorageHelper::showOrDownloadFile($file, $log->filename);
} catch (\Exception $e) {
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.file_not_found'));
} }
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found'));
} }
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist')); return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found'));
} }
@ -95,23 +89,20 @@ class AssetFilesController extends Controller
* @param int $fileId * @param int $fileId
* @since [v1.0] * @since [v1.0]
*/ */
public function destroy($assetId = null, $fileId = null) : RedirectResponse public function destroy(Asset $asset, $fileId = null) : RedirectResponse
{ {
if ($asset = Asset::find($assetId)) { $this->authorize('update', $asset);
$this->authorize('update', $asset); $rel_path = 'private_uploads/assets';
$rel_path = 'private_uploads/assets';
if ($log = Actionlog::find($fileId)) { if ($log = Actionlog::find($fileId)) {
if (Storage::exists($rel_path.'/'.$log->filename)) { if (Storage::exists($rel_path.'/'.$log->filename)) {
Storage::delete($rel_path.'/'.$log->filename); Storage::delete($rel_path.'/'.$log->filename);
}
$log->delete();
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} }
$log->delete();
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found')); return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
} }
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist')); return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found'));
} }
} }

View file

@ -30,6 +30,7 @@ use Illuminate\Http\Response;
use Illuminate\Contracts\View\View; use Illuminate\Contracts\View\View;
use Illuminate\Http\RedirectResponse; use Illuminate\Http\RedirectResponse;
use Symfony\Component\HttpFoundation\BinaryFileResponse; use Symfony\Component\HttpFoundation\BinaryFileResponse;
use TypeError;
/** /**
* This class controls all actions related to assets for * This class controls all actions related to assets for
@ -201,7 +202,7 @@ class AssetsController extends Controller
$asset->checkOut($target, auth()->user(), date('Y-m-d H:i:s'), $request->input('expected_checkin', null), 'Checked out on asset creation', $request->get('name'), $location); $asset->checkOut($target, auth()->user(), date('Y-m-d H:i:s'), $request->input('expected_checkin', null), 'Checked out on asset creation', $request->get('name'), $location);
} }
$successes[] = "<a href='" . route('hardware.show', ['hardware' => $asset->id]) . "' style='color: white;'>" . e($asset->asset_tag) . "</a>"; $successes[] = "<a href='" . route('hardware.show', $asset) . "' style='color: white;'>" . e($asset->asset_tag) . "</a>";
} else { } else {
$failures[] = join(",", $asset->getErrors()->all()); $failures[] = join(",", $asset->getErrors()->all());
@ -222,7 +223,7 @@ class AssetsController extends Controller
//the most common case, keeping it so we don't have to make every use of that translation string be trans_choice'ed //the most common case, keeping it so we don't have to make every use of that translation string be trans_choice'ed
//and re-translated //and re-translated
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets')) return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
->with('success-unescaped', trans('admin/hardware/message.create.success_linked', ['link' => route('hardware.show', ['hardware' => $asset->id]), 'id', 'tag' => e($asset->asset_tag)])); ->with('success-unescaped', trans('admin/hardware/message.create.success_linked', ['link' => route('hardware.show', $asset), 'id', 'tag' => e($asset->asset_tag)]));
} else { } else {
//multi-success //multi-success
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets')) return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
@ -240,20 +241,14 @@ class AssetsController extends Controller
* Returns a view that presents a form to edit an existing asset. * Returns a view that presents a form to edit an existing asset.
* *
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
* @param int $assetId
* @since [v1.0] * @since [v1.0]
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
*/ */
public function edit($assetId = null) : View | RedirectResponse public function edit(Asset $asset) : View | RedirectResponse
{ {
if (! $item = Asset::find($assetId)) { $this->authorize($asset);
// Redirect to the asset management page with error return view('hardware/edit')
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist')); ->with('item', $asset)
}
//Handles company checks and permissions.
$this->authorize($item);
return view('hardware/edit', compact('item'))
->with('statuslabel_list', Helper::statusLabelList()) ->with('statuslabel_list', Helper::statusLabelList())
->with('statuslabel_types', Helper::statusTypeList()); ->with('statuslabel_types', Helper::statusTypeList());
} }
@ -267,15 +262,14 @@ class AssetsController extends Controller
* @since [v1.0] * @since [v1.0]
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
*/ */
public function show($assetId = null) : View | RedirectResponse public function show(Asset $asset) : View | RedirectResponse
{ {
$asset = Asset::withTrashed()->find($assetId);
$this->authorize('view', $asset); $this->authorize('view', $asset);
$settings = Setting::getSettings(); $settings = Setting::getSettings();
if (isset($asset)) { if (isset($asset)) {
$audit_log = Actionlog::where('action_type', '=', 'audit') $audit_log = Actionlog::where('action_type', '=', 'audit')
->where('item_id', '=', $assetId) ->where('item_id', '=', $asset->id)
->where('item_type', '=', Asset::class) ->where('item_type', '=', Asset::class)
->orderBy('created_at', 'DESC')->first(); ->orderBy('created_at', 'DESC')->first();
@ -291,7 +285,7 @@ class AssetsController extends Controller
$qr_code = (object) [ $qr_code = (object) [
'display' => $settings->qr_code == '1', 'display' => $settings->qr_code == '1',
'url' => route('qr_code/hardware', $asset->id), 'url' => route('qr_code/hardware', $asset),
]; ];
return view('hardware/view', compact('asset', 'qr_code', 'settings')) return view('hardware/view', compact('asset', 'qr_code', 'settings'))
@ -308,14 +302,9 @@ class AssetsController extends Controller
* @since [v1.0] * @since [v1.0]
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
*/ */
public function update(ImageUploadRequest $request, $assetId = null) : RedirectResponse public function update(ImageUploadRequest $request, Asset $asset) : RedirectResponse
{ {
// Check if the asset exists
if (! $asset = Asset::find($assetId)) {
// Redirect to the asset management page with error
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
}
$this->authorize($asset); $this->authorize($asset);
$asset->status_id = $request->input('status_id', null); $asset->status_id = $request->input('status_id', null);
@ -430,7 +419,7 @@ class AssetsController extends Controller
session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]); session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]);
if ($asset->save()) { if ($asset->save()) {
return redirect()->to(Helper::getRedirectOption($request, $assetId, 'Assets')) return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
->with('success', trans('admin/hardware/message.update.success')); ->with('success', trans('admin/hardware/message.update.success'));
} }
@ -531,12 +520,12 @@ class AssetsController extends Controller
* @param int $assetId * @param int $assetId
* @since [v1.0] * @since [v1.0]
*/ */
public function getQrCode($assetId = null) : Response | BinaryFileResponse | string | bool public function getQrCode(Asset $asset) : Response | BinaryFileResponse | string | bool
{ {
$settings = Setting::getSettings(); $settings = Setting::getSettings();
if ($settings->qr_code == '1') { if (($settings->qr_code == '1') && ($settings->label2_2d_type !== 'none')) {
$asset = Asset::withTrashed()->find($assetId);
if ($asset) { if ($asset) {
$size = Helper::barcodeDimensions($settings->label2_2d_type); $size = Helper::barcodeDimensions($settings->label2_2d_type);
$qr_file = public_path().'/uploads/barcodes/qr-'.str_slug($asset->asset_tag).'-'.str_slug($asset->id).'.png'; $qr_file = public_path().'/uploads/barcodes/qr-'.str_slug($asset->asset_tag).'-'.str_slug($asset->id).'.png';
@ -590,7 +579,7 @@ class AssetsController extends Controller
file_put_contents($barcode_file, $barcode_obj->getPngData()); file_put_contents($barcode_file, $barcode_obj->getPngData());
return response($barcode_obj->getPngData())->header('Content-type', 'image/png'); return response($barcode_obj->getPngData())->header('Content-type', 'image/png');
} catch (\Exception $e) { } catch (\Exception|TypeError $e) {
Log::debug('The barcode format is invalid.'); Log::debug('The barcode format is invalid.');
return response(file_get_contents(public_path('uploads/barcodes/invalid_barcode.gif')))->header('Content-type', 'image/gif'); return response(file_get_contents(public_path('uploads/barcodes/invalid_barcode.gif')))->header('Content-type', 'image/gif');
@ -877,12 +866,11 @@ class AssetsController extends Controller
return view('hardware/quickscan-checkin')->with('statusLabel_list', Helper::statusLabelList()); return view('hardware/quickscan-checkin')->with('statusLabel_list', Helper::statusLabelList());
} }
public function audit($id) public function audit(Asset $asset)
{ {
$settings = Setting::getSettings(); $settings = Setting::getSettings();
$this->authorize('audit', Asset::class); $this->authorize('audit', Asset::class);
$dt = Carbon::now()->addMonths($settings->audit_interval)->toDateString(); $dt = Carbon::now()->addMonths($settings->audit_interval)->toDateString();
$asset = Asset::findOrFail($id);
return view('hardware/audit')->with('asset', $asset)->with('next_audit_date', $dt)->with('locations_list'); return view('hardware/audit')->with('asset', $asset)->with('next_audit_date', $dt)->with('locations_list');
} }
@ -901,7 +889,7 @@ class AssetsController extends Controller
} }
public function auditStore(UploadFileRequest $request, $id) public function auditStore(UploadFileRequest $request, Asset $asset)
{ {
$this->authorize('audit', Asset::class); $this->authorize('audit', Asset::class);
@ -916,8 +904,6 @@ class AssetsController extends Controller
return response()->json(Helper::formatStandardApiResponse('error', null, $validator->errors()->all())); return response()->json(Helper::formatStandardApiResponse('error', null, $validator->errors()->all()));
} }
$asset = Asset::findOrFail($id);
/** /**
* Even though we do a save() further down, we don't want to log this as a "normal" asset update, * Even though we do a save() further down, we don't want to log this as a "normal" asset update,
* which would trigger the Asset Observer and would log an asset *update* log entry (because the * which would trigger the Asset Observer and would log an asset *update* log entry (because the

View file

@ -525,21 +525,31 @@ class BulkAssetsController extends Controller
$this->authorize('delete', Asset::class); $this->authorize('delete', Asset::class);
$bulk_back_url = route('hardware.index'); $bulk_back_url = route('hardware.index');
if ($request->session()->has('bulk_back_url')) { if ($request->session()->has('bulk_back_url')) {
$bulk_back_url = $request->session()->pull('bulk_back_url'); $bulk_back_url = $request->session()->pull('bulk_back_url');
} }
$assetIds = $request->get('ids');
if ($request->filled('ids')) { if(empty($assetIds)) {
$assets = Asset::find($request->get('ids')); return redirect($bulk_back_url)->with('error', trans('admin/hardware/message.delete.nothing_updated'));
foreach ($assets as $asset) {
$asset->delete();
} // endforeach
return redirect($bulk_back_url)->with('success', trans('admin/hardware/message.delete.success'));
// no values given, nothing to update
} }
return redirect($bulk_back_url)->with('error', trans('admin/hardware/message.delete.nothing_updated')); $assignedAssets = Asset::whereIn('id', $assetIds)->whereNotNull('assigned_to')->get();
if($assignedAssets->isNotEmpty()) {
//if assets are checked out, return a list of asset tags that would need to be checked in first.
$assetTags = $assignedAssets->pluck('asset_tag')->implode(', ');
return redirect($bulk_back_url)->with('error', trans_choice('admin/hardware/message.delete.assigned_to_error', $assignedAssets->count(), ['asset_tag' => $assetTags] ));
}
foreach (Asset::wherein('id', $assetIds)->get() as $asset) {
$asset->delete();
}
return redirect($bulk_back_url)->with('success', trans('admin/hardware/message.delete.success'));
// no values given, nothing to update
} }
/** /**

View file

@ -69,6 +69,7 @@ class CategoriesController extends Controller
$category->use_default_eula = $request->input('use_default_eula', '0'); $category->use_default_eula = $request->input('use_default_eula', '0');
$category->require_acceptance = $request->input('require_acceptance', '0'); $category->require_acceptance = $request->input('require_acceptance', '0');
$category->checkin_email = $request->input('checkin_email', '0'); $category->checkin_email = $request->input('checkin_email', '0');
$category->notes = $request->input('notes');
$category->created_by = auth()->id(); $category->created_by = auth()->id();
$category = $request->handleImages($category); $category = $request->handleImages($category);
@ -87,14 +88,10 @@ class CategoriesController extends Controller
* @param int $categoryId * @param int $categoryId
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($categoryId = null) : RedirectResponse | View public function edit(Category $category) : RedirectResponse | View
{ {
$this->authorize('update', Category::class); $this->authorize('update', Category::class);
if (is_null($item = Category::find($categoryId))) { return view('categories/edit')->with('item', $category)
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
}
return view('categories/edit', compact('item'))
->with('category_types', Helper::categoryTypeList()); ->with('category_types', Helper::categoryTypeList());
} }
@ -107,19 +104,10 @@ class CategoriesController extends Controller
* @param int $categoryId * @param int $categoryId
* @since [v1.0] * @since [v1.0]
*/ */
public function update(ImageUploadRequest $request, $categoryId = null) : RedirectResponse public function update(ImageUploadRequest $request, Category $category) : RedirectResponse
{ {
$this->authorize('update', Category::class); $this->authorize('update', Category::class);
if (is_null($category = Category::find($categoryId))) {
// Redirect to the categories management page
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
}
// Update the category data
$category->name = $request->input('name'); $category->name = $request->input('name');
// If the item count is > 0, we disable the category type in the edit. Disabled items
// don't POST, so if the category_type is blank we just set it to the default.
// Don't allow the user to change the category_type once it's been created // Don't allow the user to change the category_type once it's been created
if (($request->filled('category_type') && ($category->itemCount() > 0))) { if (($request->filled('category_type') && ($category->itemCount() > 0))) {
@ -134,6 +122,7 @@ class CategoriesController extends Controller
$category->use_default_eula = $request->input('use_default_eula', '0'); $category->use_default_eula = $request->input('use_default_eula', '0');
$category->require_acceptance = $request->input('require_acceptance', '0'); $category->require_acceptance = $request->input('require_acceptance', '0');
$category->checkin_email = $request->input('checkin_email', '0'); $category->checkin_email = $request->input('checkin_email', '0');
$category->notes = $request->input('notes');
$category = $request->handleImages($category); $category = $request->handleImages($category);
@ -179,10 +168,10 @@ class CategoriesController extends Controller
* @param $id * @param $id
* @since [v1.8] * @since [v1.8]
*/ */
public function show($id) : View | RedirectResponse public function show(Category $category) : View | RedirectResponse
{ {
$this->authorize('view', Category::class); $this->authorize('view', Category::class);
if ($category = Category::find($id)) {
if ($category->category_type == 'asset') { if ($category->category_type == 'asset') {
$category_type = 'hardware'; $category_type = 'hardware';
$category_type_route = 'assets'; $category_type_route = 'assets';
@ -197,8 +186,5 @@ class CategoriesController extends Controller
return view('categories/view', compact('category')) return view('categories/view', compact('category'))
->with('category_type', $category_type) ->with('category_type', $category_type)
->with('category_type_route', $category_type_route); ->with('category_type_route', $category_type_route);
}
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
} }
} }

View file

@ -60,6 +60,7 @@ final class CompaniesController extends Controller
$company->phone = $request->input('phone'); $company->phone = $request->input('phone');
$company->fax = $request->input('fax'); $company->fax = $request->input('fax');
$company->email = $request->input('email'); $company->email = $request->input('email');
$company->notes = $request->input('notes');
$company->created_by = auth()->id(); $company->created_by = auth()->id();
$company = $request->handleImages($company); $company = $request->handleImages($company);
@ -79,16 +80,10 @@ final class CompaniesController extends Controller
* @since [v1.8] * @since [v1.8]
* @param int $companyId * @param int $companyId
*/ */
public function edit($companyId) : View | RedirectResponse public function edit(Company $company) : View | RedirectResponse
{ {
if (is_null($item = Company::find($companyId))) { $this->authorize('update', $company);
return redirect()->route('companies.index') return view('companies/edit')->with('item', $company);
->with('error', trans('admin/companies/message.does_not_exist'));
}
$this->authorize('update', $item);
return view('companies/edit')->with('item', $item);
} }
/** /**
@ -99,18 +94,15 @@ final class CompaniesController extends Controller
* @param ImageUploadRequest $request * @param ImageUploadRequest $request
* @param int $companyId * @param int $companyId
*/ */
public function update(ImageUploadRequest $request, $companyId) : RedirectResponse public function update(ImageUploadRequest $request, Company $company) : RedirectResponse
{ {
if (is_null($company = Company::find($companyId))) {
return redirect()->route('companies.index')->with('error', trans('admin/companies/message.does_not_exist'));
}
$this->authorize('update', $company); $this->authorize('update', $company);
$company->name = $request->input('name'); $company->name = $request->input('name');
$company->phone = $request->input('phone'); $company->phone = $request->input('phone');
$company->fax = $request->input('fax'); $company->fax = $request->input('fax');
$company->email = $request->input('email'); $company->email = $request->input('email');
$company->notes = $request->input('notes');
$company = $request->handleImages($company); $company = $request->handleImages($company);
@ -156,15 +148,9 @@ final class CompaniesController extends Controller
->with('success', trans('admin/companies/message.delete.success')); ->with('success', trans('admin/companies/message.delete.success'));
} }
public function show($id) : View | RedirectResponse public function show(Company $company) : View | RedirectResponse
{ {
$this->authorize('view', Company::class); $this->authorize('view', Company::class);
if (is_null($company = Company::find($id))) {
return redirect()->route('companies.index')
->with('error', trans('admin/companies/message.not_found'));
}
return view('companies/view')->with('company', $company); return view('companies/view')->with('company', $company);
} }
} }

View file

@ -107,15 +107,13 @@ class ComponentsController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function edit($componentId = null) public function edit(Component $component)
{ {
if ($item = Component::find($componentId)) {
$this->authorize('update', $item);
return view('components/edit', compact('item'))->with('category_type', 'component'); $this->authorize('update', $component);
} return view('components/edit')
->with('item', $component)
return redirect()->route('components.index')->with('error', trans('admin/components/message.does_not_exist')); ->with('category_type', 'component');
} }
@ -130,11 +128,8 @@ class ComponentsController extends Controller
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
* @since [v3.0] * @since [v3.0]
*/ */
public function update(ImageUploadRequest $request, $componentId = null) public function update(ImageUploadRequest $request, Component $component)
{ {
if (is_null($component = Component::find($componentId))) {
return redirect()->route('components.index')->with('error', trans('admin/components/message.does_not_exist'));
}
$min = $component->numCheckedOut(); $min = $component->numCheckedOut();
$validator = Validator::make($request->all(), [ $validator = Validator::make($request->all(), [
'qty' => "required|numeric|min:$min", 'qty' => "required|numeric|min:$min",
@ -201,6 +196,10 @@ class ComponentsController extends Controller
} }
} }
if ($component->numCheckedOut() > 0) {
return redirect()->route('components.index')->with('error', trans('admin/components/message.delete.error_qty'));
}
$component->delete(); $component->delete();
return redirect()->route('components.index')->with('success', trans('admin/components/message.delete.success')); return redirect()->route('components.index')->with('success', trans('admin/components/message.delete.success'));
@ -216,17 +215,9 @@ class ComponentsController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function show($componentId = null) public function show(Component $component)
{ {
$component = Component::find($componentId);
if (isset($component->id)) {
$this->authorize('view', $component); $this->authorize('view', $component);
return view('components/view', compact('component')); return view('components/view', compact('component'));
}
// Redirect to the user management page
return redirect()->route('components.index')
->with('error', trans('admin/components/message.does_not_exist'));
} }
} }

View file

@ -104,15 +104,13 @@ class ConsumablesController extends Controller
* @see ConsumablesController::postEdit() method that stores the form data. * @see ConsumablesController::postEdit() method that stores the form data.
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($consumableId = null) : View | RedirectResponse public function edit(Consumable $consumable) : View | RedirectResponse
{ {
if ($item = Consumable::find($consumableId)) { $this->authorize($consumable);
$this->authorize($item); return view('consumables/edit')
->with('item', $consumable)
->with('category_type', 'consumable');
return view('consumables/edit', compact('item'))->with('category_type', 'consumable');
}
return redirect()->route('consumables.index')->with('error', trans('admin/consumables/message.does_not_exist'));
} }
/** /**
@ -126,11 +124,8 @@ class ConsumablesController extends Controller
* @see ConsumablesController::getEdit() method that stores the form data. * @see ConsumablesController::getEdit() method that stores the form data.
* @since [v1.0] * @since [v1.0]
*/ */
public function update(StoreConsumableRequest $request, $consumableId = null) public function update(StoreConsumableRequest $request, Consumable $consumable)
{ {
if (is_null($consumable = Consumable::find($consumableId))) {
return redirect()->route('consumables.index')->with('error', trans('admin/consumables/message.does_not_exist'));
}
$min = $consumable->numCheckedOut(); $min = $consumable->numCheckedOut();
$validator = Validator::make($request->all(), [ $validator = Validator::make($request->all(), [
@ -202,16 +197,11 @@ class ConsumablesController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function show($consumableId = null) public function show(Consumable $consumable)
{ {
$consumable = Consumable::withCount('users as users_consumables')->find($consumableId); $consumable = Consumable::withCount('users as users_consumables')->find($consumable->id);
$this->authorize($consumable); $this->authorize($consumable);
if (isset($consumable->id)) { return view('consumables/view', compact('consumable'));
return view('consumables/view', compact('consumable'));
}
return redirect()->route('consumables.index')
->with('error', trans('admin/consumables/message.does_not_exist'));
} }
public function clone(Consumable $consumable) : View public function clone(Consumable $consumable) : View

View file

@ -104,6 +104,8 @@ class CustomFieldsController extends Controller
"auto_add_to_fieldsets" => $request->get("auto_add_to_fieldsets", 0), "auto_add_to_fieldsets" => $request->get("auto_add_to_fieldsets", 0),
"show_in_listview" => $request->get("show_in_listview", 0), "show_in_listview" => $request->get("show_in_listview", 0),
"show_in_requestable_list" => $request->get("show_in_requestable_list", 0), "show_in_requestable_list" => $request->get("show_in_requestable_list", 0),
"display_checkin" => $request->get("display_checkin", 0),
"display_checkout" => $request->get("display_checkout", 0),
"created_by" => auth()->id() "created_by" => auth()->id()
]); ]);
@ -193,10 +195,8 @@ class CustomFieldsController extends Controller
* @param int $id * @param int $id
* @since [v4.0] * @since [v4.0]
*/ */
public function edit(Request $request, $id) : View | RedirectResponse public function edit(Request $request, CustomField $field) : View | RedirectResponse
{ {
if ($field = CustomField::find($id)) {
$this->authorize('update', $field); $this->authorize('update', $field);
$fieldsets = CustomFieldset::get(); $fieldsets = CustomFieldset::get();
$customFormat = ''; $customFormat = '';
@ -210,11 +210,7 @@ class CustomFieldsController extends Controller
'fieldsets' => $fieldsets, 'fieldsets' => $fieldsets,
'predefinedFormats' => Helper::predefined_formats(), 'predefinedFormats' => Helper::predefined_formats(),
]); ]);
}
return redirect()->route("fields.index")
->with("error", trans('admin/custom_fields/message.field.invalid'));
} }
@ -229,13 +225,9 @@ class CustomFieldsController extends Controller
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function update(CustomFieldRequest $request, $id) : RedirectResponse public function update(CustomFieldRequest $request, CustomField $field) : RedirectResponse
{ {
$field = CustomField::find($id);
$this->authorize('update', $field); $this->authorize('update', $field);
$show_in_email = $request->get("show_in_email", 0); $show_in_email = $request->get("show_in_email", 0);
$display_in_user_view = $request->get("display_in_user_view", 0); $display_in_user_view = $request->get("display_in_user_view", 0);
@ -256,6 +248,8 @@ class CustomFieldsController extends Controller
$field->auto_add_to_fieldsets = $request->get("auto_add_to_fieldsets", 0); $field->auto_add_to_fieldsets = $request->get("auto_add_to_fieldsets", 0);
$field->show_in_listview = $request->get("show_in_listview", 0); $field->show_in_listview = $request->get("show_in_listview", 0);
$field->show_in_requestable_list = $request->get("show_in_requestable_list", 0); $field->show_in_requestable_list = $request->get("show_in_requestable_list", 0);
$field->display_checkin = $request->get("display_checkin", 0);
$field->display_checkout = $request->get("display_checkout", 0);
if ($request->get('format') == 'CUSTOM REGEX') { if ($request->get('format') == 'CUSTOM REGEX') {
$field->format = e($request->get('custom_format')); $field->format = e($request->get('custom_format'));

View file

@ -35,10 +35,12 @@ class CustomFieldsetsController extends Controller
* @param int $id * @param int $id
* @since [v1.8] * @since [v1.8]
*/ */
public function show($id) : View | RedirectResponse public function show(CustomFieldset $fieldset) : View | RedirectResponse
{ {
$cfset = CustomFieldset::with('fields') $cfset = CustomFieldset::with('fields')
->where('id', '=', $id)->orderBy('id', 'ASC')->first(); ->where('id', '=', $fieldset->id)
->orderBy('id', 'ASC')
->first();
$this->authorize('view', $cfset); $this->authorize('view', $cfset);
@ -122,16 +124,10 @@ class CustomFieldsetsController extends Controller
* @param int $id * @param int $id
* @since [v6.0.14] * @since [v6.0.14]
*/ */
public function edit($id) : View | RedirectResponse public function edit(CustomFieldset $fieldset) : View | RedirectResponse
{ {
$this->authorize('create', CustomField::class); $this->authorize('create', CustomField::class);
return view('custom_fields.fieldsets.edit')->with('item', $fieldset);
if ($fieldset = CustomFieldset::find($id)) {
return view('custom_fields.fieldsets.edit')->with('item', $fieldset);
}
return redirect()->route('fields.index')->with('error', trans('admin/custom_fields/general.fieldset_does_not_exist', ['id' => $id]));
} }
/** /**
@ -141,23 +137,18 @@ class CustomFieldsetsController extends Controller
* @param int $id * @param int $id
* @since [v6.0.14] * @since [v6.0.14]
*/ */
public function update(Request $request, $id) : RedirectResponse public function update(Request $request, CustomFieldset $fieldset) : RedirectResponse
{ {
$this->authorize('create', CustomField::class); $this->authorize('create', CustomField::class);
if ($fieldset = CustomFieldset::find($id)) { $fieldset->name = $request->input('name');
$fieldset->name = $request->input('name');
if ($fieldset->save()) {
return redirect()->route('fields.index')->with('success', trans('admin/custom_fields/general.fieldset_updated'));
}
return redirect()->back()->withInput()->withErrors($fieldset->getErrors());
if ($fieldset->save()) {
return redirect()->route('fields.index')->with('success', trans('admin/custom_fields/general.fieldset_updated'));
} }
return redirect()->route('fields.index')->with('error', trans('admin/custom_fields/general.fieldset_does_not_exist', ['id' => $id])); return redirect()->back()->withInput()->withErrors($fieldset->getErrors());
} }
/** /**

View file

@ -55,6 +55,7 @@ class DepartmentsController extends Controller
$department->manager_id = ($request->filled('manager_id') ? $request->input('manager_id') : null); $department->manager_id = ($request->filled('manager_id') ? $request->input('manager_id') : null);
$department->location_id = ($request->filled('location_id') ? $request->input('location_id') : null); $department->location_id = ($request->filled('location_id') ? $request->input('location_id') : null);
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null); $department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
$department->notes = $request->input('notes');
$department = $request->handleImages($department); $department = $request->handleImages($department);
if ($department->save()) { if ($department->save()) {
@ -72,17 +73,10 @@ class DepartmentsController extends Controller
* @param int $id * @param int $id
* @since [v4.0] * @since [v4.0]
*/ */
public function show($id) : View | RedirectResponse public function show(Department $department) : View | RedirectResponse
{ {
$department = Department::find($id);
$this->authorize('view', $department); $this->authorize('view', $department);
return view('departments/view', compact('department'));
if (isset($department->id)) {
return view('departments/view', compact('department'));
}
return redirect()->route('departments.index')->with('error', trans('admin/departments/message.does_not_exist'));
} }
/** /**
@ -138,15 +132,10 @@ class DepartmentsController extends Controller
* @param int $departmentId * @param int $departmentId
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($departmentId = null) : View | RedirectResponse public function edit(Department $department) : View | RedirectResponse
{ {
if (is_null($item = Department::find($departmentId))) { $this->authorize('update', $department);
return redirect()->back()->with('error', trans('admin/locations/message.does_not_exist')); return view('departments/edit')->with('item', $department);
}
$this->authorize('update', $item);
return view('departments/edit', compact('item'));
} }
/** /**
@ -157,11 +146,8 @@ class DepartmentsController extends Controller
* @param int $departmentId * @param int $departmentId
* @since [v1.0] * @since [v1.0]
*/ */
public function update(ImageUploadRequest $request, $id) : RedirectResponse public function update(ImageUploadRequest $request, Department $department) : RedirectResponse
{ {
if (is_null($department = Department::find($id))) {
return redirect()->route('departments.index')->with('error', trans('admin/departments/message.does_not_exist'));
}
$this->authorize('update', $department); $this->authorize('update', $department);
@ -171,7 +157,7 @@ class DepartmentsController extends Controller
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null); $department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
$department->phone = $request->input('phone'); $department->phone = $request->input('phone');
$department->fax = $request->input('fax'); $department->fax = $request->input('fax');
$department->notes = $request->input('notes');
$department = $request->handleImages($department); $department = $request->handleImages($department);
if ($department->save()) { if ($department->save()) {

View file

@ -95,17 +95,11 @@ class DepreciationsController extends Controller
* @param int $depreciationId * @param int $depreciationId
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($depreciationId = null) : RedirectResponse | View public function edit(Depreciation $depreciation) : RedirectResponse | View
{ {
// Check if the depreciation exists
if (is_null($item = Depreciation::find($depreciationId))) {
// Redirect to the blogs management page
return redirect()->route('depreciations.index')->with('error', trans('admin/depreciations/message.does_not_exist'));
}
$this->authorize('update', $item); $this->authorize('update', $depreciation);
return view('depreciations/edit')->with('item', $depreciation);
return view('depreciations/edit', compact('item'));
} }
/** /**
@ -117,17 +111,10 @@ class DepreciationsController extends Controller
* @param int $depreciationId * @param int $depreciationId
* @since [v1.0] * @since [v1.0]
*/ */
public function update(Request $request, $depreciationId = null) : RedirectResponse public function update(Request $request, Depreciation $depreciation) : RedirectResponse
{ {
// Check if the depreciation exists
if (is_null($depreciation = Depreciation::find($depreciationId))) {
// Redirect to the blogs management page
return redirect()->route('depreciations.index')->with('error', trans('admin/depreciations/message.does_not_exist'));
}
$this->authorize('update', $depreciation); $this->authorize('update', $depreciation);
// Depreciation data
$depreciation->name = $request->input('name'); $depreciation->name = $request->input('name');
$depreciation->months = $request->input('months'); $depreciation->months = $request->input('months');
@ -191,12 +178,12 @@ class DepreciationsController extends Controller
* @param int $depreciationId * @param int $depreciationId
* @since [v1.0] * @since [v1.0]
*/ */
public function show($id) : View | RedirectResponse public function show(Depreciation $depreciation) : View | RedirectResponse
{ {
$depreciation = Depreciation::withCount('assets as assets_count') $depreciation = Depreciation::withCount('assets as assets_count')
->withCount('models as models_count') ->withCount('models as models_count')
->withCount('licenses as licenses_count') ->withCount('licenses as licenses_count')
->find($id); ->find($depreciation->id);
$this->authorize('view', $depreciation); $this->authorize('view', $depreciation);

View file

@ -62,6 +62,7 @@ class GroupsController extends Controller
$group->name = $request->input('name'); $group->name = $request->input('name');
$group->permissions = json_encode($request->input('permission')); $group->permissions = json_encode($request->input('permission'));
$group->created_by = auth()->id(); $group->created_by = auth()->id();
$group->notes = $request->input('notes');
if ($group->save()) { if ($group->save()) {
return redirect()->route('groups.index')->with('success', trans('admin/groups/message.success.create')); return redirect()->route('groups.index')->with('success', trans('admin/groups/message.success.create'));
@ -78,19 +79,12 @@ class GroupsController extends Controller
* @param int $id * @param int $id
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($id) : View | RedirectResponse public function edit(Group $group) : View | RedirectResponse
{ {
$group = Group::find($id); $permissions = config('permissions');
$groupPermissions = $group->decodePermissions();
if ($group) { $selected_array = Helper::selectedPermissionsArray($permissions, $groupPermissions);
$permissions = config('permissions'); return view('groups.edit', compact('group', 'permissions', 'selected_array', 'groupPermissions'));
$groupPermissions = $group->decodePermissions();
$selected_array = Helper::selectedPermissionsArray($permissions, $groupPermissions);
return view('groups.edit', compact('group', 'permissions', 'selected_array', 'groupPermissions'));
}
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
} }
/** /**
@ -101,13 +95,11 @@ class GroupsController extends Controller
* @param int $id * @param int $id
* @since [v1.0] * @since [v1.0]
*/ */
public function update(Request $request, $id = null) : RedirectResponse public function update(Request $request, Group $group) : RedirectResponse
{ {
if (! $group = Group::find($id)) {
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
}
$group->name = $request->input('name'); $group->name = $request->input('name');
$group->permissions = json_encode($request->input('permission')); $group->permissions = json_encode($request->input('permission'));
$group->notes = $request->input('notes');
if (! config('app.lock_passwords')) { if (! config('app.lock_passwords')) {
if ($group->save()) { if ($group->save()) {
@ -149,14 +141,8 @@ class GroupsController extends Controller
* @param $id * @param $id
* @since [v4.0.11] * @since [v4.0.11]
*/ */
public function show($id) : View | RedirectResponse public function show(Group $group) : View | RedirectResponse
{ {
$group = Group::find($id); return view('groups/view', compact('group'));
if ($group) {
return view('groups/view', compact('group'));
}
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
} }
} }

View file

@ -4,10 +4,8 @@ namespace App\Http\Controllers\Kits;
use App\Http\Controllers\CheckInOutRequest; use App\Http\Controllers\CheckInOutRequest;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Models\PredefinedKit;
use App\Models\Asset; use App\Models\Asset;
use App\Models\PredefinedLicence; use App\Models\PredefinedKit;
use App\Models\PredefinedModel;
use App\Models\User; use App\Models\User;
use App\Services\PredefinedKitCheckoutService; use App\Services\PredefinedKitCheckoutService;
use Illuminate\Http\Request; use Illuminate\Http\Request;
@ -35,12 +33,9 @@ class CheckoutKitController extends Controller
* @author [D. Minaev.] [<dmitriy.minaev.v@gmail.com>] * @author [D. Minaev.] [<dmitriy.minaev.v@gmail.com>]
* @return \Illuminate\Contracts\View\View View to checkout * @return \Illuminate\Contracts\View\View View to checkout
*/ */
public function showCheckout($kit_id) public function showCheckout(PredefinedKit $kit)
{ {
$this->authorize('checkout', Asset::class); $this->authorize('checkout', Asset::class);
$kit = PredefinedKit::findOrFail($kit_id);
return view('kits/checkout')->with('kit', $kit); return view('kits/checkout')->with('kit', $kit);
} }

View file

@ -76,17 +76,15 @@ class PredefinedKitsController extends Controller
* @param int $kit_id * @param int $kit_id
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
*/ */
public function edit($kit_id = null) public function edit(PredefinedKit $kit)
{ {
$this->authorize('update', PredefinedKit::class); $this->authorize('update', PredefinedKit::class);
if ($kit = PredefinedKit::find($kit_id)) {
return view('kits/edit') return view('kits/edit')
->with('item', $kit) ->with('item', $kit)
->with('models', $kit->models) ->with('models', $kit->models)
->with('licenses', $kit->licenses); ->with('licenses', $kit->licenses);
}
return redirect()->route('kits.index')->with('error', trans('admin/kits/general.kit_none'));
} }
/** /**
@ -98,15 +96,9 @@ class PredefinedKitsController extends Controller
* @param int $kit_id * @param int $kit_id
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
*/ */
public function update(ImageUploadRequest $request, $kit_id = null) public function update(ImageUploadRequest $request, PredefinedKit $kit)
{ {
$this->authorize('update', PredefinedKit::class); $this->authorize('update', PredefinedKit::class);
// Check if the kit exists
if (is_null($kit = PredefinedKit::find($kit_id))) {
// Redirect to the kits management page
return redirect()->route('kits.index')->with('error', trans('admin/kits/general.kit_none'));
}
$kit->name = $request->input('name'); $kit->name = $request->input('name');
if ($kit->save()) { if ($kit->save()) {
@ -153,9 +145,9 @@ class PredefinedKitsController extends Controller
* @param int $modelId * @param int $modelId
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
*/ */
public function show($kit_id = null) public function show(PredefinedKit $kit)
{ {
return $this->edit($kit_id); return $this->edit($kit);
} }
/** /**

View file

@ -28,16 +28,11 @@ class LicenseCheckinController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function create($seatId = null, $backTo = null) public function create(LicenseSeat $licenseSeat, $backTo = null)
{ {
// Check if the asset exists // Check if the asset exists
if (is_null($licenseSeat = LicenseSeat::find($seatId)) || is_null($license = License::find($licenseSeat->license_id))) { $license = License::find($licenseSeat->license_id);
// Redirect to the asset management page with error
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
}
$this->authorize('checkout', $license); $this->authorize('checkout', $license);
return view('licenses/checkin', compact('licenseSeat'))->with('backto', $backTo); return view('licenses/checkin', compact('licenseSeat'))->with('backto', $backTo);
} }

View file

@ -28,33 +28,24 @@ class LicenseCheckoutController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function create($id) public function create(License $license)
{ {
$this->authorize('checkout', $license);
if ($license = License::find($id)) { if ($license->category) {
$this->authorize('checkout', $license); // Make sure there is at least one available to checkout
if ($license->availCount()->count() < 1) {
if ($license->category) { return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.checkout.not_enough_seats'));
// Make sure there is at least one available to checkout
if ($license->availCount()->count() < 1){
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.checkout.not_enough_seats'));
}
// Return the checkout view
return view('licenses/checkout', compact('license'));
} }
// Invalid category // Return the checkout view
return redirect()->route('licenses.edit', ['license' => $license->id]) return view('licenses/checkout', compact('license'));
->with('error', trans('general.invalid_item_category_single', ['type' => trans('general.license')]));
} }
// Not found // Invalid category
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found')); return redirect()->route('licenses.edit', ['license' => $license->id])
->with('error', trans('general.invalid_item_category_single', ['type' => trans('general.license')]));
} }

View file

@ -121,13 +121,10 @@ class LicensesController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function edit($licenseId = null) public function edit(License $license)
{ {
if (is_null($item = License::find($licenseId))) {
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
}
$this->authorize('update', $item); $this->authorize('update', $license);
$maintained_list = [ $maintained_list = [
'' => 'Maintained', '' => 'Maintained',
@ -135,7 +132,8 @@ class LicensesController extends Controller
'0' => 'No', '0' => 'No',
]; ];
return view('licenses/edit', compact('item')) return view('licenses/edit')
->with('item', $license)
->with('depreciation_list', Helper::depreciationList()) ->with('depreciation_list', Helper::depreciationList())
->with('maintained_list', $maintained_list); ->with('maintained_list', $maintained_list);
} }
@ -153,11 +151,9 @@ class LicensesController extends Controller
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function update(Request $request, $licenseId = null) public function update(Request $request, License $license)
{ {
if (is_null($license = License::find($licenseId))) {
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
}
$this->authorize('update', $license); $this->authorize('update', $license);
@ -201,10 +197,10 @@ class LicensesController extends Controller
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function destroy($licenseId) public function destroy(License $license)
{ {
// Check if the license exists // Check if the license exists
if (is_null($license = License::find($licenseId))) { if (is_null($license = License::find($license->id))) {
// Redirect to the license management page // Redirect to the license management page
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found')); return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
} }
@ -238,14 +234,9 @@ class LicensesController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function show($licenseId = null) public function show(License $license)
{ {
$license = License::with('assignedusers')->find($licenseId); $license = License::with('assignedusers')->find($license->id);
if (!$license) {
return redirect()->route('licenses.index')
->with('error', trans('admin/licenses/message.does_not_exist'));
}
$users_count = User::where('autoassign_licenses', '1')->count(); $users_count = User::where('autoassign_licenses', '1')->count();
$total_seats_count = $license->totalSeatsByLicenseID(); $total_seats_count = $license->totalSeatsByLicenseID();
@ -267,10 +258,10 @@ class LicensesController extends Controller
* *
* @author [A. Gianotto] [<snipe@snipe.net>] * @author [A. Gianotto] [<snipe@snipe.net>]
* @param int $licenseId * @param int $licenseId
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse | \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function getClone($licenseId = null) public function getClone($licenseId = null) : \Illuminate\Contracts\View\View | \Illuminate\Http\RedirectResponse
{ {
if (is_null($license_to_clone = License::find($licenseId))) { if (is_null($license_to_clone = License::find($licenseId))) {
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist')); return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));

View file

@ -78,6 +78,7 @@ class LocationsController extends Controller
$location->created_by = auth()->id(); $location->created_by = auth()->id();
$location->phone = request('phone'); $location->phone = request('phone');
$location->fax = request('fax'); $location->fax = request('fax');
$location->notes = $request->input('notes');
$location = $request->handleImages($location); $location = $request->handleImages($location);
@ -96,15 +97,10 @@ class LocationsController extends Controller
* @param int $locationId * @param int $locationId
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($locationId = null) : View | RedirectResponse public function edit(Location $location) : View | RedirectResponse
{ {
$this->authorize('update', Location::class); $this->authorize('update', Location::class);
// Check if the location exists return view('locations/edit');
if (is_null($item = Location::find($locationId))) {
return redirect()->route('locations.index')->with('error', trans('admin/locations/message.does_not_exist'));
}
return view('locations/edit', compact('item'));
} }
/** /**
@ -116,15 +112,10 @@ class LocationsController extends Controller
* @param int $locationId * @param int $locationId
* @since [v1.0] * @since [v1.0]
*/ */
public function update(ImageUploadRequest $request, $locationId = null) : RedirectResponse public function update(ImageUploadRequest $request, Location $location) : RedirectResponse
{ {
$this->authorize('update', Location::class); $this->authorize('update', Location::class);
// Check if the location exists
if (is_null($location = Location::find($locationId))) {
return redirect()->route('locations.index')->with('error', trans('admin/locations/message.does_not_exist'));
}
// Update the location data
$location->name = $request->input('name'); $location->name = $request->input('name');
$location->parent_id = $request->input('parent_id', null); $location->parent_id = $request->input('parent_id', null);
$location->currency = $request->input('currency', '$'); $location->currency = $request->input('currency', '$');
@ -138,6 +129,7 @@ class LocationsController extends Controller
$location->fax = request('fax'); $location->fax = request('fax');
$location->ldap_ou = $request->input('ldap_ou'); $location->ldap_ou = $request->input('ldap_ou');
$location->manager_id = $request->input('manager_id'); $location->manager_id = $request->input('manager_id');
$location->notes = $request->input('notes');
$location = $request->handleImages($location); $location = $request->handleImages($location);
@ -192,7 +184,7 @@ class LocationsController extends Controller
* @param int $id * @param int $id
* @since [v1.0] * @since [v1.0]
*/ */
public function show($id = null) : View | RedirectResponse public function show(Location $location) : View | RedirectResponse
{ {
$location = Location::withCount('assignedAssets as assigned_assets_count') $location = Location::withCount('assignedAssets as assigned_assets_count')
->withCount('assets as assets_count') ->withCount('assets as assets_count')
@ -200,7 +192,7 @@ class LocationsController extends Controller
->withCount('children as children_count') ->withCount('children as children_count')
->withCount('users as users_count') ->withCount('users as users_count')
->withTrashed() ->withTrashed()
->find($id); ->find($location->id);
if (isset($location->id)) { if (isset($location->id)) {
return view('locations/view', compact('location')); return view('locations/view', compact('location'));

View file

@ -67,6 +67,7 @@ class ManufacturersController extends Controller
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url'); $manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
$manufacturer->support_phone = $request->input('support_phone'); $manufacturer->support_phone = $request->input('support_phone');
$manufacturer->support_email = $request->input('support_email'); $manufacturer->support_email = $request->input('support_email');
$manufacturer->notes = $request->input('notes');
$manufacturer = $request->handleImages($manufacturer); $manufacturer = $request->handleImages($manufacturer);
if ($manufacturer->save()) { if ($manufacturer->save()) {
@ -84,18 +85,10 @@ class ManufacturersController extends Controller
* @param int $manufacturerId * @param int $manufacturerId
* @since [v1.0] * @since [v1.0]
*/ */
public function edit($manufacturerId = null) : View | RedirectResponse public function edit(Manufacturer $manufacturer) : View | RedirectResponse
{ {
// Handles manufacturer checks and permissions.
$this->authorize('update', Manufacturer::class); $this->authorize('update', Manufacturer::class);
return view('manufacturers/edit')->with('item', $manufacturer);
// Check if the manufacturer exists
if (! $item = Manufacturer::find($manufacturerId)) {
return redirect()->route('manufacturers.index')->with('error', trans('admin/manufacturers/message.does_not_exist'));
}
// Show the page
return view('manufacturers/edit', compact('item'));
} }
/** /**
@ -107,22 +100,17 @@ class ManufacturersController extends Controller
* @param int $manufacturerId * @param int $manufacturerId
* @since [v1.0] * @since [v1.0]
*/ */
public function update(ImageUploadRequest $request, $manufacturerId = null) : RedirectResponse public function update(ImageUploadRequest $request, Manufacturer $manufacturer) : RedirectResponse
{ {
$this->authorize('update', Manufacturer::class); $this->authorize('update', Manufacturer::class);
// Check if the manufacturer exists
if (is_null($manufacturer = Manufacturer::find($manufacturerId))) {
// Redirect to the manufacturer page
return redirect()->route('manufacturers.index')->with('error', trans('admin/manufacturers/message.does_not_exist'));
}
// Save the data
$manufacturer->name = $request->input('name'); $manufacturer->name = $request->input('name');
$manufacturer->url = $request->input('url'); $manufacturer->url = $request->input('url');
$manufacturer->support_url = $request->input('support_url'); $manufacturer->support_url = $request->input('support_url');
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url'); $manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
$manufacturer->support_phone = $request->input('support_phone'); $manufacturer->support_phone = $request->input('support_phone');
$manufacturer->support_email = $request->input('support_email'); $manufacturer->support_email = $request->input('support_email');
$manufacturer->notes = $request->input('notes');
// Set the model's image property to null if the image is being deleted // Set the model's image property to null if the image is being deleted
if ($request->input('image_delete') == 1) { if ($request->input('image_delete') == 1) {
@ -183,18 +171,10 @@ class ManufacturersController extends Controller
* @param int $manufacturerId * @param int $manufacturerId
* @since [v1.0] * @since [v1.0]
*/ */
public function show($manufacturerId = null) : View | RedirectResponse public function show(Manufacturer $manufacturer) : View | RedirectResponse
{ {
$this->authorize('view', Manufacturer::class); $this->authorize('view', Manufacturer::class);
$manufacturer = Manufacturer::find($manufacturerId); return view('manufacturers/view', compact('manufacturer'));
if (isset($manufacturer->id)) {
return view('manufacturers/view', compact('manufacturer'));
}
$error = trans('admin/manufacturers/message.does_not_exist');
// Redirect to the user management page
return redirect()->route('manufacturers.index')->with('error', $error);
} }
/** /**

View file

@ -0,0 +1,42 @@
<?php
namespace App\Http\Controllers;
use App\Models\Actionlog;
use App\Models\Asset;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Validation\Rule;
class NotesController extends Controller
{
public function store(Request $request)
{
$this->authorize('update', Asset::class);
$validated = $request->validate([
'id' => 'required',
'note' => 'required|string|max:50000',
'type' => [
'required',
Rule::in(['asset']),
],
]);
$item = Asset::findOrFail($validated['id']);
$this->authorize('update', $item);
$logaction = new Actionlog;
$logaction->item_id = $item->id;
$logaction->item_type = get_class($item);
$logaction->note = $validated['note'];
$logaction->created_by = Auth::id();
$logaction->logaction('note added');
return redirect()
->route('hardware.show', $validated['id'])
->withFragment('history')
->with('success', trans('general.note_added'));
}
}

View file

@ -800,6 +800,7 @@ class SettingsController extends Controller
} }
if ($setting->save()) { if ($setting->save()) {
return redirect()->route('settings.labels.index') return redirect()->route('settings.labels.index')
->with('success', trans('admin/settings/message.update.success')); ->with('success', trans('admin/settings/message.update.success'));
} }
@ -869,7 +870,6 @@ class SettingsController extends Controller
} }
if ($setting->save()) { if ($setting->save()) {
$setting->update_client_side_cert_files();
return redirect()->route('settings.ldap.index') return redirect()->route('settings.ldap.index')
->with('success', trans('admin/settings/message.update.success')); ->with('success', trans('admin/settings/message.update.success'));
} }

View file

@ -26,14 +26,10 @@ class StatuslabelsController extends Controller
return view('statuslabels.index'); return view('statuslabels.index');
} }
public function show($id) : View | RedirectResponse public function show(Statuslabel $statuslabel) : View | RedirectResponse
{ {
$this->authorize('view', Statuslabel::class); $this->authorize('view', Statuslabel::class);
if ($statuslabel = Statuslabel::find($id)) { return view('statuslabels.view')->with('statuslabel', $statuslabel);
return view('statuslabels.view')->with('statuslabel', $statuslabel);
}
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
} }
/** /**
@ -91,20 +87,15 @@ class StatuslabelsController extends Controller
* *
* @param int $statuslabelId * @param int $statuslabelId
*/ */
public function edit($statuslabelId = null) : View | RedirectResponse public function edit(Statuslabel $statuslabel) : View | RedirectResponse
{ {
$this->authorize('update', Statuslabel::class); $this->authorize('update', Statuslabel::class);
// Check if the Statuslabel exists
if (is_null($item = Statuslabel::find($statuslabelId))) {
// Redirect to the blogs management page
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
}
$use_statuslabel_type = $item->getStatuslabelType();
$statuslabel_types = ['' => trans('admin/hardware/form.select_statustype')] + ['undeployable' => trans('admin/hardware/general.undeployable')] + ['pending' => trans('admin/hardware/general.pending')] + ['archived' => trans('admin/hardware/general.archived')] + ['deployable' => trans('admin/hardware/general.deployable')]; $statuslabel_types = ['' => trans('admin/hardware/form.select_statustype')] + ['undeployable' => trans('admin/hardware/general.undeployable')] + ['pending' => trans('admin/hardware/general.pending')] + ['archived' => trans('admin/hardware/general.archived')] + ['deployable' => trans('admin/hardware/general.deployable')];
return view('statuslabels/edit', compact('item', 'statuslabel_types'))->with('use_statuslabel_type', $use_statuslabel_type); return view('statuslabels/edit', compact('statuslabel_types'))
->with('item', $statuslabel)
->with('use_statuslabel_type', $statuslabel);
} }
/** /**
@ -112,14 +103,9 @@ class StatuslabelsController extends Controller
* *
* @param int $statuslabelId * @param int $statuslabelId
*/ */
public function update(Request $request, $statuslabelId = null) : RedirectResponse public function update(Request $request, Statuslabel $statuslabel) : RedirectResponse
{ {
$this->authorize('update', Statuslabel::class); $this->authorize('update', Statuslabel::class);
// Check if the Statuslabel exists
if (is_null($statuslabel = Statuslabel::find($statuslabelId))) {
// Redirect to the blogs management page
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
}
if (! $request->filled('statuslabel_types')) { if (! $request->filled('statuslabel_types')) {
return redirect()->back()->withInput()->withErrors(['statuslabel_types' => trans('validation.statuslabel_type')]); return redirect()->back()->withInput()->withErrors(['statuslabel_types' => trans('validation.statuslabel_type')]);

View file

@ -77,17 +77,10 @@ class SuppliersController extends Controller
* *
* @param int $supplierId * @param int $supplierId
*/ */
public function edit($supplierId = null) : View | RedirectResponse public function edit(Supplier $supplier) : View | RedirectResponse
{ {
$this->authorize('update', Supplier::class); $this->authorize('update', Supplier::class);
// Check if the supplier exists return view('suppliers/edit')->with('item', $supplier);
if (is_null($item = Supplier::find($supplierId))) {
// Redirect to the supplier page
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
}
// Show the page
return view('suppliers/edit', compact('item'));
} }
/** /**
@ -95,14 +88,9 @@ class SuppliersController extends Controller
* *
* @param int $supplierId * @param int $supplierId
*/ */
public function update($supplierId, ImageUploadRequest $request) : RedirectResponse public function update(ImageUploadRequest $request, Supplier $supplier) : RedirectResponse
{ {
$this->authorize('update', Supplier::class); $this->authorize('update', Supplier::class);
if (is_null($supplier = Supplier::find($supplierId))) {
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
}
// Save the data // Save the data
$supplier->name = request('name'); $supplier->name = request('name');
$supplier->address = request('address'); $supplier->address = request('address');
@ -163,15 +151,10 @@ class SuppliersController extends Controller
* @param null $supplierId * @param null $supplierId
* @internal param int $assetId * @internal param int $assetId
*/ */
public function show($supplierId = null) : View | RedirectResponse public function show(Supplier $supplier) : View | RedirectResponse
{ {
$this->authorize('view', Supplier::class); $this->authorize('view', Supplier::class);
$supplier = Supplier::find($supplierId); return view('suppliers/view', compact('supplier'));
if (isset($supplier->id)) {
return view('suppliers/view', compact('supplier'));
}
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
} }
} }

View file

@ -15,6 +15,7 @@ use App\Models\ConsumableAssignment;
use App\Models\Consumable; use App\Models\Consumable;
use App\Models\Setting; use App\Models\Setting;
use App\Models\User; use App\Models\User;
use App\Notifications\CurrentInventory;
use Carbon\Carbon; use Carbon\Carbon;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Illuminate\Support\Collection; use Illuminate\Support\Collection;
@ -52,6 +53,28 @@ class BulkUsersController extends Controller
return view('users/bulk-edit', compact('users')) return view('users/bulk-edit', compact('users'))
->with('groups', Group::pluck('name', 'id')); ->with('groups', Group::pluck('name', 'id'));
// bulk send assigned inventory
} elseif ($request->input('bulk_actions') == 'send_assigned') {
$this->authorize('update', User::class);
$users_without_email = 0;
foreach ($users as $user) {
if (empty($user->email)) {
$users_without_email++;
} else {
$user->notify((new CurrentInventory($user)));
}
}
if ($users_without_email == 0) {
return redirect()->back()->with('success', trans_choice('admin/users/general.users_notified', $users->count()));
} else {
return redirect()->back()->with('warning', trans_choice('admin/users/general.users_notified_warning', $users->count(), ['no_email' => $users_without_email]));
}
// bulk delete, display the bulk delete confirmation form // bulk delete, display the bulk delete confirmation form
} elseif ($request->input('bulk_actions') == 'delete') { } elseif ($request->input('bulk_actions') == 'delete') {
$this->authorize('delete', User::class); $this->authorize('delete', User::class);

View file

@ -22,43 +22,34 @@ class UserFilesController extends Controller
*@author [A. Gianotto] [<snipe@snipe.net>] *@author [A. Gianotto] [<snipe@snipe.net>]
* @since [v1.6] * @since [v1.6]
*/ */
public function store(UploadFileRequest $request, $userId = null) public function store(UploadFileRequest $request, User $user)
{ {
$user = User::find($userId); $this->authorize('update', $user);
$destinationPath = config('app.private_uploads').'/users'; $files = $request->file('file');
if (isset($user->id)) { if (is_null($files)) {
$this->authorize('update', $user); return redirect()->back()->with('error', trans('admin/users/message.upload.nofiles'));
}
$logActions = []; foreach ($files as $file) {
$files = $request->file('file'); $file_name = $request->handleFile('private_uploads/users/', 'user-'.$user->id, $file);
if (is_null($files)) { //Log the uploaded file to the log
return redirect()->back()->with('error', trans('admin/users/message.upload.nofiles')); $logAction = new Actionlog();
} $logAction->item_id = $user->id;
foreach ($files as $file) { $logAction->item_type = User::class;
$file_name = $request->handleFile('private_uploads/users/', 'user-'.$user->id, $file); $logAction->created_by = auth()->id();
$logAction->note = $request->input('notes');
//Log the uploaded file to the log $logAction->target_id = null;
$logAction = new Actionlog(); $logAction->created_at = date("Y-m-d H:i:s");
$logAction->item_id = $user->id; $logAction->filename = $file_name;
$logAction->item_type = User::class; $logAction->action_type = 'uploaded';
$logAction->created_by = auth()->id();
$logAction->note = $request->input('notes'); if (! $logAction->save()) {
$logAction->target_id = null; return JsonResponse::create(['error' => 'Failed validation: '.print_r($logAction->getErrors(), true)], 500);
$logAction->created_at = date("Y-m-d H:i:s"); }
$logAction->filename = $file_name;
$logAction->action_type = 'uploaded'; return redirect()->back()->withFragment('files')->with('success', trans('admin/users/message.upload.success'));
if (! $logAction->save()) {
return JsonResponse::create(['error' => 'Failed validation: '.print_r($logAction->getErrors(), true)], 500);
}
$logActions[] = $logAction;
}
// dd($logActions);
return redirect()->back()->withFragment('files')->with('success', trans('admin/users/message.upload.success'));
} }
return redirect()->back()->with('error', trans('admin/users/message.upload.nofiles'));
} }
@ -110,7 +101,7 @@ class UserFilesController extends Controller
* @return mixed * @return mixed
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function show($userId = null, $fileId = null) public function show(User $user, $fileId = null)
{ {
@ -118,29 +109,21 @@ class UserFilesController extends Controller
return redirect()->route('users.show')->with('error', 'Invalid file request'); return redirect()->route('users.show')->with('error', 'Invalid file request');
} }
if ($user = User::find($userId)) {
$this->authorize('view', $user); $this->authorize('view', $user);
if ($log = Actionlog::whereNotNull('filename')->where('item_id', $user->id)->find($fileId)) { if ($log = Actionlog::whereNotNull('filename')->where('item_id', $user->id)->find($fileId)) {
$file = 'private_uploads/users/'.$log->filename; $file = 'private_uploads/users/'.$log->filename;
try { try {
return StorageHelper::showOrDownloadFile($file, $log->filename); return StorageHelper::showOrDownloadFile($file, $log->filename);
} catch (\Exception $e) { } catch (\Exception $e) {
return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.file_not_found')); return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.file_not_found'));
}
} }
// The log record doesn't exist somehow
return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.log_record_not_found'));
return redirect()->back()->with('error', trans('general.file_not_found'));
} }
// Redirect to the user management page if the user doesn't exist // The log record doesn't exist somehow
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', ['id' => $userId])); return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.log_record_not_found'));
} }
} }

View file

@ -182,11 +182,11 @@ class UsersController extends Controller
* @internal param int $id * @internal param int $id
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function edit($id) public function edit(User $user)
{ {
$this->authorize('update', User::class); $this->authorize('update', User::class);
$user = User::with(['assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc'])->withTrashed()->find($id); $user = User::with(['assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc'])->withTrashed()->find($user->id);
if ($user) { if ($user) {
@ -198,7 +198,7 @@ class UsersController extends Controller
$userPermissions = Helper::selectedPermissionsArray($permissions, $user->permissions); $userPermissions = Helper::selectedPermissionsArray($permissions, $user->permissions);
$permissions = $this->filterDisplayable($permissions); $permissions = $this->filterDisplayable($permissions);
return view('users/edit', compact('user', 'groups', 'userGroups', 'permissions', 'userPermissions'))->with('item', $user); return view('users/edit', compact('user', 'groups', 'userGroups', 'permissions', 'userPermissions'));
} }
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', compact('id'))); return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', compact('id')));
@ -324,7 +324,7 @@ class UsersController extends Controller
* @return \Illuminate\Http\RedirectResponse * @return \Illuminate\Http\RedirectResponse
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function destroy(DeleteUserRequest $request, $id = null) public function destroy(DeleteUserRequest $request, $id)
{ {
$this->authorize('delete', User::class); $this->authorize('delete', User::class);
@ -333,13 +333,6 @@ class UsersController extends Controller
$this->authorize('delete', $user); $this->authorize('delete', $user);
if ($user->delete()) { if ($user->delete()) {
if (Storage::disk('public')->exists('avatars/' . $user->avatar)) {
try {
Storage::disk('public')->delete('avatars/' . $user->avatar);
} catch (\Exception $e) {
Log::debug($e);
}
}
return redirect()->route('users.index')->with('success', trans('admin/users/message.success.delete')); return redirect()->route('users.index')->with('success', trans('admin/users/message.success.delete'));
} }
} }
@ -398,23 +391,18 @@ class UsersController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function show($userId = null) public function show(User $user)
{ {
// Make sure the user can view users at all // Make sure the user can view users at all
$this->authorize('view', User::class); $this->authorize('view', User::class);
$user = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($userId); $user = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($user->id);
// Make sure they can view this particular user // Make sure they can view this particular user
$this->authorize('view', $user); $this->authorize('view', $user);
if ($user) {
$userlog = $user->userlog->load('item'); $userlog = $user->userlog->load('item');
return view('users/view', compact('user', 'userlog'))->with('settings', Setting::getSettings()); return view('users/view', compact('user', 'userlog'))->with('settings', Setting::getSettings());
}
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', ['id' => $userId]));
} }
@ -428,7 +416,7 @@ class UsersController extends Controller
* @return \Illuminate\Contracts\View\View * @return \Illuminate\Contracts\View\View
* @throws \Illuminate\Auth\Access\AuthorizationException * @throws \Illuminate\Auth\Access\AuthorizationException
*/ */
public function getClone(Request $request, $id = null) public function getClone(Request $request, User $user)
{ {
$this->authorize('create', User::class); $this->authorize('create', User::class);
@ -438,7 +426,7 @@ class UsersController extends Controller
app('request')->request->set('permissions', $permissions); app('request')->request->set('permissions', $permissions);
$user_to_clone = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($id); $user_to_clone = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($user->id);
// Make sure they can view this particular user // Make sure they can view this particular user
$this->authorize('view', $user_to_clone); $this->authorize('view', $user_to_clone);

View file

@ -24,7 +24,7 @@ class DeleteUserRequest extends FormRequest
public function prepareForValidation(): void public function prepareForValidation(): void
{ {
$user_to_delete = User::withTrashed()->find(request()->route('user')); $user_to_delete = User::withTrashed()->with('managesUsers')->find(request()->route('user'));
if ($user_to_delete) { if ($user_to_delete) {
$this->merge([ $this->merge([
@ -61,7 +61,8 @@ class DeleteUserRequest extends FormRequest
public function messages(): array public function messages(): array
{ {
$user_to_delete = User::withTrashed()->find(request()->route('user')); $user_to_delete = User::withTrashed()->with('managesUsers')->find(request()->route('user'));
$messages = []; $messages = [];
if ($user_to_delete) { if ($user_to_delete) {

View file

@ -62,7 +62,7 @@ class SettingsSamlRequest extends FormRequest
$custom_privateKey = ''; $custom_privateKey = '';
$custom_x509certNew = ''; $custom_x509certNew = '';
if (! empty($this->input('saml_custom_settings'))) { if (! empty($this->input('saml_custom_settings'))) {
$req_custom_settings = preg_split('/\r\n|\r|\n/', $this->input('saml_custom_settings')); $req_custom_settings = preg_split('/\r\n|\r|\n/', $this->input('saml_custom_settings', ''));
$custom_settings = []; $custom_settings = [];
foreach ($req_custom_settings as $custom_setting) { foreach ($req_custom_settings as $custom_setting) {

View file

@ -2,8 +2,11 @@
namespace App\Http\Requests; namespace App\Http\Requests;
use App\Models\Labels\Label;
use Illuminate\Foundation\Http\FormRequest; use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Support\Facades\Gate; use Illuminate\Support\Facades\Gate;
use Illuminate\Validation\Rule;
class StoreLabelSettings extends FormRequest class StoreLabelSettings extends FormRequest
{ {
@ -22,6 +25,10 @@ class StoreLabelSettings extends FormRequest
*/ */
public function rules(): array public function rules(): array
{ {
$names = Label::find()?->map(function ($label) {
return $label->getName();
})->values()->toArray();
return [ return [
'labels_per_page' => 'numeric', 'labels_per_page' => 'numeric',
'labels_width' => 'numeric', 'labels_width' => 'numeric',
@ -36,6 +43,10 @@ class StoreLabelSettings extends FormRequest
'labels_pagewidth' => 'numeric|nullable', 'labels_pagewidth' => 'numeric|nullable',
'labels_pageheight' => 'numeric|nullable', 'labels_pageheight' => 'numeric|nullable',
'qr_text' => 'max:31|nullable', 'qr_text' => 'max:31|nullable',
'label2_template' => [
'required',
Rule::in($names),
],
]; ];
} }
} }

View file

@ -140,13 +140,13 @@ class ActionlogsTransformer
} else { } else {
if ($actionlog->item) { if ($actionlog->item) {
if ($actionlog->itemType() == 'asset') { if ($actionlog->itemType() == 'asset') {
$file_url = route('show/assetfile', ['assetId' => $actionlog->item->id, 'fileId' => $actionlog->id]); $file_url = route('show/assetfile', ['asset' => $actionlog->item->id, 'fileId' => $actionlog->id]);
} elseif ($actionlog->itemType() == 'accessory') { } elseif ($actionlog->itemType() == 'accessory') {
$file_url = route('show.accessoryfile', ['accessoryId' => $actionlog->item->id, 'fileId' => $actionlog->id]); $file_url = route('show.accessoryfile', ['accessoryId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
} elseif ($actionlog->itemType() == 'license') { } elseif ($actionlog->itemType() == 'license') {
$file_url = route('show.licensefile', ['licenseId' => $actionlog->item->id, 'fileId' => $actionlog->id]); $file_url = route('show.licensefile', ['licenseId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
} elseif ($actionlog->itemType() == 'user') { } elseif ($actionlog->itemType() == 'user') {
$file_url = route('show/userfile', ['userId' => $actionlog->item->id, 'fileId' => $actionlog->id]); $file_url = route('show/userfile', ['user' => $actionlog->item->id, 'fileId' => $actionlog->id]);
} }
} }
} }

View file

@ -309,6 +309,7 @@ class AssetsTransformer
'id' => $accessory_checkout->accessory->id, 'id' => $accessory_checkout->accessory->id,
'name' => $accessory_checkout->accessory->name, 'name' => $accessory_checkout->accessory->name,
], ],
'assigned_to' => $accessory_checkout->assigned_to,
'image' => ($accessory_checkout->accessory->image) ? Storage::disk('public')->url('accessories/'.e($accessory_checkout->accessory->image)) : null, 'image' => ($accessory_checkout->accessory->image) ? Storage::disk('public')->url('accessories/'.e($accessory_checkout->accessory->image)) : null,
'note' => $accessory_checkout->note ? e($accessory_checkout->note) : null, 'note' => $accessory_checkout->note ? e($accessory_checkout->note) : null,
'created_by' => $accessory_checkout->adminuser ? [ 'created_by' => $accessory_checkout->adminuser ? [

View file

@ -66,6 +66,7 @@ class CategoriesTransformer
'id' => (int) $category->adminuser->id, 'id' => (int) $category->adminuser->id,
'name'=> e($category->adminuser->present()->fullName()), 'name'=> e($category->adminuser->present()->fullName()),
] : null, ] : null,
'notes' => Helper::parseEscapedMarkedownInline($category->notes),
'created_at' => Helper::getFormattedDateObject($category->created_at, 'datetime'), 'created_at' => Helper::getFormattedDateObject($category->created_at, 'datetime'),
'updated_at' => Helper::getFormattedDateObject($category->updated_at, 'datetime'), 'updated_at' => Helper::getFormattedDateObject($category->updated_at, 'datetime'),
]; ];

View file

@ -40,6 +40,7 @@ class CompaniesTransformer
'id' => (int) $company->adminuser->id, 'id' => (int) $company->adminuser->id,
'name'=> e($company->adminuser->present()->fullName()), 'name'=> e($company->adminuser->present()->fullName()),
] : null, ] : null,
'notes' => Helper::parseEscapedMarkedownInline($company->notes),
'created_at' => Helper::getFormattedDateObject($company->created_at, 'datetime'), 'created_at' => Helper::getFormattedDateObject($company->created_at, 'datetime'),
'updated_at' => Helper::getFormattedDateObject($company->updated_at, 'datetime'), 'updated_at' => Helper::getFormattedDateObject($company->updated_at, 'datetime'),
]; ];

View file

@ -62,7 +62,7 @@ class ComponentsTransformer
'checkout' => Gate::allows('checkout', Component::class), 'checkout' => Gate::allows('checkout', Component::class),
'checkin' => Gate::allows('checkin', Component::class), 'checkin' => Gate::allows('checkin', Component::class),
'update' => Gate::allows('update', Component::class), 'update' => Gate::allows('update', Component::class),
'delete' => Gate::allows('delete', Component::class), 'delete' => $component->isDeletable(),
]; ];
$array += $permissions_array; $array += $permissions_array;

View file

@ -44,6 +44,7 @@ class DepartmentsTransformer
'name' => e($department->location->name), 'name' => e($department->location->name),
] : null, ] : null,
'users_count' => e($department->users_count), 'users_count' => e($department->users_count),
'notes' => Helper::parseEscapedMarkedownInline($department->notes),
'created_at' => Helper::getFormattedDateObject($department->created_at, 'datetime'), 'created_at' => Helper::getFormattedDateObject($department->created_at, 'datetime'),
'updated_at' => Helper::getFormattedDateObject($department->updated_at, 'datetime'), 'updated_at' => Helper::getFormattedDateObject($department->updated_at, 'datetime'),
]; ];

View file

@ -26,6 +26,7 @@ class GroupsTransformer
'name' => e($group->name), 'name' => e($group->name),
'permissions' => json_decode($group->permissions), 'permissions' => json_decode($group->permissions),
'users_count' => (int) $group->users_count, 'users_count' => (int) $group->users_count,
'notes' => Helper::parseEscapedMarkedownInline($group->notes),
'created_by' => ($group->adminuser) ? [ 'created_by' => ($group->adminuser) ? [
'id' => (int) $group->adminuser->id, 'id' => (int) $group->adminuser->id,
'name'=> e($group->adminuser->present()->fullName()), 'name'=> e($group->adminuser->present()->fullName()),

View file

@ -55,6 +55,7 @@ class LocationsTransformer
'users_count' => (int) $location->users_count, 'users_count' => (int) $location->users_count,
'currency' => ($location->currency) ? e($location->currency) : null, 'currency' => ($location->currency) ? e($location->currency) : null,
'ldap_ou' => ($location->ldap_ou) ? e($location->ldap_ou) : null, 'ldap_ou' => ($location->ldap_ou) ? e($location->ldap_ou) : null,
'notes' => Helper::parseEscapedMarkedownInline($location->notes),
'created_at' => Helper::getFormattedDateObject($location->created_at, 'datetime'), 'created_at' => Helper::getFormattedDateObject($location->created_at, 'datetime'),
'updated_at' => Helper::getFormattedDateObject($location->updated_at, 'datetime'), 'updated_at' => Helper::getFormattedDateObject($location->updated_at, 'datetime'),
'parent' => ($location->parent) ? [ 'parent' => ($location->parent) ? [
@ -99,6 +100,7 @@ class LocationsTransformer
$array = [ $array = [
'id' => $accessory_checkout->id, 'id' => $accessory_checkout->id,
'assigned_to' => $accessory_checkout->assigned_to,
'accessory' => [ 'accessory' => [
'id' => $accessory_checkout->accessory->id, 'id' => $accessory_checkout->accessory->id,
'name' => $accessory_checkout->accessory->name, 'name' => $accessory_checkout->accessory->name,

View file

@ -37,6 +37,7 @@ class ManufacturersTransformer
'consumables_count' => (int) $manufacturer->consumables_count, 'consumables_count' => (int) $manufacturer->consumables_count,
'accessories_count' => (int) $manufacturer->accessories_count, 'accessories_count' => (int) $manufacturer->accessories_count,
'components_count' => (int) $manufacturer->components_count, 'components_count' => (int) $manufacturer->components_count,
'notes' => Helper::parseEscapedMarkedownInline($manufacturer->notes),
'created_by' => ($manufacturer->adminuser) ? [ 'created_by' => ($manufacturer->adminuser) ? [
'id' => (int) $manufacturer->adminuser->id, 'id' => (int) $manufacturer->adminuser->id,
'name'=> e($manufacturer->adminuser->present()->fullName()), 'name'=> e($manufacturer->adminuser->present()->fullName()),

View file

@ -74,6 +74,8 @@ class LocationImporter extends ItemImporter
$this->item['ldap_ou'] = trim($this->findCsvMatch($row, 'ldap_ou')); $this->item['ldap_ou'] = trim($this->findCsvMatch($row, 'ldap_ou'));
$this->item['manager'] = trim($this->findCsvMatch($row, 'manager')); $this->item['manager'] = trim($this->findCsvMatch($row, 'manager'));
$this->item['manager_username'] = trim($this->findCsvMatch($row, 'manager_username')); $this->item['manager_username'] = trim($this->findCsvMatch($row, 'manager_username'));
$this->item['notes'] = trim($this->findCsvMatch($row, 'notes'));
if ($this->findCsvMatch($row, 'parent_location')) { if ($this->findCsvMatch($row, 'parent_location')) {
$this->item['parent_id'] = $this->createOrFetchLocation(trim($this->findCsvMatch($row, 'parent_location'))); $this->item['parent_id'] = $this->createOrFetchLocation(trim($this->findCsvMatch($row, 'parent_location')));

View file

@ -106,7 +106,13 @@ class CheckoutableListener
} }
} }
} catch (ClientException $e) { } catch (ClientException $e) {
Log::error("ClientException caught during checkin notification: " . $e->getMessage()); if (strpos($e->getMessage(), 'channel_not_found') !== false) {
Log::warning(Setting::getSettings()->webhook_selected." notification failed: " . $e->getMessage());
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_channel_not_found') );
}
else {
Log::error("ClientException caught during checkin notification: " . $e->getMessage());
}
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_fail') ); return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_fail') );
} catch (Exception $e) { } catch (Exception $e) {
Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [ Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
@ -156,7 +162,7 @@ class CheckoutableListener
$ccEmails = array_filter($adminCcEmailsArray); $ccEmails = array_filter($adminCcEmailsArray);
$mailable = $this->getCheckinMailType($event); $mailable = $this->getCheckinMailType($event);
$notifiable = $this->getNotifiables($event); $notifiable = $this->getNotifiables($event);
if ($event->checkedOutTo->locale){ if ($event->checkedOutTo?->locale) {
$mailable->locale($event->checkedOutTo->locale); $mailable->locale($event->checkedOutTo->locale);
} }
// Send email notifications // Send email notifications
@ -196,8 +202,14 @@ class CheckoutableListener
} }
} }
} catch (ClientException $e) { } catch (ClientException $e) {
Log::error("ClientException caught during checkin notification: " . $e->getMessage()); if (strpos($e->getMessage(), 'channel_not_found') !== false) {
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_fail')); Log::warning(Setting::getSettings()->webhook_selected." notification failed: " . $e->getMessage());
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_channel_not_found') );
}
else {
Log::error("ClientException caught during checkin notification: " . $e->getMessage());
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) . trans('admin/settings/message.webhook.webhook_fail'));
}
} catch (Exception $e) { } catch (Exception $e) {
Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [ Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
'error' => $e->getMessage(), 'error' => $e->getMessage(),

View file

@ -17,7 +17,6 @@ use App\Events\ItemAccepted;
use App\Events\ItemDeclined; use App\Events\ItemDeclined;
use App\Events\LicenseCheckedIn; use App\Events\LicenseCheckedIn;
use App\Events\LicenseCheckedOut; use App\Events\LicenseCheckedOut;
use App\Events\NoteAdded;
use App\Models\Actionlog; use App\Models\Actionlog;
use App\Models\User; use App\Models\User;
use App\Models\LicenseSeat; use App\Models\LicenseSeat;
@ -129,23 +128,6 @@ class LogListener
} }
/**
* Note is added to action log
*
*/
public function onNoteAdded(NoteAdded $event)
{
$logaction = new Actionlog();
$logaction->item_id = $event->itemNoteAddedOn->id;
$logaction->item_type = get_class($event->itemNoteAddedOn);
$logaction->note = $event->note; //this is the received alphanumeric text from the box
$logaction->created_by = $event->noteAddedBy->id;
$logaction->action_type = 'note_added';
$logaction->save();
}
/** /**
* Register the listeners for the subscriber. * Register the listeners for the subscriber.
* *

View file

@ -342,6 +342,7 @@ class Importer extends Component
'manager_username' => trans('general.importer.manager_username'), 'manager_username' => trans('general.importer.manager_username'),
'manager' => trans('general.importer.manager_full_name'), 'manager' => trans('general.importer.manager_full_name'),
'parent_location' => trans('admin/locations/table.parent'), 'parent_location' => trans('admin/locations/table.parent'),
'notes' => trans('general.notes'),
]; ];
$this->assetmodels_fields = [ $this->assetmodels_fields = [

View file

@ -0,0 +1,62 @@
<?php
namespace App\Mail;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Mail\Mailable;
use Illuminate\Mail\Mailables\Address;
use Illuminate\Mail\Mailables\Content;
use Illuminate\Mail\Mailables\Envelope;
use Illuminate\Queue\SerializesModels;
class ExpiringAssetsMail extends Mailable
{
use Queueable, SerializesModels;
/**
* Create a new message instance.
*/
public function __construct($params, $threshold)
{
$this->assets = $params;
$this->threshold = $threshold;
}
/**
* Get the message envelope.
*/
public function envelope(): Envelope
{
$from = new Address(config('mail.from.address'), config('mail.from.name'));
return new Envelope(
from: $from,
subject: trans('mail.Expiring_Assets_Report'),
);
}
/**
* Get the message content definition.
*/
public function content(): Content
{
return new Content(
markdown: 'notifications.markdown.report-expiring-assets',
with: [
'assets' => $this->assets,
'threshold' => $this->threshold,
]
);
}
/**
* Get the attachments for the message.
*
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
*/
public function attachments(): array
{
return [];
}
}

View file

@ -0,0 +1,62 @@
<?php
namespace App\Mail;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Mail\Mailable;
use Illuminate\Mail\Mailables\Address;
use Illuminate\Mail\Mailables\Content;
use Illuminate\Mail\Mailables\Envelope;
use Illuminate\Queue\SerializesModels;
class ExpiringLicenseMail extends Mailable
{
use Queueable, SerializesModels;
/**
* Create a new message instance.
*/
public function __construct($params, $threshold)
{
$this->licenses = $params;
$this->threshold = $threshold;
}
/**
* Get the message envelope.
*/
public function envelope(): Envelope
{
$from = new Address(config('mail.from.address'), config('mail.from.name'));
return new Envelope(
from: $from,
subject: trans('mail.Expiring_Licenses_Report'),
);
}
/**
* Get the message content definition.
*/
public function content(): Content
{
return new Content(
markdown: 'notifications.markdown.report-expiring-licenses',
with: [
'licenses' => $this->licenses,
'threshold' => $this->threshold,
]
);
}
/**
* Get the attachments for the message.
*
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
*/
public function attachments(): array
{
return [];
}
}

View file

@ -57,7 +57,7 @@ class AccessoryCheckout extends Model
*/ */
public function adminuser() public function adminuser()
{ {
return $this->hasOne(\App\Models\User::class, 'created_by'); return $this->hasOne(\App\Models\User::class, 'id', 'created_by');
} }
/** /**
@ -118,7 +118,7 @@ class AccessoryCheckout extends Model
$query->where('assigned_type', '=', Location::class); $query->where('assigned_type', '=', Location::class);
} }
public function scopeAssetAssigned(Builder $query): void public function scopeAssetsAssigned(Builder $query): void
{ {
$query->where('assigned_type', '=', Asset::class); $query->where('assigned_type', '=', Asset::class);
} }

View file

@ -413,6 +413,17 @@ class Asset extends Depreciable
return $this->rules; return $this->rules;
} }
public function customFieldsForCheckinCheckout($checkin_checkout) {
// Check to see if any of the custom fields were included on the form and if they have any values
if (($this->model) && ($this->model->fieldset) && ($this->model->fieldset->fields)) {
foreach ($this->model->fieldset->fields as $field) {
if (($field->{$checkin_checkout} == 1) && (request()->has($field->db_column))){
$this->{$field->db_column} = request()->get($field->db_column);
}
}
}
}
/** /**
* Establishes the asset -> depreciation relationship * Establishes the asset -> depreciation relationship
@ -523,6 +534,18 @@ class Asset extends Depreciable
return $this->morphMany(self::class, 'assigned', 'assigned_type', 'assigned_to')->withTrashed(); return $this->morphMany(self::class, 'assigned', 'assigned_type', 'assigned_to')->withTrashed();
} }
/**
* Establishes the accessory -> asset assignment relationship
*
* @author A. Gianotto <snipe@snipe.net>
* @since [v3.0]
* @return \Illuminate\Database\Eloquent\Relations\Relation
*/
public function assignedAccessories()
{
return $this->morphMany(\App\Models\AccessoryCheckout::class, 'assigned', 'assigned_type', 'assigned_to');
}
/** /**
* Get the asset's location based on the assigned user * Get the asset's location based on the assigned user

View file

@ -71,6 +71,7 @@ class Category extends SnipeModel
'require_acceptance', 'require_acceptance',
'use_default_eula', 'use_default_eula',
'created_by', 'created_by',
'notes',
]; ];
use Searchable; use Searchable;
@ -80,7 +81,7 @@ class Category extends SnipeModel
* *
* @var array * @var array
*/ */
protected $searchableAttributes = ['name', 'category_type']; protected $searchableAttributes = ['name', 'category_type', 'notes'];
/** /**
* The relations and their attributes that should be included when searching the model. * The relations and their attributes that should be included when searching the model.

View file

@ -6,6 +6,7 @@ use App\Models\Traits\Searchable;
use App\Presenters\Presentable; use App\Presenters\Presentable;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\SoftDeletes; use Illuminate\Database\Eloquent\SoftDeletes;
use Illuminate\Support\Facades\Gate;
use Watson\Validating\ValidatingTrait; use Watson\Validating\ValidatingTrait;
/** /**
@ -104,6 +105,13 @@ class Component extends SnipeModel
]; ];
public function isDeletable()
{
return Gate::allows('delete', $this)
&& ($this->numCheckedOut() === 0)
&& ($this->deleted_at == '');
}
/** /**
* Establishes the components -> action logs -> uploads relationship * Establishes the components -> action logs -> uploads relationship
* *
@ -234,13 +242,24 @@ class Component extends SnipeModel
// In case there are elements checked out to assets that belong to a different company // In case there are elements checked out to assets that belong to a different company
// than this asset and full multiple company support is on we'll remove the global scope, // than this asset and full multiple company support is on we'll remove the global scope,
// so they are included in the count. // so they are included in the count.
foreach ($this->assets()->withoutGlobalScope(new CompanyableScope)->get() as $checkout) { return $this->uncontrainedAssets->sum('pivot.assigned_qty');
$checkedout += $checkout->pivot->assigned_qty;
}
return $checkedout;
} }
/**
* @return \Illuminate\Database\Eloquent\Relations\BelongsToMany
*
* This allows us to get the assets with assigned components without the company restriction
*/
public function uncontrainedAssets() {
return $this->belongsToMany(\App\Models\Asset::class, 'components_assets')
->withPivot('id', 'assigned_qty', 'created_at', 'created_by', 'note')
->withoutGlobalScope(new CompanyableScope);
}
/** /**
* Check how many items within a component are remaining * Check how many items within a component are remaining
* *

View file

@ -307,7 +307,7 @@ class CustomField extends Model
public function formatFieldValuesAsArray() public function formatFieldValuesAsArray()
{ {
$result = []; $result = [];
$arr = preg_split('/\\r\\n|\\r|\\n/', $this->field_values); $arr = preg_split('/\\r\\n|\\r|\\n/', $this->field_values ?? '');
if (($this->element != 'checkbox') && ($this->element != 'radio')) { if (($this->element != 'checkbox') && ($this->element != 'radio')) {
$result[''] = 'Select '.strtolower($this->format); $result[''] = 'Select '.strtolower($this->format);

View file

@ -18,7 +18,8 @@ class Group extends SnipeModel
protected $fillable = [ protected $fillable = [
'name', 'name',
'permissions' 'permissions',
'notes',
]; ];
/** /**
@ -37,7 +38,7 @@ class Group extends SnipeModel
* *
* @var array * @var array
*/ */
protected $searchableAttributes = ['name', 'created_at']; protected $searchableAttributes = ['name', 'created_at', 'notes'];
/** /**
* The relations and their attributes that should be included when searching the model. * The relations and their attributes that should be included when searching the model.

View file

@ -411,14 +411,14 @@ abstract class Label
/** /**
* Checks the template is internally valid * Checks the template is internally valid
*/ */
public final function validate() { public final function validate() : void {
$this->validateUnits(); $this->validateUnits();
$this->validateSize(); $this->validateSize();
$this->validateMargins(); $this->validateMargins();
$this->validateSupport(); $this->validateSupport();
} }
private function validateUnits() { private function validateUnits() : void {
$validUnits = [ 'pt', 'mm', 'cm', 'in' ]; $validUnits = [ 'pt', 'mm', 'cm', 'in' ];
$unit = $this->getUnit(); $unit = $this->getUnit();
if (!in_array(strtolower($unit), $validUnits)) { if (!in_array(strtolower($unit), $validUnits)) {
@ -430,7 +430,7 @@ abstract class Label
} }
} }
private function validateSize() { private function validateSize() : void {
$width = $this->getWidth(); $width = $this->getWidth();
if (!is_numeric($width) || is_string($width)) { if (!is_numeric($width) || is_string($width)) {
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [ throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
@ -450,7 +450,7 @@ abstract class Label
} }
} }
private function validateMargins() { private function validateMargins() : void {
$marginTop = $this->getMarginTop(); $marginTop = $this->getMarginTop();
if (!is_numeric($marginTop) || is_string($marginTop)) { if (!is_numeric($marginTop) || is_string($marginTop)) {
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [ throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
@ -488,7 +488,7 @@ abstract class Label
} }
} }
private function validateSupport() { private function validateSupport() : void {
$support1D = $this->getSupport1DBarcode(); $support1D = $this->getSupport1DBarcode();
if (!is_bool($support1D)) { if (!is_bool($support1D)) {
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [ throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [

View file

@ -72,6 +72,7 @@ class Location extends SnipeModel
'currency', 'currency',
'manager_id', 'manager_id',
'image', 'image',
'notes',
]; ];
protected $hidden = ['user_id']; protected $hidden = ['user_id'];
@ -82,7 +83,7 @@ class Location extends SnipeModel
* *
* @var array * @var array
*/ */
protected $searchableAttributes = ['name', 'address', 'city', 'state', 'zip', 'created_at', 'ldap_ou', 'phone', 'fax']; protected $searchableAttributes = ['name', 'address', 'city', 'state', 'zip', 'created_at', 'ldap_ou', 'phone', 'fax', 'notes'];
/** /**
* The relations and their attributes that should be included when searching the model. * The relations and their attributes that should be included when searching the model.

View file

@ -34,7 +34,11 @@ trait Loggable
*/ */
public function logCheckout($note, $target, $action_date = null, $originalValues = []) public function logCheckout($note, $target, $action_date = null, $originalValues = [])
{ {
$log = new Actionlog; $log = new Actionlog;
$fields_array = [];
$log = $this->determineLogItemType($log); $log = $this->determineLogItemType($log);
if (auth()->user()) { if (auth()->user()) {
$log->created_by = auth()->id(); $log->created_by = auth()->id();
@ -55,6 +59,7 @@ trait Loggable
$log->target_type = get_class($target); $log->target_type = get_class($target);
$log->target_id = $target->id; $log->target_id = $target->id;
// Figure out what the target is // Figure out what the target is
if ($log->target_type == Location::class) { if ($log->target_type == Location::class) {
$log->location_id = $target->id; $log->location_id = $target->id;
@ -64,6 +69,21 @@ trait Loggable
$log->location_id = $target->location_id; $log->location_id = $target->location_id;
} }
if (static::class == Asset::class) {
if ($asset = Asset::find($log->item_id)) {
// add the custom fields that were changed
if ($asset->model->fieldset) {
$fields_array = [];
foreach ($asset->model->fieldset->fields as $field) {
if ($field->display_checkout == 1) {
$fields_array[$field->db_column] = $asset->{$field->db_column};
}
}
}
}
}
$log->note = $note; $log->note = $note;
$log->action_date = $action_date; $log->action_date = $action_date;
@ -72,7 +92,10 @@ trait Loggable
} }
$changed = []; $changed = [];
$originalValues = array_intersect_key($originalValues, array_flip(['action_date','name','status_id','location_id','expected_checkin'])); $array_to_flip = array_keys($fields_array);
$array_to_flip = array_merge($array_to_flip, ['action_date','name','status_id','location_id','expected_checkin']);
$originalValues = array_intersect_key($originalValues, array_flip($array_to_flip));
foreach ($originalValues as $key => $value) { foreach ($originalValues as $key => $value) {
if ($key == 'action_date' && $value != $action_date) { if ($key == 'action_date' && $value != $action_date) {
@ -119,6 +142,8 @@ trait Loggable
{ {
$log = new Actionlog; $log = new Actionlog;
$fields_array = [];
if($target != null){ if($target != null){
$log->target_type = get_class($target); $log->target_type = get_class($target);
$log->target_id = $target->id; $log->target_id = $target->id;
@ -135,6 +160,16 @@ trait Loggable
if (static::class == Asset::class) { if (static::class == Asset::class) {
if ($asset = Asset::find($log->item_id)) { if ($asset = Asset::find($log->item_id)) {
$asset->increment('checkin_counter', 1); $asset->increment('checkin_counter', 1);
// add the custom fields that were changed
if ($asset->model->fieldset) {
$fields_array = [];
foreach ($asset->model->fieldset->fields as $field) {
if ($field->display_checkin == 1) {
$fields_array[$field->db_column] = $asset->{$field->db_column};
}
}
}
} }
} }
} }
@ -152,9 +187,14 @@ trait Loggable
} }
$changed = []; $changed = [];
$originalValues = array_intersect_key($originalValues, array_flip(['action_date','name','status_id','location_id','rtd_location_id','expected_checkin']));
$array_to_flip = array_keys($fields_array);
$array_to_flip = array_merge($array_to_flip, ['action_date','name','status_id','location_id','expected_checkin']);
$originalValues = array_intersect_key($originalValues, array_flip($array_to_flip));
foreach ($originalValues as $key => $value) { foreach ($originalValues as $key => $value) {
if ($key == 'action_date' && $value != $action_date) { if ($key == 'action_date' && $value != $action_date) {
$changed[$key]['old'] = $value; $changed[$key]['old'] = $value;
$changed[$key]['new'] = is_string($action_date) ? $action_date : $action_date->format('Y-m-d H:i:s'); $changed[$key]['new'] = is_string($action_date) ? $action_date : $action_date->format('Y-m-d H:i:s');

View file

@ -53,6 +53,7 @@ class Manufacturer extends SnipeModel
'support_url', 'support_url',
'url', 'url',
'warranty_lookup_url', 'warranty_lookup_url',
'notes',
]; ];
use Searchable; use Searchable;
@ -62,7 +63,7 @@ class Manufacturer extends SnipeModel
* *
* @var array * @var array
*/ */
protected $searchableAttributes = ['name', 'created_at']; protected $searchableAttributes = ['name', 'created_at', 'notes'];
/** /**
* The relations and their attributes that should be included when searching the model. * The relations and their attributes that should be included when searching the model.

View file

@ -2,6 +2,7 @@
namespace App\Models; namespace App\Models;
use Carbon\Carbon;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Notifications\Notifiable; use Illuminate\Notifications\Notifiable;
@ -339,6 +340,33 @@ class Setting extends Model
return collect($ldapSettings); return collect($ldapSettings);
} }
/**
* For a particular cache-file, refresh it if the settings have
* been updated more recently than the file. Then return the
* full filepath
*/
public static function get_fresh_file_path($attribute, $path)
{
$full_path = storage_path().'/'.$path;
$file_exists = file_exists($full_path);
if ($file_exists) {
$statblock = stat($full_path);
}
if (!$file_exists || Carbon::createFromTimestamp($statblock['mtime']) < Setting::getSettings()->updated_at) {
if (Setting::getSettings()->{$attribute}) {
file_put_contents($full_path, Setting::getSettings()->{$attribute});
} else {
//this doesn't fire when you might expect it to because a lot of the time we do something like:
// if ($settings->ldap_client_tls_cert && ...
// so we never get a chance to 'uncache' the file.
if ($file_exists) {
unlink($full_path);
}
}
}
return $full_path;
}
/** /**
* Return the filename for the client-side SSL cert * Return the filename for the client-side SSL cert
* *
@ -346,7 +374,7 @@ class Setting extends Model
*/ */
public static function get_client_side_cert_path() public static function get_client_side_cert_path()
{ {
return storage_path().'/ldap_client_tls.cert'; return self::get_fresh_file_path('ldap_client_tls_cert', 'ldap_client_tls.cert');
} }
/** /**
@ -356,36 +384,7 @@ class Setting extends Model
*/ */
public static function get_client_side_key_path() public static function get_client_side_key_path()
{ {
return storage_path().'/ldap_client_tls.key'; return self::get_fresh_file_path('ldap_client_tls_key', 'ldap_client_tls.key');
} }
public function update_client_side_cert_files()
{
/**
* I'm not sure if it makes sense to have a cert but no key
* nor vice versa, but for now I'm just leaving it like this.
*
* Also, we could easily set this up with an event handler and
* self::saved() or something like that but there's literally only
* one place where we will do that, so I'll just explicitly call
* this method at that spot instead. It'll be easier to debug and understand.
*/
if ($this->ldap_client_tls_cert) {
file_put_contents(self::get_client_side_cert_path(), $this->ldap_client_tls_cert);
} else {
if (file_exists(self::get_client_side_cert_path())) {
unlink(self::get_client_side_cert_path());
}
}
if ($this->ldap_client_tls_key) {
file_put_contents(self::get_client_side_key_path(), $this->ldap_client_tls_key);
} else {
if (file_exists(self::get_client_side_key_path())) {
unlink(self::get_client_side_key_path());
}
}
}
} }

View file

@ -626,6 +626,8 @@ class User extends SnipeModel implements AuthenticatableContract, AuthorizableCo
$username = str_slug(substr($first_name, 0, 1).'.'.str_slug($last_name)); $username = str_slug(substr($first_name, 0, 1).'.'.str_slug($last_name));
} elseif ($format == 'lastname_firstinitial') { } elseif ($format == 'lastname_firstinitial') {
$username = str_slug($last_name).'_'.str_slug(substr($first_name, 0, 1)); $username = str_slug($last_name).'_'.str_slug(substr($first_name, 0, 1));
} elseif ($format == 'lastname.firstinitial') {
$username = str_slug($last_name).'.'.str_slug(substr($first_name, 0, 1));
} elseif ($format == 'firstnamelastname') { } elseif ($format == 'firstnamelastname') {
$username = str_slug($first_name).str_slug($last_name); $username = str_slug($first_name).str_slug($last_name);
} elseif ($format == 'firstnamelastinitial') { } elseif ($format == 'firstnamelastinitial') {

View file

@ -79,7 +79,7 @@ class CheckinAssetNotification extends Notification
$fields = [ $fields = [
trans('general.administrator') => '<'.$admin->present()->viewUrl().'|'.$admin->present()->fullName().'>', trans('general.administrator') => '<'.$admin->present()->viewUrl().'|'.$admin->present()->fullName().'>',
trans('general.status') => $item->assetstatus->name, trans('general.status') => $item->assetstatus?->name,
trans('general.location') => ($item->location) ? $item->location->name : '', trans('general.location') => ($item->location) ? $item->location->name : '',
]; ];
@ -106,9 +106,9 @@ class CheckinAssetNotification extends Notification
->title(trans('mail.Asset_Checkin_Notification')) ->title(trans('mail.Asset_Checkin_Notification'))
->addStartGroupToSection('activityText') ->addStartGroupToSection('activityText')
->fact(htmlspecialchars_decode($item->present()->name), '', 'activityText') ->fact(htmlspecialchars_decode($item->present()->name), '', 'activityText')
->fact(trans('mail.checked_into'), $item->location->name ? $item->location->name : '') ->fact(trans('mail.checked_into'), ($item->location) ? $item->location->name : '')
->fact(trans('mail.Asset_Checkin_Notification') . " by ", $admin->present()->fullName()) ->fact(trans('mail.Asset_Checkin_Notification') . " by ", $admin->present()->fullName())
->fact(trans('admin/hardware/form.status'), $item->assetstatus->name) ->fact(trans('admin/hardware/form.status'), $item->assetstatus?->name)
->fact(trans('mail.notes'), $note ?: ''); ->fact(trans('mail.notes'), $note ?: '');
} }
@ -116,9 +116,9 @@ class CheckinAssetNotification extends Notification
$message = trans('mail.Asset_Checkin_Notification'); $message = trans('mail.Asset_Checkin_Notification');
$details = [ $details = [
trans('mail.asset') => htmlspecialchars_decode($item->present()->name), trans('mail.asset') => htmlspecialchars_decode($item->present()->name),
trans('mail.checked_into') => $item->location->name ? $item->location->name : '', trans('mail.checked_into') => ($item->location) ? $item->location->name : '',
trans('mail.Asset_Checkin_Notification')." by " => $admin->present()->fullName(), trans('mail.Asset_Checkin_Notification')." by " => $admin->present()->fullName(),
trans('admin/hardware/form.status') => $item->assetstatus->name, trans('admin/hardware/form.status') => $item->assetstatus?->name,
trans('mail.notes') => $note ?: '', trans('mail.notes') => $note ?: '',
]; ];
@ -142,8 +142,8 @@ class CheckinAssetNotification extends Notification
Section::create( Section::create(
KeyValue::create( KeyValue::create(
trans('mail.checked_into') ?: '', trans('mail.checked_into') ?: '',
$item->location->name ? $item->location->name : '', ($item->location) ? $item->location->name : '',
trans('admin/hardware/form.status').": ".$item->assetstatus->name, trans('admin/hardware/form.status').": ".$item->assetstatus?->name,
) )
->onClick(route('hardware.show', $item->id)) ->onClick(route('hardware.show', $item->id))
) )

View file

@ -20,6 +20,7 @@ class AssetAuditPresenter extends Presenter
[ [
'field' => 'checkbox', 'field' => 'checkbox',
'checkbox' => true, 'checkbox' => true,
'titleTooltip' => trans('general.select_all_none'),
], ],
[ [
'field' => 'id', 'field' => 'id',

View file

@ -22,6 +22,15 @@ class AssetMaintenancesPresenter extends Presenter
'title' => trans('general.id'), 'title' => trans('general.id'),
'visible' => false, 'visible' => false,
], [ ], [
'field' => 'title',
'searchable' => true,
'sortable' => true,
'switchable' => true,
'title' => trans('general.name'),
'visible' => true,
'formatter' => 'maintenancesLinkFormatter',
],
[
'field' => 'company', 'field' => 'company',
'searchable' => true, 'searchable' => true,
'sortable' => true, 'sortable' => true,

View file

@ -15,6 +15,7 @@ class AssetModelPresenter extends Presenter
[ [
'field' => 'checkbox', 'field' => 'checkbox',
'checkbox' => true, 'checkbox' => true,
'titleTooltip' => trans('general.select_all_none'),
], ],
[ [
'field' => 'id', 'field' => 'id',

View file

@ -21,6 +21,7 @@ class AssetPresenter extends Presenter
[ [
'field' => 'checkbox', 'field' => 'checkbox',
'checkbox' => true, 'checkbox' => true,
'titleTooltip' => trans('general.select_all_none'),
], [ ], [
'field' => 'id', 'field' => 'id',
'searchable' => false, 'searchable' => false,
@ -35,7 +36,7 @@ class AssetPresenter extends Presenter
'switchable' => true, 'switchable' => true,
'title' => trans('general.company'), 'title' => trans('general.company'),
'visible' => false, 'visible' => false,
'formatter' => 'assetCompanyObjFilterFormatter', 'formatter' => 'companiesLinkObjFormatter',
], [ ], [
'field' => 'name', 'field' => 'name',
'searchable' => true, 'searchable' => true,
@ -352,6 +353,74 @@ class AssetPresenter extends Presenter
return json_encode($layout); return json_encode($layout);
} }
public static function assignedAccessoriesDataTableLayout()
{
$layout = [
[
'field' => 'id',
'searchable' => false,
'sortable' => false,
'switchable' => true,
'title' => trans('general.id'),
'visible' => false,
],
[
'field' => 'accessory',
'searchable' => false,
'sortable' => false,
'switchable' => true,
'title' => trans('general.accessory'),
'visible' => true,
'formatter' => 'accessoriesLinkObjFormatter',
],
[
'field' => 'image',
'searchable' => false,
'sortable' => false,
'switchable' => true,
'title' => trans('general.image'),
'visible' => true,
'formatter' => 'imageFormatter',
],
[
'field' => 'note',
'searchable' => false,
'sortable' => false,
'switchable' => true,
'title' => trans('general.notes'),
'visible' => true,
],
[
'field' => 'created_at',
'searchable' => false,
'sortable' => false,
'switchable' => true,
'title' => trans('admin/hardware/table.checkout_date'),
'visible' => true,
'formatter' => 'dateDisplayFormatter',
],
[
'field' => 'created_by',
'searchable' => false,
'sortable' => false,
'title' => trans('general.created_by'),
'visible' => false,
'formatter' => 'usersLinkObjFormatter',
],
[
'field' => 'available_actions',
'searchable' => false,
'sortable' => false,
'switchable' => false,
'title' => trans('table.actions'),
'formatter' => 'accessoriesInOutFormatter',
],
];
return json_encode($layout);
}
/** /**
* Generate html link to this items name. * Generate html link to this items name.
* @return string * @return string

View file

@ -77,7 +77,13 @@ class CategoryPresenter extends Presenter
"title" => trans('admin/categories/general.use_default_eula_column'), "title" => trans('admin/categories/general.use_default_eula_column'),
'visible' => true, 'visible' => true,
"formatter" => 'trueFalseFormatter', "formatter" => 'trueFalseFormatter',
],[ ], [
'field' => 'notes',
'searchable' => true,
'sortable' => true,
'visible' => false,
'title' => trans('general.notes'),
], [
'field' => 'created_by', 'field' => 'created_by',
'searchable' => false, 'searchable' => false,
'sortable' => true, 'sortable' => true,

View file

@ -105,7 +105,13 @@ class CompanyPresenter extends Presenter
'title' => trans('general.components'), 'title' => trans('general.components'),
'visible' => true, 'visible' => true,
'class' => 'css-component', 'class' => 'css-component',
],[ ], [
'field' => 'notes',
'searchable' => true,
'sortable' => true,
'visible' => false,
'title' => trans('general.notes'),
], [
'field' => 'created_by', 'field' => 'created_by',
'searchable' => false, 'searchable' => false,
'sortable' => true, 'sortable' => true,

View file

@ -102,13 +102,13 @@ class LocationPresenter extends Presenter
'titleTooltip' => trans('general.people'), 'titleTooltip' => trans('general.people'),
'visible' => true, 'visible' => true,
'class' => 'css-house-user', 'class' => 'css-house-user',
// 'data-tooltip' => true, - not working, but I want to try to use regular tooltips here
], [ ], [
'field' => 'currency', 'field' => 'currency',
'searchable' => true, 'searchable' => true,
'sortable' => true, 'sortable' => true,
'switchable' => true, 'switchable' => true,
'title' => trans('general.currency'), 'title' => trans('general.currency_text'),
'titleTooltip' => trans('general.currency_text'),
'visible' => true, 'visible' => true,
'class' => 'css-currency', 'class' => 'css-currency',
], [ ], [
@ -184,6 +184,12 @@ class LocationPresenter extends Presenter
'title' => trans('admin/users/table.manager'), 'title' => trans('admin/users/table.manager'),
'visible' => false, 'visible' => false,
'formatter' => 'usersLinkObjFormatter', 'formatter' => 'usersLinkObjFormatter',
], [
'field' => 'notes',
'searchable' => true,
'sortable' => true,
'visible' => false,
'title' => trans('general.notes'),
], [ ], [
'field' => 'created_at', 'field' => 'created_at',
'searchable' => true, 'searchable' => true,

View file

@ -28,7 +28,7 @@ class StatusLabelPresenter extends Presenter
'switchable' => false, 'switchable' => false,
'title' => trans('general.name'), 'title' => trans('general.name'),
'visible' => true, 'visible' => true,
'formatter' => 'statuslabelsAssetLinkFormatter', 'formatter' => 'statuslabelsLinkFormatter',
],[ ],[
'field' => 'type', 'field' => 'type',
'searchable' => false, 'searchable' => false,

View file

@ -24,6 +24,7 @@ class UserPresenter extends Presenter
[ [
'field' => 'checkbox', 'field' => 'checkbox',
'checkbox' => true, 'checkbox' => true,
'titleTooltip' => trans('general.select_all_none'),
], ],
[ [
'field' => 'id', 'field' => 'id',

View file

@ -87,9 +87,9 @@ class AuthServiceProvider extends ServiceProvider
]); ]);
$this->registerPolicies(); $this->registerPolicies();
Passport::tokensExpireIn(Carbon::now()->addYears(config('passport.expiration_years'))); Passport::tokensExpireIn(Carbon::now()->addYears((int)config('passport.expiration_years')));
Passport::refreshTokensExpireIn(Carbon::now()->addYears(config('passport.expiration_years'))); Passport::refreshTokensExpireIn(Carbon::now()->addYears((int)config('passport.expiration_years')));
Passport::personalAccessTokensExpireIn(Carbon::now()->addYears(config('passport.expiration_years'))); Passport::personalAccessTokensExpireIn(Carbon::now()->addYears((int)config('passport.expiration_years')));
Passport::cookie(config('passport.cookie_name')); Passport::cookie(config('passport.cookie_name'));

View file

@ -0,0 +1,555 @@
<?php namespace App\Providers;
use App\Models\Accessory;
use App\Models\Asset;
use App\Models\AssetMaintenance;
use App\Models\AssetModel;
use App\Models\Category;
use App\Models\Company;
use App\Models\Component;
use App\Models\Consumable;
use App\Models\CustomField;
use App\Models\CustomFieldset;
use App\Models\Department;
use App\Models\Depreciation;
use App\Models\Group;
use App\Models\License;
use App\Models\LicenseSeat;
use App\Models\Location;
use App\Models\Manufacturer;
use App\Models\PredefinedKit;
use App\Models\Statuslabel;
use App\Models\Supplier;
use App\Models\User;
use Illuminate\Support\ServiceProvider;
use Tabuna\Breadcrumbs\Breadcrumbs;
use Tabuna\Breadcrumbs\Trail;
class BreadcrumbsServiceProvider extends ServiceProvider
{
/**
* Handles the resource routes for first-class objects
*
* @return void
*/
public function boot()
{
// Default home
Breadcrumbs::for('home', fn (Trail $trail) =>
$trail->push('<x-icon type="home" /><span class="sr-only">'.trans('general.dashboard').'</span>', route('home'))
);
/**
* Asset Breadcrumbs
*/
if ((request()->is('hardware*')) && (request()->status!='')) {
Breadcrumbs::for('hardware.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.assets'), route('hardware.index'))
->push(request()->status.' '.trans('general.assets'), route('hardware.index', ['status' => request()->status]))
);
} else {
Breadcrumbs::for('hardware.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.assets'), route('hardware.index'))
);
}
Breadcrumbs::for('hardware.create', fn (Trail $trail) =>
$trail->parent('hardware.index', route('hardware.index'))
->push(trans('general.create'), route('hardware.create'))
);
Breadcrumbs::for('hardware.show', fn (Trail $trail, Asset $asset) =>
$trail->parent('hardware.index', route('hardware.index'))
->push($asset->present()->fullName(), route('hardware.show', $asset))
);
Breadcrumbs::for('hardware.edit', fn (Trail $trail, Asset $asset) =>
$trail->parent('hardware.index', route('hardware.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $asset->asset_tag]), route('hardware.edit', $asset))
);
/**
* Asset Model Breadcrumbs
*/
if ((request()->is('models*')) && (request()->status=='deleted')) {
Breadcrumbs::for('models.index', fn (Trail $trail) =>
$trail->parent('hardware.index', route('hardware.index'))
->push(trans('general.asset_models'), route('models.index'))
->push(trans('general.deleted_models'), route('models.index', ['status' => request()->status]))
);
} else {
Breadcrumbs::for('models.index', fn (Trail $trail) =>
$trail->parent('hardware.index', route('hardware.index'))
->push(trans('general.asset_models'), route('models.index'))
);
}
Breadcrumbs::for('models.create', fn (Trail $trail) =>
$trail->parent('models.index', route('models.index'))
->push(trans('general.create'), route('models.create'))
);
Breadcrumbs::for('models.show', fn (Trail $trail, AssetModel $model) =>
$trail->parent('models.index', route('models.index'))
->push($model->name, route('models.show', $model))
);
Breadcrumbs::for('models.edit', fn (Trail $trail, AssetModel $model) =>
$trail->parent('models.index', route('models.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $model->name]), route('models.edit', $model))
);
/**
* Accessories Breadcrumbs
*/
Breadcrumbs::for('accessories.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.accessories'), route('accessories.index'))
);
Breadcrumbs::for('accessories.create', fn (Trail $trail) =>
$trail->parent('accessories.index', route('accessories.index'))
->push(trans('general.create'), route('accessories.create'))
);
Breadcrumbs::for('accessories.show', fn (Trail $trail, Accessory $accessory) =>
$trail->parent('accessories.index', route('accessories.index'))
->push($accessory->name, route('accessories.show', $accessory))
);
Breadcrumbs::for('accessories.edit', fn (Trail $trail, Accessory $accessory) =>
$trail->parent('accessories.index', route('accessories.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $accessory->name]), route('accessories.edit', $accessory))
);
/**
* Categories Breadcrumbs
*/
Breadcrumbs::for('categories.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.categories'), route('categories.index'))
);
Breadcrumbs::for('categories.create', fn (Trail $trail) =>
$trail->parent('categories.index', route('categories.index'))
->push(trans('general.create'), route('categories.create'))
);
Breadcrumbs::for('categories.show', fn (Trail $trail, Category $category) =>
$trail->parent('categories.index', route('categories.index'))
->push($category->name, route('categories.show', $category))
);
Breadcrumbs::for('categories.edit', fn (Trail $trail, Category $category) =>
$trail->parent('categories.index', route('categories.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $category->name]), route('categories.edit', $category))
);
/**
* Company Breadcrumbs
*/
Breadcrumbs::for('companies.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.companies'), route('companies.index'))
);
Breadcrumbs::for('companies.create', fn (Trail $trail) =>
$trail->parent('companies.index', route('companies.index'))
->push(trans('general.create'), route('companies.create'))
);
Breadcrumbs::for('companies.show', fn (Trail $trail, Company $company) =>
$trail->parent('companies.index', route('companies.index'))
->push($company->name, route('companies.show', $company))
);
Breadcrumbs::for('companies.edit', fn (Trail $trail, Company $company) =>
$trail->parent('companies.index', route('companies.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $company->name]), route('companies.edit', $company))
);
/**
* Components Breadcrumbs
*/
Breadcrumbs::for('components.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.components'), route('components.index'))
);
Breadcrumbs::for('components.create', fn (Trail $trail) =>
$trail->parent('components.index', route('components.index'))
->push(trans('general.create'), route('components.create'))
);
Breadcrumbs::for('components.show', fn (Trail $trail, Component $component) =>
$trail->parent('components.index', route('components.index'))
->push($component->name, route('components.show', $component))
);
Breadcrumbs::for('components.edit', fn (Trail $trail, Component $component) =>
$trail->parent('components.index', route('components.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $component->name]), route('components.edit', $component))
);
/**
* Consumables Breadcrumbs
*/
Breadcrumbs::for('consumables.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.consumables'), route('consumables.index'))
);
Breadcrumbs::for('consumables.create', fn (Trail $trail) =>
$trail->parent('consumables.index', route('consumables.index'))
->push(trans('general.create'), route('consumables.create'))
);
Breadcrumbs::for('consumables.show', fn (Trail $trail, Consumable $consumable) =>
$trail->parent('consumables.index', route('consumables.index'))
->push($consumable->name, route('consumables.show', $consumable))
);
Breadcrumbs::for('consumables.edit', fn (Trail $trail, Consumable $consumable) =>
$trail->parent('consumables.index', route('consumables.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $consumable->name]), route('consumables.edit', $consumable))
);
/**
* Custom fields Breadcrumbs
*/
Breadcrumbs::for('fields.index', fn (Trail $trail) =>
$trail->parent('models.index', route('models.index'))
->push(trans('admin/custom_fields/general.custom_fields'), route('fields.index'))
);
Breadcrumbs::for('fields.create', fn (Trail $trail) =>
$trail->parent('fields.index', route('fields.index'))
->push(trans('general.create'), route('fields.create'))
);
Breadcrumbs::for('fields.edit', fn (Trail $trail, CustomField $field) =>
$trail->parent('fields.index', route('fields.index'))
->push($field->name, route('fields.edit', $field))
);
/**
* Custom fieldsets Breadcrumbs
*/
Breadcrumbs::for('fieldsets.create', fn (Trail $trail) =>
$trail->parent('fields.index', route('fields.index'))
->push(trans('general.create'), route('fieldsets.create'))
);
Breadcrumbs::for('fieldsets.show', fn (Trail $trail, CustomFieldset $fieldset) =>
$trail->parent('fields.index', route('fields.index'))
->push($fieldset->name, route('fields.index'))
);
Breadcrumbs::for('fieldsets.edit', fn (Trail $trail, CustomFieldset $fieldset) =>
$trail->parent('fields.index', route('fields.index'))
->push($fieldset->name, route('fieldsets.edit', $fieldset))
);
/**
* Department Breadcrumbs
*/
Breadcrumbs::for('departments.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.departments'), route('departments.index'))
);
Breadcrumbs::for('departments.create', fn (Trail $trail) =>
$trail->parent('departments.index', route('departments.index'))
->push(trans('general.create'), route('departments.create'))
);
Breadcrumbs::for('departments.show', fn (Trail $trail, Department $department) =>
$trail->parent('departments.index', route('departments.index'))
->push($department->name, route('home'))
);
Breadcrumbs::for('departments.edit', fn (Trail $trail, Department $department) =>
$trail->parent('departments.index', route('departments.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $department->name]), route('departments.edit', $department))
);
/**
* Department Breadcrumbs
*/
Breadcrumbs::for('depreciations.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.depreciations'), route('depreciations.index'))
);
Breadcrumbs::for('depreciations.create', fn (Trail $trail) =>
$trail->parent('depreciations.index', route('depreciations.index'))
->push(trans('general.create'), route('depreciations.create'))
);
Breadcrumbs::for('depreciations.show', fn (Trail $trail, Depreciation $depreciation) =>
$trail->parent('depreciations.index', route('depreciations.index'))
->push($depreciation->name, route('depreciations.show', $depreciation))
);
Breadcrumbs::for('depreciations.edit', fn (Trail $trail, Depreciation $depreciation) =>
$trail->parent('depreciations.index', route('depreciations.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $depreciation->name]), route('depreciations.edit', $depreciation))
);
/**
* Groups Breadcrumbs
*/
Breadcrumbs::for('groups.index', fn (Trail $trail) =>
$trail->parent('settings.index', route('settings.index'))
->push(trans('general.groups'), route('groups.index'))
);
Breadcrumbs::for('groups.create', fn (Trail $trail) =>
$trail->parent('groups.index', route('groups.index'))
->push(trans('general.create'), route('groups.create'))
);
Breadcrumbs::for('groups.show', fn (Trail $trail, Group $group) =>
$trail->parent('groups.index', route('groups.index'))
->push($group->name, route('groups.show', $group))
);
Breadcrumbs::for('groups.edit', fn (Trail $trail, Group $group) =>
$trail->parent('groups.index', route('groups.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $group->name]), route('groups.edit', $group))
);
/**
* Licenses Breadcrumbs
*/
Breadcrumbs::for('licenses.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.licenses'), route('licenses.index'))
);
Breadcrumbs::for('licenses.create', fn (Trail $trail) =>
$trail->parent('licenses.index', route('licenses.index'))
->push(trans('general.create'), route('licenses.create'))
);
Breadcrumbs::for('licenses.show', fn (Trail $trail, License $license) =>
$trail->parent('licenses.index', route('licenses.index'))
->push($license->name, route('licenses.show', $license))
);
Breadcrumbs::for('licenses.edit', fn (Trail $trail, License $license) =>
$trail->parent('licenses.index', route('licenses.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $license->name]), route('licenses.edit', $license))
);
/**
* Locations Breadcrumbs
*/
Breadcrumbs::for('locations.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.locations'), route('locations.index'))
);
Breadcrumbs::for('locations.create', fn (Trail $trail) =>
$trail->parent('locations.index', route('locations.index'))
->push(trans('general.create'), route('locations.create'))
);
Breadcrumbs::for('locations.show', fn (Trail $trail, Location $location) =>
$trail->parent('locations.index', route('locations.index'))
->push($location->name, route('locations.show', $location))
);
Breadcrumbs::for('locations.edit', fn (Trail $trail, Location $location) =>
$trail->parent('locations.index', route('locations.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $location->name]), route('locations.edit', $location))
);
/**
* Maintenances Breadcrumbs
*/
Breadcrumbs::for('maintenances.index', fn (Trail $trail) =>
$trail->parent('hardware.index', route('hardware.index'))
->push(trans('general.maintenances'), route('maintenances.index'))
);
Breadcrumbs::for('maintenances.create', fn (Trail $trail) =>
$trail->parent('maintenances.index', route('maintenances.index'))
->push(trans('general.create'), route('maintenances.create'))
);
Breadcrumbs::for('maintenances.show', fn (Trail $trail, AssetMaintenance $maintenance) =>
$trail->parent('maintenances.index', route('locations.index'))
->push($maintenance->title, route('maintenances.show', $maintenance))
);
Breadcrumbs::for('manufacturers.edit', fn (Trail $trail, Manufacturer $manufacturer) =>
$trail->parent('manufacturers.index', route('manufacturers.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $manufacturer->name]), route('manufacturers.edit', $manufacturer))
);
/**
* Manufacturers Breadcrumbs
*/
Breadcrumbs::for('manufacturers.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.manufacturers'), route('manufacturers.index'))
);
Breadcrumbs::for('manufacturers.create', fn (Trail $trail) =>
$trail->parent('manufacturers.index', route('manufacturers.index'))
->push(trans('general.create'), route('manufacturers.create'))
);
Breadcrumbs::for('manufacturers.show', fn (Trail $trail, Manufacturer $manufacturer) =>
$trail->parent('manufacturers.index', route('manufacturers.index'))
->push($manufacturer->name, route('home'))
);
Breadcrumbs::for('manufacturers.edit', fn (Trail $trail, Manufacturer $manufacturer) =>
$trail->parent('manufacturers.index', route('manufacturers.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $manufacturer->name]), route('manufacturers.edit', $manufacturer))
);
/**
* Predefined Kits Breadcrumbs
*/
Breadcrumbs::for('kits.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.kits'), route('kits.index'))
);
Breadcrumbs::for('kits.create', fn (Trail $trail) =>
$trail->parent('kits.index', route('kits.index'))
->push(trans('general.create'), route('kits.create'))
);
Breadcrumbs::for('kits.show', fn (Trail $trail, PredefinedKit $kit) =>
$trail->parent('kits.index', route('kits.index'))
->push($kit->name, route('kits.show', $kit))
);
Breadcrumbs::for('kits.edit', fn (Trail $trail, PredefinedKit $kit) =>
$trail->parent('kits.index', route('kits.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $kit->name]), route('kits.edit', $kit))
);
/**
* Status Labels Breadcrumbs
*/
Breadcrumbs::for('statuslabels.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.status_labels'), route('statuslabels.index'))
);
Breadcrumbs::for('statuslabels.create', fn (Trail $trail) =>
$trail->parent('statuslabels.index', route('statuslabels.index'))
->push(trans('general.create'), route('statuslabels.create'))
);
Breadcrumbs::for('statuslabels.show', fn (Trail $trail, Statuslabel $statuslabel) =>
$trail->parent('statuslabels.index', route('statuslabels.index'))
->push($statuslabel->name, route('statuslabels.show', $statuslabel))
);
Breadcrumbs::for('statuslabels.edit', fn (Trail $trail, Statuslabel $statuslabel) =>
$trail->parent('statuslabels.index', route('statuslabels.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $statuslabel->name]), route('statuslabels.edit', $statuslabel))
);
/**
* Settings Breadcrumbs
*/
Breadcrumbs::for('settings.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.admin'), route('settings.index'))
);
/**
* Suppliers Breadcrumbs
*/
Breadcrumbs::for('suppliers.index', fn (Trail $trail) =>
$trail->parent('home', route('home'))
->push(trans('general.suppliers'), route('suppliers.index'))
);
Breadcrumbs::for('suppliers.create', fn (Trail $trail) =>
$trail->parent('suppliers.index', route('suppliers.index'))
->push(trans('general.create'), route('suppliers.create'))
);
Breadcrumbs::for('suppliers.show', fn (Trail $trail, Supplier $supplier) =>
$trail->parent('suppliers.index', route('suppliers.index'))
->push($supplier->name, route('home'))
);
Breadcrumbs::for('suppliers.edit', fn (Trail $trail, Supplier $supplier) =>
$trail->parent('suppliers.index', route('suppliers.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $supplier->name]), route('suppliers.edit', $supplier))
);
/**
* Users Breadcrumbs
*/
if ((request()->is('users*')) && (request()->status=='deleted')) {
Breadcrumbs::for('users.index', fn(Trail $trail) => $trail->parent('home', route('home'))
->push(trans('general.users'), route('users.index'))
->push(trans('general.deleted_users'), route('users.index'))
);
} else {
Breadcrumbs::for('users.index', fn(Trail $trail) => $trail->parent('home', route('home'))
->push(trans('general.users'), route('users.index'))
);
}
Breadcrumbs::for('users.create', fn (Trail $trail) =>
$trail->parent('users.index', route('users.index'))
->push(trans('general.create'), route('users.create'))
);
Breadcrumbs::for('users.show', fn (Trail $trail, User $user) =>
$trail->parent('users.index', route('users.index'))
->push($user->username, route('users.show', $user))
);
Breadcrumbs::for('users.edit', fn (Trail $trail, User $user) =>
$trail->parent('users.index', route('users.index'))
->push(trans('general.breadcrumb_button_actions.edit_item', ['name' => $user->name]), route('users.edit', $user))
);
}
}

View file

@ -49,10 +49,6 @@ class SamlServiceProvider extends ServiceProvider
'uses' => 'Auth\SamlController@login', ] 'uses' => 'Auth\SamlController@login', ]
); );
Route::group(['prefix' => 'admin', 'middleware' => ['web', 'auth', 'authorize:superuser']], function () {
Route::get('saml', ['as' => 'settings.saml.index', 'uses' => 'SettingsController@getSamlSettings']);
Route::post('saml', ['as' => 'settings.saml.save', 'uses' => 'SettingsController@postSamlSettings']);
});
}); });
} }

View file

@ -209,7 +209,7 @@ class Saml
} }
} }
$custom_settings = preg_split('/\r\n|\r|\n/', $setting->saml_custom_settings); $custom_settings = preg_split('/\r\n|\r|\n/', $setting->saml_custom_settings ?? '');
if ($custom_settings) { if ($custom_settings) {
foreach ($custom_settings as $custom_setting) { foreach ($custom_settings as $custom_setting) {
$split = explode('=', $custom_setting, 2); $split = explode('=', $custom_setting, 2);

View file

@ -26,17 +26,19 @@ class SnipeTranslator extends Translator {
public function choice($key, $number, array $replace = [], $locale = null) public function choice($key, $number, array $replace = [], $locale = null)
{ {
$line = $this->get( $line = $this->get(
$key, $replace, $locale = $this->localeForChoice($locale) $key, [], $locale = $this->localeForChoice($key, $locale)
); );
// If the given "number" is actually an array or countable we will simply count the // If the given "number" is actually an array or countable we will simply count the
// number of elements in an instance. This allows developers to pass an array of // number of elements in an instance. This allows developers to pass an array of
// items without having to count it on their end first which gives bad syntax. // items without having to count it on their end first which gives bad syntax.
if (is_array($number) || $number instanceof Countable) { if (is_countable($number)) {
$number = count($number); $number = count($number);
} }
$replace['count'] = $number; if (!isset($replace['count'])) {
$replace['count'] = $number;
}
$underscored_locale = str_replace("-","_",$locale); // OUR CHANGE. $underscored_locale = str_replace("-","_",$locale); // OUR CHANGE.
return $this->makeReplacements( // BELOW - that $underscored_locale is the *ONLY* modified part return $this->makeReplacements( // BELOW - that $underscored_locale is the *ONLY* modified part

View file

@ -7,6 +7,7 @@ use App\Models\Labels\Label as LabelModel;
use App\Models\Labels\Sheet; use App\Models\Labels\Sheet;
use Illuminate\Contracts\View\View; use Illuminate\Contracts\View\View;
use Illuminate\Support\Collection; use Illuminate\Support\Collection;
use Illuminate\Support\Facades\Log;
use Illuminate\Support\Facades\Storage; use Illuminate\Support\Facades\Storage;
use Illuminate\Support\Traits\Macroable; use Illuminate\Support\Traits\Macroable;
use TCPDF; use TCPDF;
@ -38,7 +39,7 @@ class Label implements View
$settings = $this->data->get('settings'); $settings = $this->data->get('settings');
$assets = $this->data->get('assets'); $assets = $this->data->get('assets');
$offset = $this->data->get('offset'); $offset = $this->data->get('offset');
$template = LabelModel::find($settings->label2_template);
// If disabled, pass to legacy view // If disabled, pass to legacy view
if ((!$settings->label2_enable)) { if ((!$settings->label2_enable)) {
@ -49,6 +50,12 @@ class Label implements View
->with('count', $this->data->get('count')); ->with('count', $this->data->get('count'));
} }
$template = LabelModel::find($settings->label2_template);
if ($template === null) {
return redirect()->route('settings.labels.index')->with('error', trans('admin/settings/message.labels.null_template'));
}
$template->validate(); $template->validate();
$pdf = new TCPDF( $pdf = new TCPDF(
@ -116,12 +123,9 @@ class Label implements View
} }
} }
if ($template->getSupport2DBarcode()) { if ($template->getSupport2DBarcode()) {
$barcode2DType = $settings->label2_2d_type; $barcode2DType = $settings->label2_2d_type;
$barcode2DType = ($barcode2DType == 'default') ? if (($barcode2DType != 'none') && (!is_null($barcode2DType))) {
$settings->barcode_type :
$barcode2DType;
if (($barcode2DType != 'none') && (!is_null($barcode2DType))) {
switch ($settings->label2_2d_target) { switch ($settings->label2_2d_target) {
case 'ht_tag': case 'ht_tag':
$barcode2DTarget = route('ht/assetTag', $asset->asset_tag); $barcode2DTarget = route('ht/assetTag', $asset->asset_tag);

View file

@ -14,10 +14,14 @@
{ {
"type": "vcs", "type": "vcs",
"url": "https://github.com/grokability/laravel-scim-server" "url": "https://github.com/grokability/laravel-scim-server"
},
{
"type": "vcs",
"url": "https://github.com/grokability/html"
} }
], ],
"require": { "require": {
"php": "^8.1", "php": "^8.2",
"ext-curl": "*", "ext-curl": "*",
"ext-fileinfo": "*", "ext-fileinfo": "*",
"ext-iconv": "*", "ext-iconv": "*",
@ -25,7 +29,7 @@
"ext-mbstring": "*", "ext-mbstring": "*",
"ext-pdo": "*", "ext-pdo": "*",
"alek13/slack": "^2.0", "alek13/slack": "^2.0",
"arietimmerman/laravel-scim-server": "dev-master", "arietimmerman/laravel-scim-server": "dev-laravel_11_compatibility",
"bacon/bacon-qr-code": "^2.0", "bacon/bacon-qr-code": "^2.0",
"barryvdh/laravel-debugbar": "^3.13", "barryvdh/laravel-debugbar": "^3.13",
"barryvdh/laravel-dompdf": "^2.0", "barryvdh/laravel-dompdf": "^2.0",
@ -40,20 +44,20 @@
"javiereguiluz/easyslugger": "^1.0", "javiereguiluz/easyslugger": "^1.0",
"laravel-notification-channels/google-chat": "^3.0", "laravel-notification-channels/google-chat": "^3.0",
"laravel-notification-channels/microsoft-teams": "^1.2", "laravel-notification-channels/microsoft-teams": "^1.2",
"laravel/framework": "^10.0", "laravel/framework": "^11.0",
"laravel/helpers": "^1.4", "laravel/helpers": "^1.4",
"laravel/passport": "^11.0", "laravel/passport": "^12.0",
"laravel/slack-notification-channel": "^2.3", "laravel/slack-notification-channel": "^3.4",
"laravel/socialite": "^5.6", "laravel/socialite": "^5.6",
"laravel/tinker": "^2.6", "laravel/tinker": "^2.6",
"laravel/ui": "^4.0", "laravel/ui": "^4.0",
"laravelcollective/html": "^6.2", "laravelcollective/html": "6.x-dev",
"league/csv": "^9.7", "league/csv": "^9.7",
"league/flysystem-aws-s3-v3": "^3.0", "league/flysystem-aws-s3-v3": "^3.0",
"livewire/livewire": "^3.5", "livewire/livewire": "^3.5",
"neitanod/forceutf8": "^2.0", "neitanod/forceutf8": "^2.0",
"nesbot/carbon": "^2.32", "nesbot/carbon": "^3.0",
"nunomaduro/collision": "^7.0", "nunomaduro/collision": "^8.1",
"okvpn/clock-lts": "^1.0", "okvpn/clock-lts": "^1.0",
"onelogin/php-saml": "^3.4", "onelogin/php-saml": "^3.4",
"onnov/detect-encoding": "^2.0", "onnov/detect-encoding": "^2.0",
@ -66,6 +70,7 @@
"rollbar/rollbar-laravel": "^8.0", "rollbar/rollbar-laravel": "^8.0",
"spatie/laravel-backup": "^8.8", "spatie/laravel-backup": "^8.8",
"spatie/laravel-ignition": "^2.0", "spatie/laravel-ignition": "^2.0",
"tabuna/breadcrumbs": "^4.2",
"tecnickcom/tc-lib-barcode": "^1.15", "tecnickcom/tc-lib-barcode": "^1.15",
"tecnickcom/tcpdf": "^6.5", "tecnickcom/tcpdf": "^6.5",
"unicodeveloper/laravel-password": "^1.0", "unicodeveloper/laravel-password": "^1.0",
@ -85,8 +90,7 @@
"phpunit/phpunit": "^10.0", "phpunit/phpunit": "^10.0",
"squizlabs/php_codesniffer": "^3.5", "squizlabs/php_codesniffer": "^3.5",
"symfony/css-selector": "^4.4", "symfony/css-selector": "^4.4",
"symfony/dom-crawler": "^4.4", "symfony/dom-crawler": "^4.4"
"vimeo/psalm": "^5.13"
}, },
"extra": { "extra": {
"laravel": { "laravel": {

2652
composer.lock generated

File diff suppressed because it is too large Load diff

Some files were not shown because too many files have changed in this diff Show more