Improve web imported. Show a list of all items with that were not imported correctly after import. Modify ObjectImporter and add a web-import parameter that causes it to spit out json errors exclusively. Long term I want to separate the console command and the logic so we aren't calling the console command directly, but rather a class that does everything. This would allow for easier progress reports and ajaxification.

This commit is contained in:
Daniel Meltzer 2016-05-26 21:29:29 -05:00
parent 715e385925
commit 004c63cd5d
4 changed files with 113 additions and 47 deletions

View file

@ -55,15 +55,18 @@ class ObjectImportCommand extends Command {
public function fire() public function fire()
{ {
$filename = $this->argument('filename'); $filename = $this->argument('filename');
$logFile = $this->option('logfile');
\Log::useFiles($logFile);
if ($this->option('testrun')) {
$this->comment('====== TEST ONLY Asset Import for '.$filename.' ====');
$this->comment('============== NO DATA WILL BE WRITTEN ==============');
} else {
$this->comment('======= Importing Assets from '.$filename.' ========='); if(!$this->option('web-importer')) {
$logFile = $this->option('logfile');
\Log::useFiles($logFile);
if ($this->option('testrun')) {
$this->comment('====== TEST ONLY Asset Import for '.$filename.' ====');
$this->comment('============== NO DATA WILL BE WRITTEN ==============');
} else {
$this->comment('======= Importing Assets from '.$filename.' =========');
}
} }
if (! ini_get("auto_detect_line_endings")) { if (! ini_get("auto_detect_line_endings")) {
@ -87,11 +90,12 @@ class ObjectImportCommand extends Command {
$this->manufacturers = Manufacturer::All(['name', 'id']); $this->manufacturers = Manufacturer::All(['name', 'id']);
$this->asset_models = AssetModel::All(['name','modelno','category_id','manufacturer_id', 'id']); $this->asset_models = AssetModel::All(['name','modelno','category_id','manufacturer_id', 'id']);
$this->companies = Company::All(['name', 'id']); $this->companies = Company::All(['name', 'id']);
$this->status_labels = Statuslabel::All(['name', 'id']);
$this->suppliers = Supplier::All(['name', 'id']);
$this->assets = Asset::all(['asset_tag']); $this->assets = Asset::all(['asset_tag']);
$this->suppliers = Supplier::All(['name']);
$this->accessories = Accessory::All(['name']); $this->accessories = Accessory::All(['name']);
$this->consumables = Consumable::All(['name']); $this->consumables = Consumable::All(['name']);
$this->status_labels = Statuslabel::All(['name']);
// Loop through the records // Loop through the records
DB::transaction(function() use (&$newarray){ DB::transaction(function() use (&$newarray){
$item_type = strtolower($this->option('item-type')); $item_type = strtolower($this->option('item-type'));
@ -124,6 +128,7 @@ class ObjectImportCommand extends Command {
$this->log('Purchase Date: ' . $item["purchase_date"]); $this->log('Purchase Date: ' . $item["purchase_date"]);
$this->log('Purchase Cost: ' . $item["purchase_cost"]); $this->log('Purchase Cost: ' . $item["purchase_cost"]);
$this->log('Company Name: ' . $item_company_name); $this->log('Company Name: ' . $item_company_name);
$this->log('Status: ' . $item_status_name);
$item["user"] = $this->createOrFetchUser($row); $item["user"] = $this->createOrFetchUser($row);
@ -149,17 +154,30 @@ class ObjectImportCommand extends Command {
} }
}); });
$this->log('====================================='); $this->log('=====================================');
if(!empty($this->errors)) { if(!$this->option('web-importer'))
$this->comment("The following Errors were encountered."); {
foreach($this->errors as $error) if(!empty($this->errors)) {
{ $this->comment("The following Errors were encountered.");
$this->comment($error); foreach($this->errors as $asset => $error)
{
$this->comment('Error: Item: ' . $asset . 'failed validation: ' . json_encode($error));
}
} else {
$this->comment("All Items imported successfully!");
}
} else {
if(empty($this->errors))
return 0;
else {
$this->comment(json_encode($this->errors)); //Send a big string to the
return 1;
} }
} }
$this->comment(""); $this->comment("");
return true; return 2;
} }
// Tracks the current item for error messages // Tracks the current item for error messages
private $current_assetId; private $current_assetId;
@ -167,12 +185,11 @@ class ObjectImportCommand extends Command {
// An array of errors encountered while parsing // An array of errors encountered while parsing
private $errors; private $errors;
public function error($string, $verbosity = null) public function jsonError($field, $errorString)
{ {
$errorString = 'Error: Item ' . $this->current_assetId . ': ' . $string; $this->errors[$this->current_assetId] = array($field => $errorString);
$this->errors[] = $errorString;
if($this->option('verbose')) if($this->option('verbose'))
parent::error($string, $verbosity); parent::error($errorString);
} }
/** /**
@ -183,6 +200,8 @@ class ObjectImportCommand extends Command {
*/ */
private function log($string, $level = 'info') private function log($string, $level = 'info')
{ {
if($this->option('web-importer'))
return;
if($level === 'warning') if($level === 'warning')
{ {
\Log::warning($string); \Log::warning($string);
@ -252,7 +271,7 @@ class ObjectImportCommand extends Command {
$this->log('Asset Model ' . $asset_model_name . ' with model number ' . $asset_modelno . ' was created'); $this->log('Asset Model ' . $asset_model_name . ' with model number ' . $asset_modelno . ' was created');
return $asset_model; return $asset_model;
} else { } else {
$this->error('Asset Model: ' . $asset_model->getErrors()); $this->jsonError('Asset Model', $asset_model->getErrors());
return $asset_model; return $asset_model;
} }
} else { } else {
@ -295,7 +314,7 @@ class ObjectImportCommand extends Command {
$this->log('Category ' . $asset_category . ' was created'); $this->log('Category ' . $asset_category . ' was created');
return $category; return $category;
} else { } else {
$this->error('Category: ' . $category->getErrors()); $this->jsonError('Category', $category->getErrors());
return $category; return $category;
} }
} else { } else {
@ -329,7 +348,7 @@ class ObjectImportCommand extends Command {
$this->log('Company ' . $asset_company_name . ' was created'); $this->log('Company ' . $asset_company_name . ' was created');
return $company; return $company;
} else { } else {
$this->error('Company: ' . $company->getErrors()); $this->jsonError('Company', $company->getErrors());
} }
} else { } else {
$this->companies->add($company); $this->companies->add($company);
@ -361,7 +380,7 @@ class ObjectImportCommand extends Command {
$this->log('Status ' . $asset_statuslabel_name . ' was created'); $this->log('Status ' . $asset_statuslabel_name . ' was created');
return $status; return $status;
} else { } else {
$this->error('Status: ' . $status->getErrors()); $this->jsonError('Status', $status->getErrors());
return $status; return $status;
} }
} else { } else {
@ -390,7 +409,7 @@ class ObjectImportCommand extends Command {
foreach ($this->manufacturers as $tempmanufacturer) { foreach ($this->manufacturers as $tempmanufacturer) {
if ($tempmanufacturer->name === $asset_mfgr) { if ($tempmanufacturer->name === $asset_mfgr) {
$this->log('Manufacturer ' . $asset_mfgr . ' already exists'); $this->log('{Manufacturer [' . $asset_mfgr . ' already exists') . ']}';
return $tempmanufacturer; return $tempmanufacturer;
} }
} }
@ -407,7 +426,7 @@ class ObjectImportCommand extends Command {
$this->log('Manufacturer ' . $manufacturer->name . ' was created'); $this->log('Manufacturer ' . $manufacturer->name . ' was created');
return $manufacturer; return $manufacturer;
} else { } else {
$this->error('Manufacturer: ' . $manufacturer->getErrors()); $this->jsonError('Manufacturer', $manufacturer->getErrors());
return $manufacturer; return $manufacturer;
} }
@ -451,7 +470,7 @@ class ObjectImportCommand extends Command {
$this->log('Location ' . $asset_location . ' was created'); $this->log('Location ' . $asset_location . ' was created');
return $location; return $location;
} else { } else {
$this->error('Location: ' . $location->getErrors()); $this->jsonError('Location', $location->getErrors()) ;
return $location; return $location;
} }
} else { } else {
@ -493,7 +512,7 @@ class ObjectImportCommand extends Command {
$this->log('Supplier ' . $supplier_name . ' was created'); $this->log('Supplier ' . $supplier_name . ' was created');
return $supplier; return $supplier;
} else { } else {
$this->error('Supplier: ' . $supplier->getErrors()); $this->jsonError('Supplier', $supplier->getErrors());
return $supplier; return $supplier;
} }
} else { } else {
@ -579,7 +598,7 @@ class ObjectImportCommand extends Command {
if ($user->save()) { if ($user->save()) {
$this->log('User '.$first_name.' created'); $this->log('User '.$first_name.' created');
} else { } else {
$this->error('User: ' . $user->getErrors()); $this->jsonError('User', $user->getErrors());
} }
} else { } else {
@ -620,16 +639,17 @@ class ObjectImportCommand extends Command {
foreach ($this->assets as $tempasset) { foreach ($this->assets as $tempasset) {
if ($tempasset->asset_tag === $asset_tag ) { if ($tempasset->asset_tag === $asset_tag ) {
$this->log('A matching Asset ' . $asset_tag . ' already exists'); $this->log('A matching Asset ' . $asset_tag . ' already exists');
$this->comment('A matching Asset ' . $asset_tag . ' already exists'); // $this->comment('A matching Asset ' . $asset_tag . ' already exists');
return; return;
} }
} }
if(empty($item["status_label"])) { if($item["status_label"]) {
$status_id = $item["status_label"]->id;
} else {
$this->log("No status field found, defaulting to id 1."); $this->log("No status field found, defaulting to id 1.");
$status_id = 1; $status_id = 1;
} else {
$status_id = $item["status_label"]->id;
} }
$asset = new Asset(); $asset = new Asset();
@ -640,8 +660,10 @@ class ObjectImportCommand extends Command {
$asset->purchase_date = NULL; $asset->purchase_date = NULL;
} }
if (!empty($item_purchase_cost)) { if (!empty($item["purchase_cost"])) {
$asset->purchase_cost = number_format($item["purchase_cost"],2); //TODO How to generalize this for not USD?
$purchase_cost = substr($item["purchase_cost"],0,1) === '$' ? substr($item["purchase_cost"],1) : $item["purchase_cost"];
$asset->purchase_cost = number_format($purchase_cost,2);
$this->log("Asset cost parsed: " . $asset->purchase_cost); $this->log("Asset cost parsed: " . $asset->purchase_cost);
} else { } else {
$asset->purchase_cost = 0.00; $asset->purchase_cost = 0.00;
@ -656,6 +678,7 @@ class ObjectImportCommand extends Command {
if($item["location"]) if($item["location"])
$asset->rtd_location_id = $item["location"]->id; $asset->rtd_location_id = $item["location"]->id;
$asset->user_id = 1; $asset->user_id = 1;
$this->log("status_id: " . $status_id);
$asset->status_id = $status_id; $asset->status_id = $status_id;
if($item["company"]) if($item["company"])
$asset->company_id = $item["company"]->id; $asset->company_id = $item["company"]->id;
@ -669,9 +692,8 @@ class ObjectImportCommand extends Command {
if ($asset->save()) { if ($asset->save()) {
$this->log('Asset ' . $item["item_name"] . ' with serial number ' . $asset_serial . ' was created'); $this->log('Asset ' . $item["item_name"] . ' with serial number ' . $asset_serial . ' was created');
$this->comment('Asset ' . $item["item_name"] . ' with serial number ' . $asset_serial . ' was created');
} else { } else {
$this->error('Asset: ' . $asset->getErrors()); $this->jsonError('Asset', $asset->getErrors());
} }
} else { } else {
@ -732,10 +754,10 @@ class ObjectImportCommand extends Command {
if (!$this->option('testrun')) { if (!$this->option('testrun')) {
if ($accessory->save()) { if ($accessory->save()) {
$this->log('Accessory ' . $item["item_name"] . ' was created'); $this->log('Accessory ' . $item["item_name"] . ' was created');
$this->comment('Accessory ' . $item["item_name"] . ' was created'); // $this->comment('Accessory ' . $item["item_name"] . ' was created');
} else { } else {
$this->error('Accessory: ' . $accessory->getErrors()); $this->jsonError('Accessory', $accessory->getErrors()) ;
} }
} else { } else {
$this->log('TEST RUN - Accessory ' . $item["item_name"] . ' not created'); $this->log('TEST RUN - Accessory ' . $item["item_name"] . ' not created');
@ -791,10 +813,10 @@ class ObjectImportCommand extends Command {
if(!$this->option("testrun")) { if(!$this->option("testrun")) {
if($consumable->save()) { if($consumable->save()) {
$this->log("Consumable " . $item["item_name"] . ' was created'); $this->log("Consumable " . $item["item_name"] . ' was created');
$this->comment("Consumable " . $item["item_name"] . ' was created'); // $this->comment("Consumable " . $item["item_name"] . ' was created');
} else { } else {
$this->error('Consumable: ' . $consumable->getErrors()); $this->jsonError('Consumable', $consumable->getErrors());
} }
} else { } else {
$this->log('TEST RUN - Consumable ' . $item['item_name'] . ' not created'); $this->log('TEST RUN - Consumable ' . $item['item_name'] . ' not created');
@ -826,7 +848,8 @@ class ObjectImportCommand extends Command {
array('username_format', null, InputOption::VALUE_REQUIRED, 'The format of the username that should be generated. Options are firstname.lastname, firstname, filastname, email', null), array('username_format', null, InputOption::VALUE_REQUIRED, 'The format of the username that should be generated. Options are firstname.lastname, firstname, filastname, email', null),
array('testrun', null, InputOption::VALUE_NONE, 'If set, will parse and output data without adding to database', null), array('testrun', null, InputOption::VALUE_NONE, 'If set, will parse and output data without adding to database', null),
array('logfile', null, InputOption::VALUE_REQUIRED, 'The path to log output to. storage/logs/importer.log by default', storage_path('logs/importer.log') ), array('logfile', null, InputOption::VALUE_REQUIRED, 'The path to log output to. storage/logs/importer.log by default', storage_path('logs/importer.log') ),
array('item-type', null, InputOption::VALUE_REQUIRED, 'Item Type To import. Valid Options are Asset, Consumable, Or Accessory', 'Asset') array('item-type', null, InputOption::VALUE_REQUIRED, 'Item Type To import. Valid Options are Asset, Consumable, Or Accessory', 'Asset'),
array('web-importer', null, InputOption::VALUE_NONE, 'Internal: packages output for use with the web importer')
); );
} }

View file

@ -843,14 +843,20 @@ class AssetsController extends Controller
return redirect()->to('hardware')->with('error', trans('general.insufficient_permissions')); return redirect()->to('hardware')->with('error', trans('general.insufficient_permissions'));
} }
$output = new BufferedOutput; $return = Artisan::call('snipeit:import',
Artisan::call('snipeit:import', ['filename'=> config('app.private_uploads').'/imports/assets/'.$filename, '--email_format'=>'firstname.lastname', '--username_format'=>'firstname.lastname'], $output); ['filename'=> config('app.private_uploads').'/imports/assets/'.$filename,
$display_output = $output->fetch(); '--email_format'=>'firstname.lastname',
'--username_format'=>'firstname.lastname',
'--web-importer' => true
]);
$display_output = Artisan::output();
$file = config('app.private_uploads').'/imports/assets/'.str_replace('.csv', '', $filename).'-output-'.date("Y-m-d-his").'.txt'; $file = config('app.private_uploads').'/imports/assets/'.str_replace('.csv', '', $filename).'-output-'.date("Y-m-d-his").'.txt';
file_put_contents($file, $display_output); file_put_contents($file, $display_output);
if( $return === 0) //Success
return redirect()->to('hardware')->with('success', trans('admin/hardware/message.import.success'));
return redirect()->to('hardware')->with('success', 'Your file has been imported'); else if( $return === 1) // Failure
return redirect()->back()->with('import_errors', json_decode($display_output))->with('error', trans('admin/hardware/message.import.error'));
dd("Shouldn't be here");
} }

View file

@ -36,6 +36,12 @@ return array(
'invalidfiles' => 'One or more of your files is too large or is a filetype that is not allowed. Allowed filetypes are png, gif, jpg, doc, docx, pdf, and txt.', 'invalidfiles' => 'One or more of your files is too large or is a filetype that is not allowed. Allowed filetypes are png, gif, jpg, doc, docx, pdf, and txt.',
), ),
'import' => array(
'error' => 'Some Items did not import Correctly.',
'errorDetail' => 'The Following Items were not imported because of errors.',
'success' => "Your File has been imported",
),
'delete' => array( 'delete' => array(
'confirm' => 'Are you sure you wish to delete this asset?', 'confirm' => 'Are you sure you wish to delete this asset?',

View file

@ -66,7 +66,38 @@
</div> </div>
</div> </div>
</div> </div>
</div> </div>
@if (session()->has('import_errors'))
<div class="errors-table">
<div class="alert alert-warning">
<strong>Warning</strong> {{trans('admin/hardware/message.import.errorDetail')}}
</div>
<table class="table table-striped" id="errors-table">
<thead>
<th>Asset</th>
<th colspan="1">Field</th>
<th colspan ="1">Parameter</th>
<th colspan ="1">Errors</th>
</thead>
<tbody>
@foreach (session('import_errors') as $asset => $error)
<tr>
<td> {{ $asset }}</td>
@foreach ($error as $field => $values )
<td> {{ $field }} </td>
@foreach( $values as $fieldName=>$errorString)
<td>{{$fieldName}}</td>
<td>{{$errorString[0]}}</td>
@endforeach
@endforeach
</tr>
@endforeach
</tbody>
</table>
</div>
@endif
</div> </div>
</div> </div>
@section('moar_scripts') @section('moar_scripts')