make(Factory::class); static::saving(function (Model $model) { if (! $model->validate()) { throw new DataValidationException($model->getValidator()); } return true; }); } /** * Set the model to skip validation when saving. * * @return $this */ public function skipValidation() { $this->skipValidation = true; return $this; } /** * Returns the validator instance used by this model. * * @return \Illuminate\Validation\Validator */ public function getValidator(): Validator { $rules = $this->getKey() ? static::getRulesForUpdate($this) : static::getRules(); return $this->validator ?: $this->validator = static::$validatorFactory->make( [], $rules, [], [] ); } /** * Returns the rules associated with this model. * * @return array */ public static function getRules() { $rules = static::$validationRules; foreach ($rules as $key => &$rule) { $rule = is_array($rule) ? $rule : explode('|', $rule); } return $rules; } /** * Returns the rules associated with the model, specifically for updating the given model * rather than just creating it. * * @param \Illuminate\Database\Eloquent\Model|int|string $id * @param string $primaryKey * @return array */ public static function getRulesForUpdate($id, string $primaryKey = 'id') { if ($id instanceof Model) { [$primaryKey, $id] = [$id->getKeyName(), $id->getKey()]; } $rules = static::getRules(); foreach ($rules as $key => &$data) { // For each rule in a given field, iterate over it and confirm if the rule // is one for a unique field. If that is the case, append the ID of the current // working model so we don't run into errors due to the way that field validation // works. foreach ($data as &$datum) { if (! is_string($datum) || ! Str::startsWith($datum, 'unique')) { continue; } [, $args] = explode(':', $datum); $args = explode(',', $args); $datum = Rule::unique($args[0], $args[1] ?? $key)->ignore($id, $primaryKey)->__toString(); } } return $rules; } /** * Determines if the model is in a valid state or not. * * @return bool */ public function validate() { if ($this->skipValidation) { return true; } return $this->getValidator()->setData( // Trying to do self::toArray() here will leave out keys based on the whitelist/blacklist // for that model. Doing this will return all of the attributes in a format that can // properly be validated. $this->addCastAttributesToArray( $this->getAttributes(), $this->getMutatedAttributes() ) )->passes(); } /** * Return a timestamp as DateTime object. * * @param mixed $value * @return \Illuminate\Support\Carbon|\Carbon\CarbonImmutable */ protected function asDateTime($value) { if (! $this->immutableDates) { return parent::asDateTime($value); } return parent::asDateTime($value)->toImmutable(); } }