123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 |
- <?php
- /**
- * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- * @link https://cakephp.org CakePHP(tm) Project
- * @since 3.0.7
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Datasource;
- use InvalidArgumentException;
- /**
- * Contains logic for storing and checking rules on entities
- *
- * RulesCheckers are used by Table classes to ensure that the
- * current entity state satisfies the application logic and business rules.
- *
- * RulesCheckers afford different rules to be applied in the create and update
- * scenario.
- *
- * ### Adding rules
- *
- * Rules must be callable objects that return true/false depending on whether or
- * not the rule has been satisfied. You can use RulesChecker::add(), RulesChecker::addCreate(),
- * RulesChecker::addUpdate() and RulesChecker::addDelete to add rules to a checker.
- *
- * ### Running checks
- *
- * Generally a Table object will invoke the rules objects, but you can manually
- * invoke the checks by calling RulesChecker::checkCreate(), RulesChecker::checkUpdate() or
- * RulesChecker::checkDelete().
- */
- class RulesChecker
- {
- /**
- * Indicates that the checking rules to apply are those used for creating entities
- *
- * @var string
- */
- const CREATE = 'create';
- /**
- * Indicates that the checking rules to apply are those used for updating entities
- *
- * @var string
- */
- const UPDATE = 'update';
- /**
- * Indicates that the checking rules to apply are those used for deleting entities
- *
- * @var string
- */
- const DELETE = 'delete';
- /**
- * The list of rules to be checked on both create and update operations
- *
- * @var callable[]
- */
- protected $_rules = [];
- /**
- * The list of rules to check during create operations
- *
- * @var callable[]
- */
- protected $_createRules = [];
- /**
- * The list of rules to check during update operations
- *
- * @var callable[]
- */
- protected $_updateRules = [];
- /**
- * The list of rules to check during delete operations
- *
- * @var callable[]
- */
- protected $_deleteRules = [];
- /**
- * List of options to pass to every callable rule
- *
- * @var array
- */
- protected $_options = [];
- /**
- * Whether or not to use I18n functions for translating default error messages
- *
- * @var bool
- */
- protected $_useI18n = false;
- /**
- * Constructor. Takes the options to be passed to all rules.
- *
- * @param array $options The options to pass to every rule
- */
- public function __construct(array $options = [])
- {
- $this->_options = $options;
- $this->_useI18n = function_exists('__d');
- }
- /**
- * Adds a rule that will be applied to the entity both on create and update
- * operations.
- *
- * ### Options
- *
- * The options array accept the following special keys:
- *
- * - `errorField`: The name of the entity field that will be marked as invalid
- * if the rule does not pass.
- * - `message`: The error message to set to `errorField` if the rule does not pass.
- *
- * @param callable $rule A callable function or object that will return whether
- * the entity is valid or not.
- * @param string|null $name The alias for a rule.
- * @param array $options List of extra options to pass to the rule callable as
- * second argument.
- * @return $this
- */
- public function add(callable $rule, $name = null, array $options = [])
- {
- $this->_rules[] = $this->_addError($rule, $name, $options);
- return $this;
- }
- /**
- * Adds a rule that will be applied to the entity on create operations.
- *
- * ### Options
- *
- * The options array accept the following special keys:
- *
- * - `errorField`: The name of the entity field that will be marked as invalid
- * if the rule does not pass.
- * - `message`: The error message to set to `errorField` if the rule does not pass.
- *
- * @param callable $rule A callable function or object that will return whether
- * the entity is valid or not.
- * @param string|null $name The alias for a rule.
- * @param array $options List of extra options to pass to the rule callable as
- * second argument.
- * @return $this
- */
- public function addCreate(callable $rule, $name = null, array $options = [])
- {
- $this->_createRules[] = $this->_addError($rule, $name, $options);
- return $this;
- }
- /**
- * Adds a rule that will be applied to the entity on update operations.
- *
- * ### Options
- *
- * The options array accept the following special keys:
- *
- * - `errorField`: The name of the entity field that will be marked as invalid
- * if the rule does not pass.
- * - `message`: The error message to set to `errorField` if the rule does not pass.
- *
- * @param callable $rule A callable function or object that will return whether
- * the entity is valid or not.
- * @param string|null $name The alias for a rule.
- * @param array $options List of extra options to pass to the rule callable as
- * second argument.
- * @return $this
- */
- public function addUpdate(callable $rule, $name = null, array $options = [])
- {
- $this->_updateRules[] = $this->_addError($rule, $name, $options);
- return $this;
- }
- /**
- * Adds a rule that will be applied to the entity on delete operations.
- *
- * ### Options
- *
- * The options array accept the following special keys:
- *
- * - `errorField`: The name of the entity field that will be marked as invalid
- * if the rule does not pass.
- * - `message`: The error message to set to `errorField` if the rule does not pass.
- *
- * @param callable $rule A callable function or object that will return whether
- * the entity is valid or not.
- * @param string|null $name The alias for a rule.
- * @param array $options List of extra options to pass to the rule callable as
- * second argument.
- * @return $this
- */
- public function addDelete(callable $rule, $name = null, array $options = [])
- {
- $this->_deleteRules[] = $this->_addError($rule, $name, $options);
- return $this;
- }
- /**
- * Runs each of the rules by passing the provided entity and returns true if all
- * of them pass. The rules to be applied are depended on the $mode parameter which
- * can only be RulesChecker::CREATE, RulesChecker::UPDATE or RulesChecker::DELETE
- *
- * @param \Cake\Datasource\EntityInterface $entity The entity to check for validity.
- * @param string $mode Either 'create, 'update' or 'delete'.
- * @param array $options Extra options to pass to checker functions.
- * @return bool
- * @throws \InvalidArgumentException if an invalid mode is passed.
- */
- public function check(EntityInterface $entity, $mode, array $options = [])
- {
- if ($mode === self::CREATE) {
- return $this->checkCreate($entity, $options);
- }
- if ($mode === self::UPDATE) {
- return $this->checkUpdate($entity, $options);
- }
- if ($mode === self::DELETE) {
- return $this->checkDelete($entity, $options);
- }
- throw new InvalidArgumentException('Wrong checking mode: ' . $mode);
- }
- /**
- * Runs each of the rules by passing the provided entity and returns true if all
- * of them pass. The rules selected will be only those specified to be run on 'create'
- *
- * @param \Cake\Datasource\EntityInterface $entity The entity to check for validity.
- * @param array $options Extra options to pass to checker functions.
- * @return bool
- */
- public function checkCreate(EntityInterface $entity, array $options = [])
- {
- return $this->_checkRules($entity, $options, array_merge($this->_rules, $this->_createRules));
- }
- /**
- * Runs each of the rules by passing the provided entity and returns true if all
- * of them pass. The rules selected will be only those specified to be run on 'update'
- *
- * @param \Cake\Datasource\EntityInterface $entity The entity to check for validity.
- * @param array $options Extra options to pass to checker functions.
- * @return bool
- */
- public function checkUpdate(EntityInterface $entity, array $options = [])
- {
- return $this->_checkRules($entity, $options, array_merge($this->_rules, $this->_updateRules));
- }
- /**
- * Runs each of the rules by passing the provided entity and returns true if all
- * of them pass. The rules selected will be only those specified to be run on 'delete'
- *
- * @param \Cake\Datasource\EntityInterface $entity The entity to check for validity.
- * @param array $options Extra options to pass to checker functions.
- * @return bool
- */
- public function checkDelete(EntityInterface $entity, array $options = [])
- {
- return $this->_checkRules($entity, $options, $this->_deleteRules);
- }
- /**
- * Used by top level functions checkDelete, checkCreate and checkUpdate, this function
- * iterates an array containing the rules to be checked and checks them all.
- *
- * @param \Cake\Datasource\EntityInterface $entity The entity to check for validity.
- * @param array $options Extra options to pass to checker functions.
- * @param array $rules The list of rules that must be checked.
- * @return bool
- */
- protected function _checkRules(EntityInterface $entity, array $options = [], array $rules = [])
- {
- $success = true;
- $options += $this->_options;
- foreach ($rules as $rule) {
- $success = $rule($entity, $options) && $success;
- }
- return $success;
- }
- /**
- * Utility method for decorating any callable so that if it returns false, the correct
- * property in the entity is marked as invalid.
- *
- * @param callable $rule The rule to decorate
- * @param string $name The alias for a rule.
- * @param array $options The options containing the error message and field.
- * @return callable
- */
- protected function _addError($rule, $name, $options)
- {
- if (is_array($name)) {
- $options = $name;
- $name = null;
- }
- if (!($rule instanceof RuleInvoker)) {
- $rule = new RuleInvoker($rule, $name, $options);
- } else {
- $rule->setOptions($options)->setName($name);
- }
- return $rule;
- }
- }
|