mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-31 02:36:28 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			268 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * AttachmentHelper.php
 | |
|  * Copyright (c) 2017 thegrumpydictator@gmail.com
 | |
|  *
 | |
|  * This file is part of Firefly III.
 | |
|  *
 | |
|  * Firefly III is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * Firefly III is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace FireflyIII\Helpers\Attachments;
 | |
| 
 | |
| use Crypt;
 | |
| use FireflyIII\Models\Attachment;
 | |
| use Illuminate\Database\Eloquent\Model;
 | |
| use Illuminate\Support\Collection;
 | |
| use Illuminate\Support\MessageBag;
 | |
| use Log;
 | |
| use Storage;
 | |
| use Symfony\Component\HttpFoundation\File\UploadedFile;
 | |
| 
 | |
| /**
 | |
|  * Class AttachmentHelper.
 | |
|  */
 | |
| class AttachmentHelper implements AttachmentHelperInterface
 | |
| {
 | |
|     /** @var Collection */
 | |
|     public $attachments;
 | |
|     /** @var MessageBag */
 | |
|     public $errors;
 | |
|     /** @var MessageBag */
 | |
|     public $messages;
 | |
|     /** @var array */
 | |
|     protected $allowedMimes = [];
 | |
|     /** @var int */
 | |
|     protected $maxUploadSize = 0;
 | |
| 
 | |
|     /** @var \Illuminate\Contracts\Filesystem\Filesystem */
 | |
|     protected $uploadDisk;
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * AttachmentHelper constructor.
 | |
|      */
 | |
|     public function __construct()
 | |
|     {
 | |
|         $this->maxUploadSize = (int)config('firefly.maxUploadSize');
 | |
|         $this->allowedMimes  = (array)config('firefly.allowedMimes');
 | |
|         $this->errors        = new MessageBag;
 | |
|         $this->messages      = new MessageBag;
 | |
|         $this->attachments   = new Collection;
 | |
|         $this->uploadDisk    = Storage::disk('upload');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Attachment $attachment
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getAttachmentLocation(Attachment $attachment): string
 | |
|     {
 | |
|         $path = sprintf('%s%sat-%d.data', storage_path('upload'), DIRECTORY_SEPARATOR, (int)$attachment->id);
 | |
| 
 | |
|         return $path;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getAttachments(): Collection
 | |
|     {
 | |
|         return $this->attachments;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return MessageBag
 | |
|      */
 | |
|     public function getErrors(): MessageBag
 | |
|     {
 | |
|         return $this->errors;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return MessageBag
 | |
|      */
 | |
|     public function getMessages(): MessageBag
 | |
|     {
 | |
|         return $this->messages;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Model      $model
 | |
|      * @param array|null $files
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     public function saveAttachmentsForModel(Model $model, ?array $files): bool
 | |
|     {
 | |
|         Log::debug(sprintf('Now in saveAttachmentsForModel for model %s', \get_class($model)));
 | |
|         if (\is_array($files)) {
 | |
|             Log::debug('$files is an array.');
 | |
|             /** @var UploadedFile $entry */
 | |
|             foreach ($files as $entry) {
 | |
|                 if (null !== $entry) {
 | |
|                     $this->processFile($entry, $model);
 | |
|                 }
 | |
|             }
 | |
|             Log::debug('Done processing uploads.');
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
|         Log::debug('Array of files is not an array. Probably nothing uploaded. Will not store attachments.');
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UploadedFile $file
 | |
|      * @param Model        $model
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     protected function hasFile(UploadedFile $file, Model $model): bool
 | |
|     {
 | |
|         $md5   = md5_file($file->getRealPath());
 | |
|         $name  = $file->getClientOriginalName();
 | |
|         $class = \get_class($model);
 | |
|         $count = $model->user->attachments()->where('md5', $md5)->where('attachable_id', $model->id)->where('attachable_type', $class)->count();
 | |
| 
 | |
|         if ($count > 0) {
 | |
|             $msg = (string)trans('validation.file_already_attached', ['name' => $name]);
 | |
|             $this->errors->add('attachments', $msg);
 | |
|             Log::error($msg);
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UploadedFile $file
 | |
|      * @param Model        $model
 | |
|      *
 | |
|      * @return Attachment
 | |
|      * @throws \Illuminate\Contracts\Encryption\EncryptException
 | |
|      */
 | |
|     protected function processFile(UploadedFile $file, Model $model): Attachment
 | |
|     {
 | |
|         Log::debug('Now in processFile()');
 | |
|         $validation = $this->validateUpload($file, $model);
 | |
|         if (false === $validation) {
 | |
|             return new Attachment;
 | |
|         }
 | |
| 
 | |
|         $attachment = new Attachment; // create Attachment object.
 | |
|         $attachment->user()->associate($model->user);
 | |
|         $attachment->attachable()->associate($model);
 | |
|         $attachment->md5      = md5_file($file->getRealPath());
 | |
|         $attachment->filename = $file->getClientOriginalName();
 | |
|         $attachment->mime     = $file->getMimeType();
 | |
|         $attachment->size     = $file->getSize();
 | |
|         $attachment->uploaded = 0;
 | |
|         $attachment->save();
 | |
|         Log::debug('Created attachment:', $attachment->toArray());
 | |
| 
 | |
|         $fileObject = $file->openFile('r');
 | |
|         $fileObject->rewind();
 | |
|         $content   = $fileObject->fread($file->getSize());
 | |
|         $encrypted = Crypt::encrypt($content);
 | |
|         Log::debug(sprintf('Full file length is %d and upload size is %d.', \strlen($content), $file->getSize()));
 | |
|         Log::debug(sprintf('Encrypted content is %d', \strlen($encrypted)));
 | |
| 
 | |
|         // store it:
 | |
|         $this->uploadDisk->put($attachment->fileName(), $encrypted);
 | |
|         $attachment->uploaded = 1; // update attachment
 | |
|         $attachment->save();
 | |
|         $this->attachments->push($attachment);
 | |
| 
 | |
|         $name = e($file->getClientOriginalName()); // add message:
 | |
|         $msg  = (string)trans('validation.file_attached', ['name' => $name]);
 | |
|         $this->messages->add('attachments', $msg);
 | |
| 
 | |
|         // return it.
 | |
|         return $attachment;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UploadedFile $file
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     protected function validMime(UploadedFile $file): bool
 | |
|     {
 | |
|         Log::debug('Now in validMime()');
 | |
|         $mime = e($file->getMimeType());
 | |
|         $name = e($file->getClientOriginalName());
 | |
|         Log::debug(sprintf('Name is %s, and mime is %s', $name, $mime));
 | |
|         Log::debug('Valid mimes are', $this->allowedMimes);
 | |
| 
 | |
|         if (!\in_array($mime, $this->allowedMimes)) {
 | |
|             $msg = (string)trans('validation.file_invalid_mime', ['name' => $name, 'mime' => $mime]);
 | |
|             $this->errors->add('attachments', $msg);
 | |
|             Log::error($msg);
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @codeCoverageIgnore
 | |
|      *
 | |
|      * @param UploadedFile $file
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     protected function validSize(UploadedFile $file): bool
 | |
|     {
 | |
|         $size = $file->getSize();
 | |
|         $name = e($file->getClientOriginalName());
 | |
|         if ($size > $this->maxUploadSize) {
 | |
|             $msg = (string)trans('validation.file_too_large', ['name' => $name]);
 | |
|             $this->errors->add('attachments', $msg);
 | |
|             Log::error($msg);
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UploadedFile $file
 | |
|      * @param Model        $model
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     protected function validateUpload(UploadedFile $file, Model $model): bool
 | |
|     {
 | |
|         Log::debug('Now in validateUpload()');
 | |
|         if (!$this->validMime($file)) {
 | |
|             return false;
 | |
|         }
 | |
|         if (!$this->validSize($file)) {
 | |
|             return false; // @codeCoverageIgnore
 | |
|         }
 | |
|         if ($this->hasFile($file, $model)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| }
 |