2015-07-04 23:18:02 -05:00
|
|
|
<?php
|
|
|
|
namespace FireflyIII\Helpers\Csv;
|
|
|
|
|
|
|
|
use Auth;
|
|
|
|
use Config;
|
|
|
|
use Crypt;
|
|
|
|
use FireflyIII\Exceptions\FireflyException;
|
2015-07-05 01:45:05 -05:00
|
|
|
use FireflyIII\Helpers\Csv\Mapper\MapperInterface;
|
2015-07-04 23:18:02 -05:00
|
|
|
use League\Csv\Reader;
|
2015-07-05 01:45:05 -05:00
|
|
|
use ReflectionException;
|
2015-07-04 23:18:02 -05:00
|
|
|
use Session;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class Wizard
|
|
|
|
*
|
|
|
|
* @package FireflyIII\Helpers\Csv
|
|
|
|
*/
|
|
|
|
class Wizard implements WizardInterface
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Reader $reader
|
|
|
|
* @param array $map
|
|
|
|
* @param bool $hasHeaders
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getMappableValues($reader, array $map, $hasHeaders)
|
|
|
|
{
|
|
|
|
$values = [];
|
|
|
|
/*
|
|
|
|
* Loop over the CSV and collect mappable data:
|
|
|
|
*/
|
2015-07-07 03:05:11 -05:00
|
|
|
$keys = array_keys($map);
|
2015-07-04 23:18:02 -05:00
|
|
|
foreach ($reader as $index => $row) {
|
2015-07-07 03:05:11 -05:00
|
|
|
if ($this->useRow($hasHeaders, $index)) {
|
2015-07-04 23:18:02 -05:00
|
|
|
// collect all map values
|
2015-07-07 03:05:11 -05:00
|
|
|
|
|
|
|
foreach ($keys as $column) {
|
2015-07-04 23:18:02 -05:00
|
|
|
$values[$column][] = $row[$column];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Make each one unique.
|
|
|
|
*/
|
2015-07-08 06:05:33 -05:00
|
|
|
$values = $this->uniqueRecursive($values);
|
2015-07-04 23:18:02 -05:00
|
|
|
|
|
|
|
return $values;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $roles
|
|
|
|
* @param mixed $map
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function processSelectedMapping(array $roles, $map)
|
|
|
|
{
|
|
|
|
$configRoles = Config::get('csv.roles');
|
|
|
|
$maps = [];
|
|
|
|
|
|
|
|
|
|
|
|
if (is_array($map)) {
|
|
|
|
foreach ($map as $index => $field) {
|
|
|
|
if (isset($roles[$index])) {
|
|
|
|
$name = $roles[$index];
|
|
|
|
if ($configRoles[$name]['mappable']) {
|
|
|
|
$maps[$index] = $name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $maps;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param mixed $input
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function processSelectedRoles($input)
|
|
|
|
{
|
|
|
|
$roles = [];
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Store all rows for each column:
|
|
|
|
*/
|
|
|
|
if (is_array($input)) {
|
|
|
|
foreach ($input as $index => $role) {
|
|
|
|
if ($role != '_ignore') {
|
|
|
|
$roles[$index] = $role;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $roles;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $fields
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function sessionHasValues(array $fields)
|
|
|
|
{
|
|
|
|
foreach ($fields as $field) {
|
|
|
|
if (!Session::has($field)) {
|
2015-07-09 08:20:55 -05:00
|
|
|
Log::error('Session is missing field: ' . $field);
|
2015-07-04 23:18:02 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $map
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
* @throws FireflyException
|
|
|
|
*/
|
|
|
|
public function showOptions(array $map)
|
|
|
|
{
|
2015-07-06 09:52:18 -05:00
|
|
|
$options = [];
|
2015-07-04 23:18:02 -05:00
|
|
|
foreach ($map as $index => $columnRole) {
|
|
|
|
|
2015-07-05 01:45:05 -05:00
|
|
|
$mapper = Config::get('csv.roles.' . $columnRole . '.mapper');
|
|
|
|
if (is_null($mapper)) {
|
|
|
|
throw new FireflyException('Cannot map field of type "' . $columnRole . '".');
|
2015-07-04 23:18:02 -05:00
|
|
|
}
|
2015-07-05 01:45:05 -05:00
|
|
|
$class = 'FireflyIII\Helpers\Csv\Mapper\\' . $mapper;
|
|
|
|
try {
|
|
|
|
/** @var MapperInterface $mapObject */
|
2015-07-07 12:09:45 -05:00
|
|
|
$mapObject = app($class);
|
2015-07-05 01:45:05 -05:00
|
|
|
} catch (ReflectionException $e) {
|
|
|
|
throw new FireflyException('Column "' . $columnRole . '" cannot be mapped because class ' . $mapper . ' does not exist.');
|
|
|
|
}
|
|
|
|
$set = $mapObject->getMap();
|
2015-07-04 23:18:02 -05:00
|
|
|
$options[$index] = $set;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $options;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $path
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function storeCsvFile($path)
|
|
|
|
{
|
|
|
|
$time = str_replace(' ', '-', microtime());
|
|
|
|
$fileName = 'csv-upload-' . Auth::user()->id . '-' . $time . '.csv.encrypted';
|
|
|
|
$fullPath = storage_path('upload') . DIRECTORY_SEPARATOR . $fileName;
|
|
|
|
$content = file_get_contents($path);
|
|
|
|
$contentEncrypted = Crypt::encrypt($content);
|
|
|
|
file_put_contents($fullPath, $contentEncrypted);
|
|
|
|
|
|
|
|
return $fullPath;
|
|
|
|
|
|
|
|
|
|
|
|
}
|
2015-07-07 03:05:11 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param bool $hasHeaders
|
|
|
|
* @param int $index
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function useRow($hasHeaders, $index)
|
|
|
|
{
|
|
|
|
return ($hasHeaders && $index > 1) || !$hasHeaders;
|
|
|
|
}
|
2015-07-08 06:05:33 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $array
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function uniqueRecursive(array $array)
|
|
|
|
{
|
|
|
|
foreach ($array as $column => $found) {
|
|
|
|
$array[$column] = array_unique($found);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $array;
|
|
|
|
}
|
2015-07-04 23:18:02 -05:00
|
|
|
}
|