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

286 lines
9.3 KiB
PHP
Raw Normal View History

<?php
2016-08-05 23:21:25 -05:00
/**
* ImportController.php
* Copyright (C) 2016 thegrumpydictator@gmail.com
*
* This software may be modified and distributed under the terms of the
* Creative Commons Attribution-ShareAlike 4.0 International License.
*
* See the LICENSE file for details.
2016-08-05 23:21:25 -05:00
*/
2017-03-24 09:15:12 -05:00
declare(strict_types=1);
namespace FireflyIII\Http\Controllers;
2016-06-24 07:24:34 -05:00
use FireflyIII\Exceptions\FireflyException;
2016-06-10 14:00:00 -05:00
use FireflyIII\Http\Requests\ImportUploadRequest;
use FireflyIII\Import\Configurator\ConfiguratorInterface;
use FireflyIII\Import\FileProcessor\FileProcessorInterface;
2016-12-25 05:55:22 -06:00
use FireflyIII\Import\ImportProcedureInterface;
2016-06-10 14:00:00 -05:00
use FireflyIII\Models\ImportJob;
use FireflyIII\Repositories\ImportJob\ImportJobRepositoryInterface;
use FireflyIII\Repositories\Tag\TagRepositoryInterface;
2017-03-19 11:54:21 -05:00
use FireflyIII\Repositories\User\UserRepositoryInterface;
2016-06-24 07:24:34 -05:00
use Illuminate\Http\Request;
2017-01-14 12:43:33 -06:00
use Illuminate\Http\Response as LaravelResponse;
use Illuminate\Support\Collection;
2016-07-02 10:33:57 -05:00
use Log;
use Response;
use View;
/**
* Class ImportController
*
* @package FireflyIII\Http\Controllers
*/
class ImportController extends Controller
{
/**
*
*/
public function __construct()
{
parent::__construct();
2016-10-29 00:44:46 -05:00
$this->middleware(
function ($request, $next) {
View::share('mainTitleIcon', 'fa-archive');
View::share('title', trans('firefly.import_data_full'));
return $next($request);
}
);
}
2016-07-02 16:08:47 -05:00
/**
* This is step 3. This repeats until the job is configured.
2016-06-24 07:24:34 -05:00
*
2016-06-10 14:00:00 -05:00
* @param ImportJob $job
*
* @return View
2016-06-24 07:24:34 -05:00
* @throws FireflyException
2016-06-10 14:00:00 -05:00
*/
public function configure(ImportJob $job)
{
// create configuration class:
$configurator = $this->makeConfigurator($job);
2016-06-24 07:24:34 -05:00
// is the job already configured?
if ($configurator->isJobConfigured()) {
return redirect(route('import.status', [$job->key]));
}
$view = $configurator->getNextView();
$data = $configurator->getNextData();
2016-07-02 10:33:57 -05:00
$subTitle = trans('firefly.configure_import');
$subTitleIcon = 'fa-wrench';
2016-06-10 14:00:00 -05:00
return view($view, compact('data', 'job', 'subTitle', 'subTitleIcon'));
2016-07-02 16:08:47 -05:00
}
/**
* Generate a JSON file of the job's config and send it to the user.
*
* @param ImportJob $job
*
* @return mixed
*/
public function download(ImportJob $job)
{
Log::debug('Now in download()', ['job' => $job->key]);
$config = $job->configuration;
2017-06-14 13:13:19 -05:00
// TODO this is CSV import specific:
2016-07-02 16:08:47 -05:00
$config['column-roles-complete'] = false;
$config['column-mapping-complete'] = false;
2017-06-14 13:13:19 -05:00
$config['initial-config-complete'] = false;
2016-10-09 13:18:46 -05:00
$config['delimiter'] = $config['delimiter'] === "\t" ? 'tab' : $config['delimiter'];
2017-06-14 13:13:19 -05:00
2016-07-02 16:08:47 -05:00
$result = json_encode($config, JSON_PRETTY_PRINT);
$name = sprintf('"%s"', addcslashes('import-configuration-' . date('Y-m-d') . '.json', '"\\'));
2017-01-14 12:43:33 -06:00
/** @var LaravelResponse $response */
$response = response($result, 200);
$response->header('Content-disposition', 'attachment; filename=' . $name)
->header('Content-Type', 'application/json')
->header('Content-Description', 'File Transfer')
->header('Connection', 'Keep-Alive')
->header('Expires', '0')
->header('Cache-Control', 'must-revalidate, post-check=0, pre-check=0')
->header('Pragma', 'public')
->header('Content-Length', strlen($result));
return $response;
2016-07-02 16:08:47 -05:00
2016-06-10 14:00:00 -05:00
}
/**
2016-06-24 07:24:34 -05:00
* This is step 1. Upload a file.
*
2016-07-23 14:37:06 -05:00
* @return View
*/
public function index()
{
$subTitle = trans('firefly.import_data_index');
$subTitleIcon = 'fa-home';
$importFileTypes = [];
$defaultImportType = config('firefly.default_import_format');
foreach (array_keys(config('firefly.import_formats')) as $type) {
$importFileTypes[$type] = trans('firefly.import_file_type_' . $type);
}
return view('import.index', compact('subTitle', 'subTitleIcon', 'importFileTypes', 'defaultImportType'));
}
2016-06-10 14:00:00 -05:00
/**
* This is step 2. It creates an Import Job. Stores the import.
*
* @param ImportUploadRequest $request
* @param ImportJobRepositoryInterface $repository
* @param UserRepositoryInterface $userRepository
*
* @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector
*/
public function initialize(ImportUploadRequest $request, ImportJobRepositoryInterface $repository, UserRepositoryInterface $userRepository)
{
Log::debug('Now in initialize()');
// create import job:
$type = $request->get('import_file_type');
$job = $repository->create($type);
Log::debug('Created new job', ['key' => $job->key, 'id' => $job->id]);
// process file:
$repository->processFile($job, $request->files->get('import_file'));
// process config, if present:
if ($request->files->has('configuration_file')) {
$repository->processConfiguration($job, $request->files->get('configuration_file'));
}
return redirect(route('import.configure', [$job->key]));
}
/**
* @param ImportJob $job
*
* @return \Illuminate\Http\JsonResponse
*/
public function json(ImportJob $job)
{
2016-08-14 02:10:08 -05:00
$result = [
'started' => false,
'finished' => false,
'running' => false,
'errors' => $job->extended_status['errors'],
'percentage' => 0,
'steps' => $job->extended_status['total_steps'],
'stepsDone' => $job->extended_status['steps_done'],
'statusText' => trans('firefly.import_status_' . $job->status),
'status' => $job->status,
'finishedText' => '',
];
2016-08-14 02:10:08 -05:00
$percentage = 0;
if ($job->extended_status['total_steps'] !== 0) {
$percentage = round(($job->extended_status['steps_done'] / $job->extended_status['total_steps']) * 100, 0);
}
if ($job->status === 'complete') {
$tagId = $job->extended_status['importTag'];
/** @var TagRepositoryInterface $repository */
$repository = app(TagRepositoryInterface::class);
$tag = $repository->find($tagId);
$result['finished'] = true;
$result['finishedText'] = trans('firefly.import_finished_link', ['link' => route('tags.show', [$tag->id]), 'tag' => $tag->tag]);
2016-08-14 03:11:49 -05:00
}
2016-08-14 02:10:08 -05:00
if ($job->status === 'running') {
$result['started'] = true;
$result['running'] = true;
$result['percentage'] = $percentage;
}
return Response::json($result);
}
2016-06-10 14:00:00 -05:00
/**
2016-06-24 07:24:34 -05:00
* Step 4. Save the configuration.
*
2017-03-19 11:54:21 -05:00
* @param Request $request
* @param ImportJobRepositoryInterface $repository
* @param ImportJob $job
2016-06-24 07:24:34 -05:00
*
* @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector
*/
2017-03-19 11:54:21 -05:00
public function postConfigure(Request $request, ImportJobRepositoryInterface $repository, ImportJob $job)
2016-06-24 07:24:34 -05:00
{
2016-07-02 16:08:47 -05:00
Log::debug('Now in postConfigure()', ['job' => $job->key]);
$configurator = $this->makeConfigurator($job);
2016-06-24 07:24:34 -05:00
// is the job already configured?
if ($configurator->isJobConfigured()) {
return redirect(route('import.status', [$job->key]));
2016-07-02 10:33:57 -05:00
}
$data = $request->all();
$configurator->configureJob($data);
2016-07-02 10:33:57 -05:00
// return to configure
return redirect(route('import.configure', [$job->key]));
2016-07-02 10:33:57 -05:00
}
/**
* @param ImportJob $job
*/
public function start(ImportJob $job)
{
$objects = new Collection();
$type = $job->file_type;
$class = config(sprintf('firefly.import_processors.%s', $type));
/** @var FileProcessorInterface $processor */
2017-06-14 13:13:19 -05:00
$processor = app($class);
$processor->setJob($job);
set_time_limit(0);
if ($job->status == 'configured') {
$processor->run();
$objects = $processor->getObjects();
}
// once done, use storage thing to actually store them:
}
/**
* This is the last step before the import starts.
*
* @param ImportJob $job
*
* @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector|View
*/
public function status(ImportJob $job)
{
2016-08-13 16:28:01 -05:00
$subTitle = trans('firefly.import_status');
$subTitleIcon = 'fa-star';
return view('import.status', compact('job', 'subTitle', 'subTitleIcon'));
}
2016-06-24 07:24:34 -05:00
/**
* @param ImportJob $job
*
* @return ConfiguratorInterface
2016-09-25 01:20:17 -05:00
* @throws FireflyException
2016-06-24 07:24:34 -05:00
*/
private function makeConfigurator(ImportJob $job): ConfiguratorInterface
2016-06-24 07:24:34 -05:00
{
$type = $job->file_type;
$key = sprintf('firefly.import_configurators.%s', $type);
$className = config($key);
if (is_null($className)) {
throw new FireflyException('Cannot find configurator class for this job.');
2016-09-25 01:20:17 -05:00
}
$configurator = new $className($job);
2016-06-24 07:24:34 -05:00
return $configurator;
2016-06-24 07:24:34 -05:00
}
}