. */ declare(strict_types=1); namespace FireflyIII\Models; use Crypt; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\SoftDeletes; use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; use Watson\Validating\ValidatingTrait; /** * Class Tag. */ class Tag extends Model { use ValidatingTrait, SoftDeletes; /** * The attributes that should be casted to native types. * * @var array */ protected $casts = [ 'created_at' => 'datetime', 'updated_at' => 'datetime', 'deleted_at' => 'datetime', 'date' => 'date', 'zoomLevel' => 'int', ]; /** @var array */ protected $dates = ['date']; /** @var array */ protected $fillable = ['user_id', 'tag', 'date', 'description', 'longitude', 'latitude', 'zoomLevel', 'tagMode']; /** @var array */ protected $rules = ['tag' => 'required|between:1,200']; /** * @param array $fields * * @return Tag|null */ public static function firstOrCreateEncrypted(array $fields) { // everything but the tag: unset($fields['tagMode']); $search = $fields; unset($search['tag']); $query = self::orderBy('id'); foreach ($search as $name => $value) { $query->where($name, $value); } $set = $query->get(['tags.*']); /** @var Tag $tag */ foreach ($set as $tag) { if ($tag->tag === $fields['tag']) { return $tag; } } // create it! $fields['tagMode'] = 'nothing'; $fields['description'] = $fields['description'] ?? ''; $tag = self::create($fields); return $tag; } /** * @param string $value * * @return Tag */ public static function routeBinder(string $value): Tag { if (auth()->check()) { $tagId = intval($value); $tag = auth()->user()->tags()->find($tagId); if (!is_null($tag)) { return $tag; } } throw new NotFoundHttpException; } /** * @param Tag $tag * * @return string * * @throws \FireflyIII\Exceptions\FireflyException */ public static function tagSum(Tag $tag): string { $sum = '0'; /** @var TransactionJournal $journal */ foreach ($tag->transactionjournals as $journal) { bcadd($sum, $journal->amount()); } return $sum; } /** * @codeCoverageIgnore * * @param $value * * @return string */ public function getDescriptionAttribute($value) { if (null === $value) { return $value; } return Crypt::decrypt($value); } /** * @codeCoverageIgnore * * @param $value * * @return string */ public function getTagAttribute($value) { if (null === $value) { return null; } return Crypt::decrypt($value); } /** * Save the model to the database. * * @param array $options * * @return bool */ public function save(array $options = []) { foreach ($this->transactionJournals()->get() as $journal) { $count = $journal->tags()->count(); $journal->tag_count = $count; $journal->save(); } return parent::save($options); } /** * @codeCoverageIgnore * * @param $value */ public function setDescriptionAttribute($value) { $this->attributes['description'] = Crypt::encrypt($value); } /** * @codeCoverageIgnore * * @param $value */ public function setTagAttribute($value) { $this->attributes['tag'] = Crypt::encrypt($value); } /** * @codeCoverageIgnore * @return \Illuminate\Database\Eloquent\Relations\BelongsToMany */ public function transactionJournals() { return $this->belongsToMany('FireflyIII\Models\TransactionJournal'); } /** * @codeCoverageIgnore * @return \Illuminate\Database\Eloquent\Relations\BelongsTo */ public function user() { return $this->belongsTo('FireflyIII\User'); } }