. */ declare(strict_types=1); namespace FireflyIII\Repositories\ImportJob; use Crypt; use FireflyIII\Exceptions\FireflyException; use FireflyIII\Models\ImportJob; use FireflyIII\Repositories\User\UserRepositoryInterface; use FireflyIII\User; use Illuminate\Support\Str; use Log; use SplFileObject; use Storage; use Symfony\Component\HttpFoundation\File\UploadedFile; /** * Class ImportJobRepository. */ class ImportJobRepository implements ImportJobRepositoryInterface { /** @var User */ private $user; /** * @param string $type * * @return ImportJob * * @throws FireflyException */ public function create(string $type): ImportJob { $count = 0; $type = strtolower($type); while ($count < 30) { $key = Str::random(12); $existing = $this->findByKey($key); if (null === $existing->id) { $importJob = new ImportJob; $importJob->user()->associate($this->user); $importJob->file_type = $type; $importJob->key = Str::random(12); $importJob->status = 'new'; $importJob->configuration = []; $importJob->extended_status = [ 'steps' => 0, 'done' => 0, 'tag' => 0, 'errors' => [], ]; $importJob->save(); // breaks the loop: return $importJob; } ++$count; } throw new FireflyException('Could not create an import job with a unique key after 30 tries.'); } /** * @param string $key * * @return ImportJob */ public function findByKey(string $key): ImportJob { /** @var ImportJob $result */ $result = $this->user->importJobs()->where('key', $key)->first(['import_jobs.*']); if (null === $result) { return new ImportJob; } return $result; } /** * Return configuration of job. * * @param ImportJob $job * * @return array */ public function getConfiguration(ImportJob $job): array { $config = $job->configuration; if (is_array($config)) { return $config; } return []; } /** * @param ImportJob $job * @param UploadedFile $file * * @return bool */ public function processConfiguration(ImportJob $job, UploadedFile $file): bool { /** @var UserRepositoryInterface $repository */ $repository = app(UserRepositoryInterface::class); // demo user's configuration upload is ignored completely. if (!$repository->hasRole($this->user, 'demo')) { Log::debug( 'Uploaded configuration file', ['name' => $file->getClientOriginalName(), 'size' => $file->getSize(), 'mime' => $file->getClientMimeType()] ); $configFileObject = new SplFileObject($file->getRealPath()); $configRaw = $configFileObject->fread($configFileObject->getSize()); $configuration = json_decode($configRaw, true); if (null !== $configuration && is_array($configuration)) { Log::debug('Found configuration', $configuration); $this->setConfiguration($job, $configuration); } } return true; } /** * @param ImportJob $job * @param null|UploadedFile $file * * @return bool * * @throws \Illuminate\Contracts\Filesystem\FileNotFoundException */ public function processFile(ImportJob $job, ?UploadedFile $file): bool { if (is_null($file)) { return false; } /** @var UserRepositoryInterface $repository */ $repository = app(UserRepositoryInterface::class); $newName = sprintf('%s.upload', $job->key); $uploaded = new SplFileObject($file->getRealPath()); $content = trim($uploaded->fread($uploaded->getSize())); $contentEncrypted = Crypt::encrypt($content); $disk = Storage::disk('upload'); // user is demo user, replace upload with prepared file. if ($repository->hasRole($this->user, 'demo')) { $stubsDisk = Storage::disk('stubs'); $content = $stubsDisk->get('demo-import.csv'); $contentEncrypted = Crypt::encrypt($content); $disk->put($newName, $contentEncrypted); Log::debug('Replaced upload with demo file.'); // also set up prepared configuration. $configuration = json_decode($stubsDisk->get('demo-configuration.json'), true); $this->setConfiguration($job, $configuration); Log::debug('Set configuration for demo user', $configuration); } if (!$repository->hasRole($this->user, 'demo')) { // user is not demo, process original upload: $disk->put($newName, $contentEncrypted); Log::debug('Uploaded file', ['name' => $file->getClientOriginalName(), 'size' => $file->getSize(), 'mime' => $file->getClientMimeType()]); } return true; } /** * @param ImportJob $job * @param array $configuration * * @return ImportJob */ public function setConfiguration(ImportJob $job, array $configuration): ImportJob { Log::debug(sprintf('Incoming config for job "%s" is: ', $job->key), $configuration); $currentConfig = $job->configuration; $newConfig = array_merge($currentConfig, $configuration); $job->configuration = $newConfig; $job->save(); Log::debug(sprintf('Set config of job "%s" to: ', $job->key), $newConfig); return $job; } /** * @param User $user */ public function setUser(User $user) { $this->user = $user; } /** * @param ImportJob $job * @param string $status * * @return ImportJob */ public function updateStatus(ImportJob $job, string $status): ImportJob { $job->status = $status; $job->save(); return $job; } /** * Return import file content. * * @param ImportJob $job * * @return string * @throws \Illuminate\Contracts\Filesystem\FileNotFoundException */ public function uploadFileContents(ImportJob $job): string { return $job->uploadFileContents(); } }