firefly-iii/app/Http/Controllers/CsvController.php

411 lines
12 KiB
PHP
Raw Normal View History

2015-07-03 03:45:00 -05:00
<?php
namespace FireflyIII\Http\Controllers;
use Config;
2015-07-09 09:37:42 -05:00
use ExpandedForm;
use FireflyIII\Exceptions\FireflyException;
2015-07-04 23:18:02 -05:00
use FireflyIII\Helpers\Csv\Data;
use FireflyIII\Helpers\Csv\Importer;
use FireflyIII\Helpers\Csv\WizardInterface;
2015-07-09 09:37:42 -05:00
use FireflyIII\Repositories\Account\AccountRepositoryInterface;
use Illuminate\Http\Request;
use Input;
use Log;
2015-07-05 07:37:36 -05:00
use Preferences;
use Session;
2015-07-03 03:45:00 -05:00
use View;
/**
* Class CsvController
*
* @package FireflyIII\Http\Controllers
*/
class CsvController extends Controller
{
2015-07-04 23:18:02 -05:00
/** @var Data */
protected $data;
/** @var WizardInterface */
protected $wizard;
2015-07-03 03:45:00 -05:00
/**
*
*/
public function __construct()
{
parent::__construct();
View::share('title', trans('firefly.csv'));
2015-07-03 03:45:00 -05:00
View::share('mainTitleIcon', 'fa-file-text-o');
if (Config::get('firefly.csv_import_enabled') === false) {
throw new FireflyException('CSV Import is not enabled.');
}
2015-07-07 12:09:45 -05:00
$this->wizard = app('FireflyIII\Helpers\Csv\WizardInterface');
$this->data = app('FireflyIII\Helpers\Csv\Data');
2015-07-04 23:18:02 -05:00
2015-07-03 03:45:00 -05:00
}
/**
* Define column roles and mapping.
*
* STEP THREE
*
2015-07-06 03:39:44 -05:00
* @return \Illuminate\Http\RedirectResponse|\Illuminate\View\View
*/
public function columnRoles()
{
2015-07-04 23:18:02 -05:00
2015-07-09 11:38:15 -05:00
$fields = ['csv-file', 'csv-date-format', 'csv-has-headers', 'csv-import-account'];
2015-07-04 23:18:02 -05:00
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
2015-07-06 01:33:39 -05:00
$subTitle = trans('firefly.csv_define_column_roles');
2015-07-04 23:18:02 -05:00
$firstRow = $this->data->getReader()->fetchOne();
$count = count($firstRow);
$headers = [];
2015-07-05 07:37:36 -05:00
$example = $this->data->getReader()->fetchOne(1);
2015-07-04 23:18:02 -05:00
$availableRoles = [];
$roles = $this->data->getRoles();
$map = $this->data->getMap();
for ($i = 1; $i <= $count; $i++) {
$headers[] = trans('firefly.csv_column') . ' #' . $i;
}
2015-07-06 15:23:34 -05:00
if ($this->data->hasHeaders()) {
$headers = $firstRow;
}
foreach (Config::get('csv.roles') as $name => $role) {
2015-07-06 03:39:44 -05:00
$availableRoles[$name] = trans('firefly.csv_column_' . $name);//$role['name'];
}
2015-07-04 23:18:02 -05:00
ksort($availableRoles);
return view('csv.column-roles', compact('availableRoles', 'map', 'roles', 'headers', 'example', 'subTitle'));
}
2015-07-04 23:18:02 -05:00
/**
* Optional download of mapping.
*
* STEP FOUR THREE-A
2015-07-06 09:08:36 -05:00
*
* @return \Illuminate\Http\RedirectResponse|string
2015-07-04 23:18:02 -05:00
*/
public function downloadConfig()
{
$fields = ['csv-date-format', 'csv-has-headers'];
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
2015-07-04 23:18:02 -05:00
}
$data = [
'date-format' => Session::get('csv-date-format'),
2015-07-04 23:18:02 -05:00
'has-headers' => Session::get('csv-has-headers')
];
if (Session::has('csv-map')) {
$data['map'] = Session::get('csv-map');
}
if (Session::has('csv-roles')) {
$data['roles'] = Session::get('csv-roles');
}
if (Session::has('csv-mapped')) {
$data['mapped'] = Session::get('csv-mapped');
}
2015-07-09 12:05:59 -05:00
if (Session::has('csv-specifix')) {
$data['specifix'] = Session::get('csv-specifix');
}
2015-07-04 23:18:02 -05:00
$result = json_encode($data, JSON_PRETTY_PRINT);
$name = 'csv-configuration-' . date('Y-m-d') . '.json';
header('Content-disposition: attachment; filename=' . $name);
header('Content-type: application/json');
echo $result;
2015-07-06 09:08:36 -05:00
return '';
2015-07-04 23:18:02 -05:00
}
/**
2015-07-06 09:08:36 -05:00
* @return \Illuminate\View\View
2015-07-04 23:18:02 -05:00
*/
public function downloadConfigPage()
{
2015-07-06 01:33:39 -05:00
$subTitle = trans('firefly.csv_download_config_title');
return view('csv.download-config', compact('subTitle'));
}
/**
* This method shows the initial upload form.
*
* STEP ONE
*
2015-07-06 09:08:36 -05:00
* @return \Illuminate\View\View
*/
2015-07-09 09:37:42 -05:00
public function index(AccountRepositoryInterface $repository)
2015-07-03 03:45:00 -05:00
{
$subTitle = trans('firefly.csv_import');
Session::forget('csv-date-format');
Session::forget('csv-has-headers');
Session::forget('csv-file');
2015-07-09 09:37:42 -05:00
Session::forget('csv-import-account');
2015-07-04 23:18:02 -05:00
Session::forget('csv-map');
Session::forget('csv-roles');
Session::forget('csv-mapped');
2015-07-05 11:18:44 -05:00
Session::forget('csv-specifix');
2015-07-06 13:21:55 -05:00
// get list of supported specifix
$specifix = [];
foreach (Config::get('csv.specifix') as $entry) {
$specifix[$entry] = trans('firefly.csv_specifix_' . $entry);
}
2015-07-09 09:37:42 -05:00
// get a list of asset accounts:
$accounts = ExpandedForm::makeSelectList($repository->getAccounts(['Asset account', 'Default account']));
// can actually upload?
2015-07-03 05:22:20 -05:00
$uploadPossible = is_writable(storage_path('upload'));
$path = storage_path('upload');
2015-07-09 09:37:42 -05:00
return view('csv.index', compact('subTitle', 'uploadPossible', 'path', 'specifix', 'accounts'));
}
/**
* Parse the file.
*
* STEP FOUR
*
* @return \Illuminate\Http\RedirectResponse
*/
public function initialParse()
{
$fields = ['csv-file', 'csv-date-format', 'csv-has-headers'];
2015-07-04 23:18:02 -05:00
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
2015-07-04 23:18:02 -05:00
// process given roles and mapping:
$roles = $this->wizard->processSelectedRoles(Input::get('role'));
$maps = $this->wizard->processSelectedMapping($roles, Input::get('map'));
Session::put('csv-map', $maps);
Session::put('csv-roles', $roles);
2015-07-06 15:23:34 -05:00
// Go back when no roles defined:
if (count($roles) === 0) {
Session::flash('warning', 'Please select some roles.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.column-roles'));
}
/*
2015-07-04 23:18:02 -05:00
* Continue with map specification when necessary.
*/
2015-07-04 23:18:02 -05:00
if (count($maps) > 0) {
2015-07-06 09:08:36 -05:00
return redirect(route('csv.map'));
}
2015-07-04 23:18:02 -05:00
/*
* Or simply start processing.
*/
// proceed to download config
2015-07-06 09:08:36 -05:00
return redirect(route('csv.download-config-page'));
}
/**
*
* Map first if necessary,
*
* STEP FIVE.
*
2015-07-06 09:08:36 -05:00
* @return \Illuminate\Http\RedirectResponse|\Illuminate\View\View
* @throws FireflyException
*/
public function map()
{
2015-07-06 15:12:35 -05:00
// Make sure all fields we need are accounted for.
$fields = ['csv-file', 'csv-date-format', 'csv-has-headers', 'csv-map', 'csv-roles'];
2015-07-04 23:18:02 -05:00
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
/*
* The "options" array contains all options the user has
* per column, where the key represents the column.
*
* For each key there is an array which in turn represents
* all the options available: grouped by ID.
2015-07-04 23:18:02 -05:00
*
* options[column index] = [
2015-07-06 15:12:35 -05:00
* field id => field identifier.
2015-07-04 23:18:02 -05:00
* ]
*/
2015-07-04 23:18:02 -05:00
try {
$options = $this->wizard->showOptions($this->data->getMap());
} catch (FireflyException $e) {
return view('error', ['message' => $e->getMessage()]);
}
2015-07-06 15:12:35 -05:00
// After these values are prepped, read the actual CSV file
2015-07-04 23:18:02 -05:00
$reader = $this->data->getReader();
$map = $this->data->getMap();
2015-07-06 15:23:34 -05:00
$hasHeaders = $this->data->hasHeaders();
2015-07-04 23:18:02 -05:00
$values = $this->wizard->getMappableValues($reader, $map, $hasHeaders);
$map = $this->data->getMap();
$mapped = $this->data->getMapped();
2015-07-06 01:33:39 -05:00
$subTitle = trans('firefly.csv_map_values');
2015-07-04 23:18:02 -05:00
2015-07-06 01:33:39 -05:00
return view('csv.map', compact('map', 'options', 'values', 'mapped', 'subTitle'));
}
/**
2015-07-06 09:08:36 -05:00
*
* Finally actually process the CSV file.
*
* STEP SEVEN
2015-07-06 09:08:36 -05:00
*
* @return \Illuminate\Http\RedirectResponse|\Illuminate\View\View
*/
public function process()
{
/*
* Make sure all fields we need are accounted for.
*/
$fields = ['csv-file', 'csv-date-format', 'csv-has-headers', 'csv-map', 'csv-roles', 'csv-mapped'];
2015-07-04 23:18:02 -05:00
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
Log::debug('Created importer');
2015-07-04 23:18:02 -05:00
$importer = new Importer;
$importer->setData($this->data);
try {
$importer->run();
} catch (FireflyException $e) {
Log::error('Catch error: ' . $e->getMessage());
2015-07-04 23:18:02 -05:00
return view('error', ['message' => $e->getMessage()]);
}
Log::debug('Done importing!');
2015-07-04 23:18:02 -05:00
2015-07-05 07:37:36 -05:00
$rows = $importer->getRows();
$errors = $importer->getErrors();
$imported = $importer->getImported();
2015-07-09 11:38:15 -05:00
$journals = $importer->getJournals();
2015-07-05 07:37:36 -05:00
Preferences::mark();
2015-07-06 01:33:39 -05:00
$subTitle = trans('firefly.csv_process_title');
2015-07-09 11:38:15 -05:00
return view('csv.process', compact('rows', 'errors', 'imported', 'subTitle', 'journals'));
2015-07-04 23:18:02 -05:00
}
/**
* Store the mapping the user has made. This is
*
* STEP SIX
2015-07-06 09:08:36 -05:00
*
* @return \Illuminate\Http\RedirectResponse
*/
public function saveMapping()
{
/*
* Make sure all fields we need are accounted for.
*/
$fields = ['csv-file', 'csv-date-format', 'csv-has-headers', 'csv-map', 'csv-roles'];
2015-07-04 23:18:02 -05:00
if (!$this->wizard->sessionHasValues($fields)) {
Session::flash('warning', 'Could not recover upload.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
2015-07-04 23:18:02 -05:00
// save mapping to session.
$mapped = [];
if (!is_array(Input::get('mapping'))) {
Session::flash('warning', 'Invalid mapping.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.map'));
}
foreach (Input::get('mapping') as $index => $data) {
$mapped[$index] = [];
foreach ($data as $value => $mapping) {
2015-07-06 01:33:39 -05:00
if (intval($mapping) !== 0) {
$mapped[$index][$value] = $mapping;
}
}
}
Session::put('csv-mapped', $mapped);
// proceed to process.
2015-07-06 09:08:36 -05:00
return redirect(route('csv.download-config-page'));
}
/**
*
* This method processes the file, puts it away somewhere safe
* and sends you onwards.
*
* STEP TWO
*
* @param Request $request
*
* @return \Illuminate\Http\RedirectResponse
*/
public function upload(Request $request)
{
if (!$request->hasFile('csv')) {
Session::flash('warning', 'No file uploaded.');
2015-07-06 09:08:36 -05:00
return redirect(route('csv.index'));
}
2015-07-09 09:37:42 -05:00
$fullPath = $this->wizard->storeCsvFile($request->file('csv')->getRealPath());
$settings = [];
$settings['date-format'] = Input::get('date_format');
$settings['has-headers'] = intval(Input::get('has_headers')) === 1;
$settings['specifix'] = Input::get('specifix');
$settings['import-account'] = intval(Input::get('csv_import_account'));
$settings['map'] = [];
$settings['mapped'] = [];
$settings['roles'] = [];
if ($request->hasFile('csv_config')) { // Process config file if present.
2015-07-04 23:18:02 -05:00
$data = file_get_contents($request->file('csv_config')->getRealPath());
$json = json_decode($data, true);
2015-07-06 09:08:36 -05:00
if (is_array($json)) {
$settings = array_merge($settings, $json);
2015-07-04 23:18:02 -05:00
}
}
2015-07-04 23:18:02 -05:00
$this->data->setCsvFileLocation($fullPath);
2015-07-06 09:08:36 -05:00
$this->data->setDateFormat($settings['date-format']);
$this->data->setHasHeaders($settings['has-headers']);
$this->data->setMap($settings['map']);
$this->data->setMapped($settings['mapped']);
$this->data->setRoles($settings['roles']);
2015-07-06 13:21:55 -05:00
$this->data->setSpecifix($settings['specifix']);
2015-07-09 09:37:42 -05:00
$this->data->setImportAccount($settings['import-account']);
2015-07-06 09:08:36 -05:00
return redirect(route('csv.column-roles'));
2015-07-03 03:45:00 -05:00
}
2015-07-09 14:26:40 -05:00
}