commitd55c176199Merge:93ff9524dc0451fe17Author: snipe <snipe@snipe.net> Date: Wed Mar 16 18:35:42 2022 +0000 Merge pull request #10831 from snipe/merges/master_down_to_develop_march_16 v6.0.0-RC-5 - Merges master down to develop commitc0451fe17aAuthor: snipe <snipe@snipe.net> Date: Wed Mar 16 18:10:17 2022 +0000 Bumped version to v6.0.0-RC-5 Signed-off-by: snipe <snipe@snipe.net> commit0f95802699Author: snipe <snipe@snipe.net> Date: Wed Mar 16 18:05:47 2022 +0000 Fixed mis-automerge Signed-off-by: snipe <snipe@snipe.net> commit9db8bd782dAuthor: snipe <snipe@snipe.net> Date: Wed Mar 16 18:02:07 2022 +0000 Merging master down into develop Signed-off-by: snipe <snipe@snipe.net> commit93ff9524d6Merge:40a94707789ddbddadAuthor: snipe <snipe@snipe.net> Date: Wed Mar 16 10:04:58 2022 -0700 Merge pull request #10829 from snipe/features/add_statuslabel_filter_by_type_in_api Added filter by status_type in StatusLabels API index endpoint commit89ddbddadaAuthor: snipe <snipe@snipe.net> Date: Wed Mar 16 16:53:18 2022 +0000 Fixed comment Signed-off-by: snipe <snipe@snipe.net> commit7498fe36e9Author: snipe <snipe@snipe.net> Date: Wed Mar 16 16:45:03 2022 +0000 Removed extra space because pedantry Signed-off-by: snipe <snipe@snipe.net> commitbabf7c064bAuthor: snipe <snipe@snipe.net> Date: Wed Mar 16 16:38:45 2022 +0000 Added ability to filter status label index endpoint by status type Signed-off-by: snipe <snipe@snipe.net> commit40a9470770Merge:f499dcf7c781b42601Author: snipe <snipe@snipe.net> Date: Wed Mar 16 08:54:10 2022 -0700 Merge pull request #10828 from snipe/fixes/10826_parse_error_on_print_all Fixed #10826 - parse error on translation string for print all assets commit781b426018Author: snipe <snipe@snipe.net> Date: Wed Mar 16 08:50:47 2022 -0700 Fixed #10826 - parse error on translation string for print all assets Signed-off-by: snipe <snipe@snipe.net> commitf499dcf7c3Merge:84aa26dd554d6a4d97Author: snipe <snipe@snipe.net> Date: Wed Mar 16 08:25:29 2022 -0700 Merge pull request #10827 from snipe/fixes/modal_dropdowns_broken Fixed #10825 - selectlists in modals b0rked commit54d6a4d978Author: snipe <snipe@snipe.net> Date: Wed Mar 16 08:23:50 2022 -0700 Fixed #10825 - selectlists in modals b0rked Signed-off-by: snipe <snipe@snipe.net> commit84aa26dd50Merge:8984b3a5914bd07e97Author: snipe <snipe@snipe.net> Date: Thu Mar 10 13:02:45 2022 -0800 Merge pull request #10728 from Godmartinz/gh10191-css-dropdownmenu Fixed #10191 - font color contrast on mobile menu commit8984b3a59dMerge:d06ef4bdecdc0805fcAuthor: snipe <snipe@snipe.net> Date: Thu Mar 10 10:40:33 2022 -0800 Merge pull request #10812 from inietov/fixes/CarbonExceptions_InvalidFormatException_DateTime_develop Fixes Carbon\Exceptions\InvalidFormatException: DateTime::__construct(): Failed to parse time string develop commitcdc0805fc4Author: Ivan Nieto Vivanco <inietov@gmail.com> Date: Thu Mar 10 12:07:07 2022 -0600 Sanitize dates input in the importer before saving commit14bd07e97dMerge:4435ea95f16fd10954Author: Godfrey M <godmartinz@gmail.com> Date: Thu Mar 3 10:14:30 2022 -0800 gerge branchevelop' into gh10191-css-dropdownmenu commit4435ea95fdAuthor: Godfrey M <godmartinz@gmail.com> Date: Thu Mar 3 09:47:11 2022 -0800 removes changes to mix-manifest commitf115385243Author: Godfrey M <godmartinz@gmail.com> Date: Thu Mar 3 09:32:05 2022 -0800 C&P mix-manifest to be tracked commit708dc08b4fAuthor: Godfrey M <godmartinz@gmail.com> Date: Wed Feb 23 11:17:44 2022 -0800 mixes color contrast on dropdown menu from navbar on file Signed-off-by: snipe <snipe@snipe.net>
881 lines
35 KiB
PHP
Executable File
881 lines
35 KiB
PHP
Executable File
<?php
|
|
|
|
namespace App\Http\Controllers\Assets;
|
|
|
|
use App\Helpers\Helper;
|
|
use App\Http\Controllers\Controller;
|
|
use App\Http\Requests\ImageUploadRequest;
|
|
use App\Models\Actionlog;
|
|
use App\Models\Asset;
|
|
use App\Models\AssetModel;
|
|
use App\Models\CheckoutRequest;
|
|
use App\Models\Company;
|
|
use App\Models\Location;
|
|
use App\Models\Setting;
|
|
use App\Models\User;
|
|
use Auth;
|
|
use Carbon\Carbon;
|
|
use DB;
|
|
use Gate;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Support\Facades\Cache;
|
|
use Illuminate\Support\Facades\Storage;
|
|
use Input;
|
|
use Intervention\Image\Facades\Image;
|
|
use League\Csv\Reader;
|
|
use League\Csv\Statement;
|
|
use Paginator;
|
|
use Redirect;
|
|
use Response;
|
|
use Slack;
|
|
use Str;
|
|
use TCPDF;
|
|
use View;
|
|
|
|
/**
|
|
* This class controls all actions related to assets for
|
|
* the Snipe-IT Asset Management application.
|
|
*
|
|
* @version v1.0
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
*/
|
|
class AssetsController extends Controller
|
|
{
|
|
protected $qrCodeDimensions = ['height' => 3.5, 'width' => 3.5];
|
|
protected $barCodeDimensions = ['height' => 2, 'width' => 22];
|
|
|
|
public function __construct()
|
|
{
|
|
$this->middleware('auth');
|
|
parent::__construct();
|
|
}
|
|
|
|
/**
|
|
* Returns a view that invokes the ajax tables which actually contains
|
|
* the content for the assets listing, which is generated in getDatatable.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see AssetController::getDatatable() method that generates the JSON response
|
|
* @since [v1.0]
|
|
* @param Request $request
|
|
* @return View
|
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
|
*/
|
|
public function index(Request $request)
|
|
{
|
|
$this->authorize('index', Asset::class);
|
|
$company = Company::find($request->input('company_id'));
|
|
|
|
return view('hardware/index')->with('company', $company);
|
|
}
|
|
|
|
/**
|
|
* Returns a view that presents a form to create a new asset.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v1.0]
|
|
* @param Request $request
|
|
* @return View
|
|
* @internal param int $model_id
|
|
*/
|
|
public function create(Request $request)
|
|
{
|
|
$this->authorize('create', Asset::class);
|
|
$view = View::make('hardware/edit')
|
|
->with('statuslabel_list', Helper::statusLabelList())
|
|
->with('item', new Asset)
|
|
->with('statuslabel_types', Helper::statusTypeList());
|
|
|
|
if ($request->filled('model_id')) {
|
|
$selected_model = AssetModel::find($request->input('model_id'));
|
|
$view->with('selected_model', $selected_model);
|
|
}
|
|
|
|
return $view;
|
|
}
|
|
|
|
/**
|
|
* Validate and process new asset form data.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v1.0]
|
|
* @return Redirect
|
|
*/
|
|
public function store(ImageUploadRequest $request)
|
|
{
|
|
$this->authorize(Asset::class);
|
|
|
|
// Handle asset tags - there could be one, or potentially many.
|
|
// This is only necessary on create, not update, since bulk editing is handled
|
|
// differently
|
|
$asset_tags = $request->input('asset_tags');
|
|
|
|
$settings = Setting::getSettings();
|
|
|
|
$success = false;
|
|
$serials = $request->input('serials');
|
|
|
|
for ($a = 1; $a <= count($asset_tags); $a++) {
|
|
$asset = new Asset();
|
|
$asset->model()->associate(AssetModel::find($request->input('model_id')));
|
|
$asset->name = $request->input('name');
|
|
|
|
// Check for a corresponding serial
|
|
if (($serials) && (array_key_exists($a, $serials))) {
|
|
$asset->serial = $serials[$a];
|
|
}
|
|
|
|
if (($asset_tags) && (array_key_exists($a, $asset_tags))) {
|
|
$asset->asset_tag = $asset_tags[$a];
|
|
}
|
|
|
|
$asset->company_id = Company::getIdForCurrentUser($request->input('company_id'));
|
|
$asset->model_id = $request->input('model_id');
|
|
$asset->order_number = $request->input('order_number');
|
|
$asset->notes = $request->input('notes');
|
|
$asset->user_id = Auth::id();
|
|
$asset->archived = '0';
|
|
$asset->physical = '1';
|
|
$asset->depreciate = '0';
|
|
$asset->status_id = request('status_id', 0);
|
|
$asset->warranty_months = request('warranty_months', null);
|
|
$asset->purchase_cost = Helper::ParseCurrency($request->get('purchase_cost'));
|
|
$asset->purchase_date = request('purchase_date', null);
|
|
$asset->assigned_to = request('assigned_to', null);
|
|
$asset->supplier_id = request('supplier_id', 0);
|
|
$asset->requestable = request('requestable', 0);
|
|
$asset->rtd_location_id = request('rtd_location_id', null);
|
|
|
|
if (! empty($settings->audit_interval)) {
|
|
$asset->next_audit_date = Carbon::now()->addMonths($settings->audit_interval)->toDateString();
|
|
}
|
|
|
|
if ($asset->assigned_to == '') {
|
|
$asset->location_id = $request->input('rtd_location_id', null);
|
|
}
|
|
|
|
// Create the image (if one was chosen.)
|
|
if ($request->has('image')) {
|
|
$asset = $request->handleImages($asset);
|
|
}
|
|
|
|
// Update custom fields in the database.
|
|
// Validation for these fields is handled through the AssetRequest form request
|
|
$model = AssetModel::find($request->get('model_id'));
|
|
|
|
if (($model) && ($model->fieldset)) {
|
|
foreach ($model->fieldset->fields as $field) {
|
|
if ($field->field_encrypted == '1') {
|
|
if (Gate::allows('admin')) {
|
|
if (is_array($request->input($field->convertUnicodeDbSlug()))) {
|
|
$asset->{$field->convertUnicodeDbSlug()} = \Crypt::encrypt(e(implode(', ', $request->input($field->convertUnicodeDbSlug()))));
|
|
} else {
|
|
$asset->{$field->convertUnicodeDbSlug()} = \Crypt::encrypt(e($request->input($field->convertUnicodeDbSlug())));
|
|
}
|
|
}
|
|
} else {
|
|
if (is_array($request->input($field->convertUnicodeDbSlug()))) {
|
|
$asset->{$field->convertUnicodeDbSlug()} = implode(', ', $request->input($field->convertUnicodeDbSlug()));
|
|
} else {
|
|
$asset->{$field->convertUnicodeDbSlug()} = $request->input($field->convertUnicodeDbSlug());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Validate the asset before saving
|
|
if ($asset->isValid() && $asset->save()) {
|
|
if (request('assigned_user')) {
|
|
$target = User::find(request('assigned_user'));
|
|
$location = $target->location_id;
|
|
} elseif (request('assigned_asset')) {
|
|
$target = Asset::find(request('assigned_asset'));
|
|
$location = $target->location_id;
|
|
} elseif (request('assigned_location')) {
|
|
$target = Location::find(request('assigned_location'));
|
|
$location = $target->id;
|
|
}
|
|
|
|
if (isset($target)) {
|
|
$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);
|
|
}
|
|
|
|
$success = true;
|
|
}
|
|
}
|
|
|
|
if ($success) {
|
|
// Redirect to the asset listing page
|
|
return redirect()->route('hardware.index')
|
|
->with('success', trans('admin/hardware/message.create.success'));
|
|
}
|
|
|
|
return redirect()->back()->withInput()->withErrors($asset->getErrors());
|
|
}
|
|
|
|
/**
|
|
* Returns a view that presents a form to edit an existing asset.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function edit($assetId = null)
|
|
{
|
|
if (! $item = 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'));
|
|
}
|
|
//Handles company checks and permissions.
|
|
$this->authorize($item);
|
|
|
|
return view('hardware/edit', compact('item'))
|
|
->with('statuslabel_list', Helper::statusLabelList())
|
|
->with('statuslabel_types', Helper::statusTypeList());
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a view that presents information about an asset for detail view.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function show($assetId = null)
|
|
{
|
|
$asset = Asset::withTrashed()->find($assetId);
|
|
$this->authorize('view', $asset);
|
|
$settings = Setting::getSettings();
|
|
|
|
if (isset($asset)) {
|
|
$audit_log = Actionlog::where('action_type', '=', 'audit')
|
|
->where('item_id', '=', $assetId)
|
|
->where('item_type', '=', Asset::class)
|
|
->orderBy('created_at', 'DESC')->first();
|
|
|
|
if ($asset->location) {
|
|
$use_currency = $asset->location->currency;
|
|
} else {
|
|
if ($settings->default_currency != '') {
|
|
$use_currency = $settings->default_currency;
|
|
} else {
|
|
$use_currency = trans('general.currency');
|
|
}
|
|
}
|
|
|
|
$qr_code = (object) [
|
|
'display' => $settings->qr_code == '1',
|
|
'url' => route('qr_code/hardware', $asset->id),
|
|
];
|
|
|
|
return view('hardware/view', compact('asset', 'qr_code', 'settings'))
|
|
->with('use_currency', $use_currency)->with('audit_log', $audit_log);
|
|
}
|
|
|
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
}
|
|
|
|
/**
|
|
* Validate and process asset edit form.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return Redirect
|
|
*/
|
|
public function update(ImageUploadRequest $request, $assetId = null)
|
|
{
|
|
// 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);
|
|
|
|
$asset->status_id = $request->input('status_id', null);
|
|
$asset->warranty_months = $request->input('warranty_months', null);
|
|
$asset->purchase_cost = Helper::ParseCurrency($request->input('purchase_cost', null));
|
|
$asset->purchase_date = $request->input('purchase_date', null);
|
|
$asset->supplier_id = $request->input('supplier_id', null);
|
|
$asset->expected_checkin = $request->input('expected_checkin', null);
|
|
|
|
// If the box isn't checked, it's not in the request at all.
|
|
$asset->requestable = $request->filled('requestable');
|
|
$asset->rtd_location_id = $request->input('rtd_location_id', null);
|
|
|
|
if ($asset->assigned_to == '') {
|
|
$asset->location_id = $request->input('rtd_location_id', null);
|
|
}
|
|
|
|
|
|
if ($request->filled('image_delete')) {
|
|
try {
|
|
unlink(public_path().'/uploads/assets/'.$asset->image);
|
|
$asset->image = '';
|
|
} catch (\Exception $e) {
|
|
\Log::info($e);
|
|
}
|
|
}
|
|
|
|
// Update the asset data
|
|
$asset_tag = $request->input('asset_tags');
|
|
$serial = $request->input('serials');
|
|
$asset->name = $request->input('name');
|
|
$asset->serial = $serial[1];
|
|
$asset->company_id = Company::getIdForCurrentUser($request->input('company_id'));
|
|
$asset->model_id = $request->input('model_id');
|
|
$asset->order_number = $request->input('order_number');
|
|
$asset->asset_tag = $asset_tag[1];
|
|
$asset->notes = $request->input('notes');
|
|
$asset->physical = '1';
|
|
|
|
$asset = $request->handleImages($asset);
|
|
|
|
// Update custom fields in the database.
|
|
// Validation for these fields is handlded through the AssetRequest form request
|
|
// FIXME: No idea why this is returning a Builder error on db_column_name.
|
|
// Need to investigate and fix. Using static method for now.
|
|
$model = AssetModel::find($request->get('model_id'));
|
|
if (($model) && ($model->fieldset)) {
|
|
foreach ($model->fieldset->fields as $field) {
|
|
if ($field->field_encrypted == '1') {
|
|
if (Gate::allows('admin')) {
|
|
if (is_array($request->input($field->convertUnicodeDbSlug()))) {
|
|
$asset->{$field->convertUnicodeDbSlug()} = \Crypt::encrypt(e(implode(', ', $request->input($field->convertUnicodeDbSlug()))));
|
|
} else {
|
|
$asset->{$field->convertUnicodeDbSlug()} = \Crypt::encrypt(e($request->input($field->convertUnicodeDbSlug())));
|
|
}
|
|
}
|
|
} else {
|
|
if (is_array($request->input($field->convertUnicodeDbSlug()))) {
|
|
$asset->{$field->convertUnicodeDbSlug()} = implode(', ', $request->input($field->convertUnicodeDbSlug()));
|
|
} else {
|
|
$asset->{$field->convertUnicodeDbSlug()} = $request->input($field->convertUnicodeDbSlug());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if ($asset->save()) {
|
|
return redirect()->route('hardware.show', $assetId)
|
|
->with('success', trans('admin/hardware/message.update.success'));
|
|
}
|
|
|
|
return redirect()->back()->withInput()->withErrors($asset->getErrors());
|
|
}
|
|
|
|
/**
|
|
* Delete a given asset (mark as deleted).
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return Redirect
|
|
*/
|
|
public function destroy($assetId)
|
|
{
|
|
// 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('delete', $asset);
|
|
|
|
DB::table('assets')
|
|
->where('id', $asset->id)
|
|
->update(['assigned_to' => null]);
|
|
|
|
if ($asset->image) {
|
|
try {
|
|
Storage::disk('public')->delete('assets'.'/'.$asset->image);
|
|
} catch (\Exception $e) {
|
|
\Log::debug($e);
|
|
}
|
|
}
|
|
|
|
$asset->delete();
|
|
|
|
return redirect()->route('hardware.index')->with('success', trans('admin/hardware/message.delete.success'));
|
|
}
|
|
|
|
/**
|
|
* Searches the assets table by serial, and redirects if it finds one
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v3.0]
|
|
* @return Redirect
|
|
*/
|
|
public function getAssetBySerial(Request $request)
|
|
{
|
|
$topsearch = ($request->get('topsearch')=="true");
|
|
|
|
if (!$asset = Asset::where('serial', '=', $request->get('serial'))->first()) {
|
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
}
|
|
$this->authorize('view', $asset);
|
|
return redirect()->route('hardware.show', $asset->id)->with('topsearch', $topsearch);
|
|
}
|
|
|
|
/**
|
|
* Searches the assets table by asset tag, and redirects if it finds one
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v3.0]
|
|
* @return Redirect
|
|
*/
|
|
public function getAssetByTag(Request $request)
|
|
{
|
|
$topsearch = ($request->get('topsearch') == 'true');
|
|
|
|
if (! $asset = Asset::where('asset_tag', '=', $request->get('assetTag'))->first()) {
|
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
}
|
|
$this->authorize('view', $asset);
|
|
|
|
return redirect()->route('hardware.show', $asset->id)->with('topsearch', $topsearch);
|
|
}
|
|
|
|
|
|
/**
|
|
* Return a QR code for the asset
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return Response
|
|
*/
|
|
public function getQrCode($assetId = null)
|
|
{
|
|
$settings = Setting::getSettings();
|
|
|
|
if ($settings->qr_code == '1') {
|
|
$asset = Asset::withTrashed()->find($assetId);
|
|
if ($asset) {
|
|
$size = Helper::barcodeDimensions($settings->barcode_type);
|
|
$qr_file = public_path().'/uploads/barcodes/qr-'.str_slug($asset->asset_tag).'-'.str_slug($asset->id).'.png';
|
|
|
|
if (isset($asset->id, $asset->asset_tag)) {
|
|
if (file_exists($qr_file)) {
|
|
$header = ['Content-type' => 'image/png'];
|
|
|
|
return response()->file($qr_file, $header);
|
|
} else {
|
|
$barcode = new \Com\Tecnick\Barcode\Barcode();
|
|
$barcode_obj = $barcode->getBarcodeObj($settings->barcode_type, route('hardware.show', $asset->id), $size['height'], $size['width'], 'black', [-2, -2, -2, -2]);
|
|
file_put_contents($qr_file, $barcode_obj->getPngData());
|
|
|
|
return response($barcode_obj->getPngData())->header('Content-type', 'image/png');
|
|
}
|
|
}
|
|
}
|
|
|
|
return 'That asset is invalid';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return a 2D barcode for the asset
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return Response
|
|
*/
|
|
public function getBarCode($assetId = null)
|
|
{
|
|
$settings = Setting::getSettings();
|
|
$asset = Asset::find($assetId);
|
|
$barcode_file = public_path().'/uploads/barcodes/'.str_slug($settings->alt_barcode).'-'.str_slug($asset->asset_tag).'.png';
|
|
|
|
if (isset($asset->id, $asset->asset_tag)) {
|
|
if (file_exists($barcode_file)) {
|
|
$header = ['Content-type' => 'image/png'];
|
|
|
|
return response()->file($barcode_file, $header);
|
|
} else {
|
|
// Calculate barcode width in pixel based on label width (inch)
|
|
$barcode_width = ($settings->labels_width - $settings->labels_display_sgutter) * 200.000000000001;
|
|
|
|
$barcode = new \Com\Tecnick\Barcode\Barcode();
|
|
try {
|
|
$barcode_obj = $barcode->getBarcodeObj($settings->alt_barcode, $asset->asset_tag, ($barcode_width < 300 ? $barcode_width : 300), 50);
|
|
file_put_contents($barcode_file, $barcode_obj->getPngData());
|
|
|
|
return response($barcode_obj->getPngData())->header('Content-type', 'image/png');
|
|
} catch (\Exception $e) {
|
|
\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 a label for an individual asset.
|
|
*
|
|
* @author [L. Swartzendruber] [<logan.swartzendruber@gmail.com>
|
|
* @param int $assetId
|
|
* @return View
|
|
*/
|
|
public function getLabel($assetId = null)
|
|
{
|
|
if (isset($assetId)) {
|
|
$asset = Asset::find($assetId);
|
|
$this->authorize('view', $asset);
|
|
|
|
return view('hardware/labels')
|
|
->with('assets', Asset::find($asset))
|
|
->with('settings', Setting::getSettings())
|
|
->with('bulkedit', false)
|
|
->with('count', 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns a view that presents a form to clone an asset.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getClone($assetId = null)
|
|
{
|
|
// Check if the asset exists
|
|
if (is_null($asset_to_clone = Asset::find($assetId))) {
|
|
// Redirect to the asset management page
|
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
}
|
|
|
|
$this->authorize('create', $asset_to_clone);
|
|
|
|
$asset = clone $asset_to_clone;
|
|
$asset->id = null;
|
|
$asset->asset_tag = '';
|
|
$asset->serial = '';
|
|
$asset->assigned_to = '';
|
|
|
|
return view('hardware/edit')
|
|
->with('statuslabel_list', Helper::statusLabelList())
|
|
->with('statuslabel_types', Helper::statusTypeList())
|
|
->with('item', $asset);
|
|
}
|
|
|
|
/**
|
|
* Return history import view
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getImportHistory()
|
|
{
|
|
$this->authorize('admin');
|
|
|
|
return view('hardware/history');
|
|
}
|
|
|
|
/**
|
|
* Import history
|
|
*
|
|
* This needs a LOT of love. It's done very inelegantly right now, and there are
|
|
* a ton of optimizations that could (and should) be done.
|
|
*
|
|
* Updated to respect checkin dates:
|
|
* No checkin column, assume all items are checked in (todays date)
|
|
* Checkin date in the past, update history.
|
|
* Checkin date in future or empty, check the item out to the user.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v3.3]
|
|
* @return View
|
|
*/
|
|
public function postImportHistory(Request $request)
|
|
{
|
|
if (! $request->hasFile('user_import_csv')) {
|
|
return back()->with('error', 'No file provided. Please select a file for import and try again. ');
|
|
}
|
|
|
|
if (! ini_get('auto_detect_line_endings')) {
|
|
ini_set('auto_detect_line_endings', '1');
|
|
}
|
|
$csv = Reader::createFromPath($request->file('user_import_csv'));
|
|
$csv->setHeaderOffset(0);
|
|
$header = $csv->getHeader();
|
|
$isCheckinHeaderExplicit = in_array('checkin date', (array_map('strtolower', $header)));
|
|
$results = $csv->getRecords();
|
|
$item = [];
|
|
$status = [];
|
|
$status['error'] = [];
|
|
$status['success'] = [];
|
|
foreach ($results as $row) {
|
|
if (is_array($row)) {
|
|
$row = array_change_key_case($row, CASE_LOWER);
|
|
$asset_tag = Helper::array_smart_fetch($row, 'asset tag');
|
|
if (! array_key_exists($asset_tag, $item)) {
|
|
$item[$asset_tag] = [];
|
|
}
|
|
$batch_counter = count($item[$asset_tag]);
|
|
$item[$asset_tag][$batch_counter]['checkout_date'] = Carbon::parse(Helper::array_smart_fetch($row, 'checkout date'))->format('Y-m-d H:i:s');
|
|
|
|
if ($isCheckinHeaderExplicit) {
|
|
//checkin date not empty, assume past transaction or future checkin date (expected)
|
|
if (! empty(Helper::array_smart_fetch($row, 'checkin date'))) {
|
|
$item[$asset_tag][$batch_counter]['checkin_date'] = Carbon::parse(Helper::array_smart_fetch($row, 'checkin date'))->format('Y-m-d H:i:s');
|
|
} else {
|
|
$item[$asset_tag][$batch_counter]['checkin_date'] = '';
|
|
}
|
|
} else {
|
|
//checkin header missing, assume data is unavailable and make checkin date explicit (now) so we don't encounter invalid state.
|
|
$item[$asset_tag][$batch_counter]['checkin_date'] = Carbon::parse(now())->format('Y-m-d H:i:s');
|
|
}
|
|
|
|
$item[$asset_tag][$batch_counter]['asset_tag'] = Helper::array_smart_fetch($row, 'asset tag');
|
|
$item[$asset_tag][$batch_counter]['name'] = Helper::array_smart_fetch($row, 'name');
|
|
$item[$asset_tag][$batch_counter]['email'] = Helper::array_smart_fetch($row, 'email');
|
|
if ($asset = Asset::where('asset_tag', '=', $asset_tag)->first()) {
|
|
$item[$asset_tag][$batch_counter]['asset_id'] = $asset->id;
|
|
$base_username = User::generateFormattedNameFromFullName(Setting::getSettings()->username_format, $item[$asset_tag][$batch_counter]['name']);
|
|
$user = User::where('username', '=', $base_username['username']);
|
|
$user_query = ' on username '.$base_username['username'];
|
|
if ($request->input('match_firstnamelastname') == '1') {
|
|
$firstnamedotlastname = User::generateFormattedNameFromFullName('firstname.lastname', $item[$asset_tag][$batch_counter]['name']);
|
|
$item[$asset_tag][$batch_counter]['username'][] = $firstnamedotlastname['username'];
|
|
$user->orWhere('username', '=', $firstnamedotlastname['username']);
|
|
$user_query .= ', or on username '.$firstnamedotlastname['username'];
|
|
}
|
|
if ($request->input('match_flastname') == '1') {
|
|
$flastname = User::generateFormattedNameFromFullName('filastname', $item[$asset_tag][$batch_counter]['name']);
|
|
$item[$asset_tag][$batch_counter]['username'][] = $flastname['username'];
|
|
$user->orWhere('username', '=', $flastname['username']);
|
|
$user_query .= ', or on username '.$flastname['username'];
|
|
}
|
|
if ($request->input('match_firstname') == '1') {
|
|
$firstname = User::generateFormattedNameFromFullName('firstname', $item[$asset_tag][$batch_counter]['name']);
|
|
$item[$asset_tag][$batch_counter]['username'][] = $firstname['username'];
|
|
$user->orWhere('username', '=', $firstname['username']);
|
|
$user_query .= ', or on username '.$firstname['username'];
|
|
}
|
|
if ($request->input('match_email') == '1') {
|
|
if ($item[$asset_tag][$batch_counter]['name'] == '') {
|
|
$item[$asset_tag][$batch_counter]['username'][] = $user_email = User::generateEmailFromFullName($item[$asset_tag][$batch_counter]['name']);
|
|
$user->orWhere('username', '=', $user_email);
|
|
$user_query .= ', or on username '.$user_email;
|
|
}
|
|
}
|
|
if ($request->input('match_username') == '1') {
|
|
// Added #8825: add explicit username lookup
|
|
$raw_username = $item[$asset_tag][$batch_counter]['name'];
|
|
$user->orWhere('username', '=', $raw_username);
|
|
$user_query .= ', or on username '.$raw_username;
|
|
}
|
|
|
|
// A matching user was found
|
|
if ($user = $user->first()) {
|
|
//$user is now matched user from db
|
|
$item[$asset_tag][$batch_counter]['user_id'] = $user->id;
|
|
|
|
Actionlog::firstOrCreate([
|
|
'item_id' => $asset->id,
|
|
'item_type' => Asset::class,
|
|
'user_id' => Auth::user()->id,
|
|
'note' => 'Checkout imported by '.Auth::user()->present()->fullName().' from history importer',
|
|
'target_id' => $item[$asset_tag][$batch_counter]['user_id'],
|
|
'target_type' => User::class,
|
|
'created_at' => $item[$asset_tag][$batch_counter]['checkout_date'],
|
|
'action_type' => 'checkout',
|
|
]);
|
|
|
|
$checkin_date = $item[$asset_tag][$batch_counter]['checkin_date'];
|
|
|
|
if ($isCheckinHeaderExplicit) {
|
|
|
|
//if checkin date header exists, assume that empty or future date is still checked out
|
|
//if checkin is before todays date, assume it's checked in and do not assign user ID, if checkin date is in the future or blank, this is the expected checkin date, items is checked out
|
|
|
|
if ((strtotime($checkin_date) > strtotime(Carbon::now())) || (empty($checkin_date))
|
|
) {
|
|
//only do this if item is checked out
|
|
$asset->assigned_to = $user->id;
|
|
$asset->assigned_type = User::class;
|
|
}
|
|
}
|
|
|
|
if (! empty($checkin_date)) {
|
|
//only make a checkin there is a valid checkin date or we created one on import.
|
|
Actionlog::firstOrCreate([
|
|
'item_id' => $item[$asset_tag][$batch_counter]['asset_id'],
|
|
'item_type' => Asset::class,
|
|
'user_id' => Auth::user()->id,
|
|
'note' => 'Checkin imported by '.Auth::user()->present()->fullName().' from history importer',
|
|
'target_id' => null,
|
|
'created_at' => $checkin_date,
|
|
'action_type' => 'checkin',
|
|
]);
|
|
}
|
|
|
|
if ($asset->save()) {
|
|
$status['success'][]['asset'][$asset_tag]['msg'] = 'Asset successfully matched for '.Helper::array_smart_fetch($row, 'name').$user_query.' on '.$item[$asset_tag][$batch_counter]['checkout_date'];
|
|
} else {
|
|
$status['error'][]['asset'][$asset_tag]['msg'] = 'Asset and user was matched but could not be saved.';
|
|
}
|
|
} else {
|
|
$item[$asset_tag][$batch_counter]['user_id'] = null;
|
|
$status['error'][]['user'][Helper::array_smart_fetch($row, 'name')]['msg'] = 'User does not exist so no checkin log was created.';
|
|
}
|
|
} else {
|
|
$item[$asset_tag][$batch_counter]['asset_id'] = null;
|
|
$status['error'][]['asset'][$asset_tag]['msg'] = 'Asset does not exist so no match was attempted.';
|
|
}
|
|
}
|
|
}
|
|
|
|
return view('hardware/history')->with('status', $status);
|
|
}
|
|
|
|
public function sortByName(array $recordA, array $recordB): int
|
|
{
|
|
return strcmp($recordB['Full Name'], $recordA['Full Name']);
|
|
}
|
|
|
|
/**
|
|
* Retore a deleted asset.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $assetId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getRestore($assetId = null)
|
|
{
|
|
// Get asset information
|
|
$asset = Asset::withTrashed()->find($assetId);
|
|
$this->authorize('delete', $asset);
|
|
if (isset($asset->id)) {
|
|
// Restore the asset
|
|
Asset::withTrashed()->where('id', $assetId)->restore();
|
|
|
|
$logaction = new Actionlog();
|
|
$logaction->item_type = Asset::class;
|
|
$logaction->item_id = $asset->id;
|
|
$logaction->created_at = date('Y-m-d H:i:s');
|
|
$logaction->user_id = Auth::user()->id;
|
|
$logaction->logaction('restored');
|
|
|
|
return redirect()->route('hardware.index')->with('success', trans('admin/hardware/message.restore.success'));
|
|
}
|
|
|
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
}
|
|
|
|
public function quickScan()
|
|
{
|
|
$this->authorize('audit', Asset::class);
|
|
$dt = Carbon::now()->addMonths(12)->toDateString();
|
|
|
|
return view('hardware/quickscan')->with('next_audit_date', $dt);
|
|
}
|
|
|
|
public function quickScanCheckin()
|
|
{
|
|
$this->authorize('checkin', Asset::class);
|
|
|
|
return view('hardware/quickscan-checkin');
|
|
}
|
|
|
|
public function audit($id)
|
|
{
|
|
$settings = Setting::getSettings();
|
|
$this->authorize('audit', Asset::class);
|
|
$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');
|
|
}
|
|
|
|
public function dueForAudit()
|
|
{
|
|
$this->authorize('audit', Asset::class);
|
|
|
|
return view('hardware/audit-due');
|
|
}
|
|
|
|
public function overdueForAudit()
|
|
{
|
|
$this->authorize('audit', Asset::class);
|
|
|
|
return view('hardware/audit-overdue');
|
|
}
|
|
|
|
|
|
public function auditStore(Request $request, $id)
|
|
{
|
|
$this->authorize('audit', Asset::class);
|
|
|
|
$rules = [
|
|
'location_id' => 'exists:locations,id|nullable|numeric',
|
|
'next_audit_date' => 'date|nullable',
|
|
];
|
|
|
|
$validator = \Validator::make($request->all(), $rules);
|
|
|
|
if ($validator->fails()) {
|
|
return response()->json(Helper::formatStandardApiResponse('error', null, $validator->errors()->all()));
|
|
}
|
|
|
|
$asset = Asset::findOrFail($id);
|
|
|
|
// We don't want to log this as a normal update, so let's bypass that
|
|
$asset->unsetEventDispatcher();
|
|
|
|
$asset->next_audit_date = $request->input('next_audit_date');
|
|
$asset->last_audit_date = date('Y-m-d H:i:s');
|
|
|
|
// Check to see if they checked the box to update the physical location,
|
|
// not just note it in the audit notes
|
|
if ($request->input('update_location') == '1') {
|
|
\Log::debug('update location in audit');
|
|
$asset->location_id = $request->input('location_id');
|
|
}
|
|
|
|
|
|
if ($asset->save()) {
|
|
$file_name = '';
|
|
// Upload an image, if attached
|
|
if ($request->hasFile('image')) {
|
|
$path = 'private_uploads/audits';
|
|
if (! Storage::exists($path)) {
|
|
Storage::makeDirectory($path, 775);
|
|
}
|
|
$upload = $image = $request->file('image');
|
|
$ext = $image->getClientOriginalExtension();
|
|
$file_name = 'audit-'.str_random(18).'.'.$ext;
|
|
Storage::putFileAs($path, $upload, $file_name);
|
|
}
|
|
|
|
|
|
$asset->logAudit($request->input('note'), $request->input('location_id'), $file_name);
|
|
return redirect()->to('hardware')->with('success', trans('admin/hardware/message.audit.success'));
|
|
}
|
|
}
|
|
|
|
public function getRequestedIndex($user_id = null)
|
|
{
|
|
$requestedItems = CheckoutRequest::with('user', 'requestedItem')->whereNull('canceled_at')->with('user', 'requestedItem');
|
|
|
|
if ($user_id) {
|
|
$requestedItems->where('user_id', $user_id)->get();
|
|
}
|
|
|
|
$requestedItems = $requestedItems->orderBy('created_at', 'desc')->get();
|
|
|
|
return view('hardware/requested', compact('requestedItems'));
|
|
}
|
|
}
|