false, self::BREAK_CHAIN => false, self::ESCAPE_FILTER => 'HtmlEntities', self::MISSING_MESSAGE => "Field '%field%' is required by rule '%rule%', but the field is missing", self::NOT_EMPTY_MESSAGE => "You must give a non-empty value for field '%field%'", self::PRESENCE => self::PRESENCE_OPTIONAL ); /** * @var boolean Set to False initially, this is set to True after the * input data have been processed. Reset to False in setData() method. */ protected $_processed = false; /** * @param array $filters * @param array $validators * @param array $data OPTIONAL * @param array $options OPTIONAL */ public function __construct($filterRules, $validatorRules, array $data = null, array $options = null) { if ($options) { $this->setOptions($options); } $this->_filterRules = (array) $filterRules; $this->_validatorRules = (array) $validatorRules; if ($data) { $this->setData($data); } } /** * @param mixed $namespaces * @return void */ public function addNamespace($namespaces) { foreach((array) $namespaces as $namespace) { $this->_userNamespaces[] = $namespace; } $this->_namespaces = array_merge( $this->_userNamespaces, array('Zend_Filter', 'Zend_Validate') ); } /** * @return array */ public function getMessages() { $this->_process(); return array_merge($this->_invalidMessages, $this->_missingFields); } /** * @return array */ public function getErrors() { $this->_process(); return $this->_invalidErrors; } /** * @return array */ public function getInvalid() { $this->_process(); return $this->_invalidMessages; } /** * @return array */ public function getMissing() { $this->_process(); return $this->_missingFields; } /** * @return array */ public function getUnknown() { $this->_process(); return $this->_unknownFields; } /** * @param string $fieldName OPTIONAL * @return mixed */ public function getEscaped($fieldName = null) { $this->_process(); $escapeFilter = $this->_getDefaultEscapeFilter(); if ($fieldName === null) { return $this->_escapeRecursive($this->_validFields); } if (array_key_exists($fieldName, $this->_validFields)) { return $this->_escapeRecursive($this->_validFields[$fieldName]); } return null; } /** * @param mixed $value * @return mixed */ protected function _escapeRecursive($data) { if (!is_array($data)) { return $this->_getDefaultEscapeFilter()->filter($data); } foreach ($data as &$element) { $element = $this->_escapeRecursive($element); } return $data; } /** * @param string $fieldName OPTIONAL * @return mixed */ public function getUnescaped($fieldName = null) { $this->_process(); if ($fieldName === null) { return $this->_validFields; } if (array_key_exists($fieldName, $this->_validFields)) { return $this->_validFields[$fieldName]; } return null; } /** * @param string $fieldName * @return mixed */ public function __get($fieldName) { return $this->getEscaped($fieldName); } /** * @return boolean */ public function hasInvalid() { $this->_process(); return !(empty($this->_invalidMessages)); } /** * @return boolean */ public function hasMissing() { $this->_process(); return !(empty($this->_missingFields)); } /** * @return boolean */ public function hasUnknown() { $this->_process(); return !(empty($this->_unknownFields)); } /** * @return boolean */ public function hasValid() { $this->_process(); return !(empty($this->_validFields)); } /** * @param string $fieldName * @return boolean */ public function isValid($fieldName = null) { $this->_process(); if ($fieldName === null) { return !($this->hasMissing() || $this->hasInvalid()); } return array_key_exists($fieldName, $this->_validFields); } /** * @param string $fieldName * @return boolean */ public function __isset($fieldName) { $this->_process(); return isset($this->_validFields[$fieldName]); } /** * @return void * @throw Zend_Filter_Exception */ public function process() { $this->_process(); if ($this->hasInvalid()) { require_once 'Zend/Filter/Exception.php'; throw new Zend_Filter_Exception("Input has invalid fields"); } if ($this->hasMissing()) { require_once 'Zend/Filter/Exception.php'; throw new Zend_Filter_Exception("Input has missing fields"); } } /** * @param array $data * @return void */ public function setData(array $data) { $this->_data = $data; /** * Reset to initial state */ $this->_validFields = array(); $this->_invalidMessages = array(); $this->_invalidErrors = array(); $this->_missingFields = array(); $this->_unknownFields = array(); $this->_processed = false; } /** * @param mixed $escapeFilter * @return void */ public function setDefaultEscapeFilter($escapeFilter) { if (is_string($escapeFilter) || is_array($escapeFilter)) { $escapeFilter = $this->_getFilter($escapeFilter); } if (!$escapeFilter instanceof Zend_Filter_Interface) { require_once 'Zend/Filter/Exception.php'; throw new Zend_Filter_Exception('Escape filter specified does not implement Zend_Filter_Interface'); } $this->_defaultEscapeFilter = $escapeFilter; return $escapeFilter; } /** * @param array $options * @return void * @throws Zend_Filter_Exception if an unknown option is given */ public function setOptions(array $options) { foreach ($options as $option => $value) { switch ($option) { case self::ESCAPE_FILTER: $this->setDefaultEscapeFilter($value); break; case self::NAMESPACE: $this->addNamespace($value); break; case self::ALLOW_EMPTY: case self::BREAK_CHAIN: case self::MISSING_MESSAGE: case self::NOT_EMPTY_MESSAGE: case self::PRESENCE: $this->_defaults[$option] = $value; break; default: require_once 'Zend/Filter/Exception.php'; throw new Zend_Filter_Exception("Unknown option '$option'"); break; } } } /* * Protected methods */ /** * @return void */ protected function _filter() { foreach ($this->_filterRules as $ruleName => &$filterRule) { /** * Make sure we have an array representing this filter chain. * Don't typecast to (array) because it might be a Zend_Filter object */ if (!is_array($filterRule)) { $filterRule = array($filterRule); } /** * Filters are indexed by integer, metacommands are indexed by string. * Pick out the filters. */ $filterList = array(); foreach ($filterRule as $key => $value) { if (is_int($key)) { $filterList[] = $value; } } /** * Use defaults for filter metacommands. */ $filterRule[self::RULE] = $ruleName; if (!isset($filterRule[self::FIELDS])) { $filterRule[self::FIELDS] = $ruleName; } /** * Load all the filter classes and add them to the chain. */ if (!isset($filterRule[self::FILTER_CHAIN])) { $filterRule[self::FILTER_CHAIN] = new Zend_Filter(); foreach ($filterList as $filter) { if (is_string($filter) || is_array($filter)) { $filter = $this->_getFilter($filter); } $filterRule[self::FILTER_CHAIN]->addFilter($filter); } } /** * If the ruleName is the special wildcard rule, * then apply the filter chain to all input data. * Else just process the field named by the rule. */ if ($ruleName == self::RULE_WILDCARD) { foreach (array_keys($this->_data) as $field) { $this->_filterRule(array_merge($filterRule, array(self::FIELDS => $field))); } } else { $this->_filterRule($filterRule); } } } /** * @param array $filterRule * @return void */ protected function _filterRule(array $filterRule) { $field = $filterRule[self::FIELDS]; if (!array_key_exists($field, $this->_data)) { return; } if (is_array($this->_data[$field])) { foreach ($this->_data[$field] as $key => $value) { $this->_data[$field][$key] = $filterRule[self::FILTER_CHAIN]->filter($value); } } else { $this->_data[$field] = $filterRule[self::FILTER_CHAIN]->filter($this->_data[$field]); } } /** * @return Zend_Filter_Interface */ protected function _getDefaultEscapeFilter() { if ($this->_defaultEscapeFilter !== null) { return $this->_defaultEscapeFilter; } return $this->setDefaultEscapeFilter($this->_defaults[self::ESCAPE_FILTER]); } /** * @param string $rule * @param string $field * @return string */ protected function _getMissingMessage($rule, $field) { $message = $this->_defaults[self::MISSING_MESSAGE]; $message = str_replace('%rule%', $rule, $message); $message = str_replace('%field%', $field, $message); return $message; } /** * @return string */ protected function _getNotEmptyMessage($rule, $field) { $message = $this->_defaults[self::NOT_EMPTY_MESSAGE]; $message = str_replace('%rule%', $rule, $message); $message = str_replace('%field%', $field, $message); return $message; } /** * @return void */ protected function _process() { if ($this->_processed === false) { $this->_filter(); $this->_validate(); $this->_processed = true; } } /** * @return void */ protected function _validate() { /** * Special case: if there are no validators, treat all fields as valid. */ if (!$this->_validatorRules) { $this->_validFields = $this->_data; $this->_data = array(); return; } foreach ($this->_validatorRules as $ruleName => &$validatorRule) { /** * Make sure we have an array representing this validator chain. * Don't typecast to (array) because it might be a Zend_Validate object */ if (!is_array($validatorRule)) { $validatorRule = array($validatorRule); } /** * Validators are indexed by integer, metacommands are indexed by string. * Pick out the validators. */ $validatorList = array(); foreach ($validatorRule as $key => $value) { if (is_int($key)) { $validatorList[] = $value; } } /** * Use defaults for validation metacommands. */ $validatorRule[self::RULE] = $ruleName; if (!isset($validatorRule[self::FIELDS])) { $validatorRule[self::FIELDS] = $ruleName; } if (!isset($validatorRule[self::BREAK_CHAIN])) { $validatorRule[self::BREAK_CHAIN] = $this->_defaults[self::BREAK_CHAIN]; } if (!isset($validatorRule[self::PRESENCE])) { $validatorRule[self::PRESENCE] = $this->_defaults[self::PRESENCE]; } if (!isset($validatorRule[self::ALLOW_EMPTY])) { $validatorRule[self::ALLOW_EMPTY] = $this->_defaults[self::ALLOW_EMPTY]; } if (!isset($validatorRule[self::MESSAGES])) { $validatorRule[self::MESSAGES] = array(); } else if (!is_array($validatorRule[self::MESSAGES])) { $validatorRule[self::MESSAGES] = array($validatorRule[self::MESSAGES]); } else if (!array_intersect_key($validatorList, $validatorRule[self::MESSAGES])) { $validatorRule[self::MESSAGES] = array($validatorRule[self::MESSAGES]); } /** * Load all the validator classes and add them to the chain. */ if (!isset($validatorRule[self::VALIDATOR_CHAIN])) { $validatorRule[self::VALIDATOR_CHAIN] = new Zend_Validate(); $i = 0; foreach ($validatorList as $validator) { if (is_string($validator) || is_array($validator)) { $validator = $this->_getValidator($validator); } if (isset($validatorRule[self::MESSAGES][$i])) { $value = $validatorRule[self::MESSAGES][$i]; if (is_array($value)) { $validator->setMessages($value); } else { $validator->setMessage($value); } } $validatorRule[self::VALIDATOR_CHAIN]->addValidator($validator, $validatorRule[self::BREAK_CHAIN]); ++$i; } $validatorRule[self::VALIDATOR_CHAIN_COUNT] = $i; } /** * If the ruleName is the special wildcard rule, * then apply the validator chain to all input data. * Else just process the field named by the rule. */ if ($ruleName == self::RULE_WILDCARD) { foreach (array_keys($this->_data) as $field) { $this->_validateRule(array_merge($validatorRule, array(self::FIELDS => $field))); } } else { $this->_validateRule($validatorRule); } } /** * Unset fields in $_data that have been added to other arrays. * We have to wait until all rules have been processed because * a given field may be referenced by multiple rules. */ foreach (array_merge(array_keys($this->_missingFields), array_keys($this->_invalidMessages)) as $rule) { foreach ((array) $this->_validatorRules[$rule][self::FIELDS] as $field) { unset($this->_data[$field]); } } foreach ($this->_validFields as $field => $value) { unset($this->_data[$field]); } /** * Anything left over in $_data is an unknown field. */ $this->_unknownFields = $this->_data; } /** * @param array $validatorRule * @return void */ protected function _validateRule(array $validatorRule) { /** * Get one or more data values from input, and check for missing fields. * Apply defaults if fields are missing. */ $data = array(); foreach ((array) $validatorRule[self::FIELDS] as $field) { if (array_key_exists($field, $this->_data)) { $data[$field] = $this->_data[$field]; } else if (array_key_exists(self::DEFAULT_VALUE, $validatorRule)) { if (is_array($validatorRule[self::DEFAULT_VALUE])) { $key = array_search($field, $validatorRule[self::FIELDS]); if (array_key_exists($key, $validatorRule[self::DEFAULT_VALUE])) { $data[$field] = $validatorRule[self::DEFAULT_VALUE][$key]; } } else { $data[$field] = $validatorRule[self::DEFAULT_VALUE]; } } else if ($validatorRule[self::PRESENCE] == self::PRESENCE_REQUIRED) { $this->_missingFields[$validatorRule[self::RULE]][] = $this->_getMissingMessage($validatorRule[self::RULE], $field); } } /** * If any required fields are missing, break the loop. */ if (isset($this->_missingFields[$validatorRule[self::RULE]]) && count($this->_missingFields[$validatorRule[self::RULE]]) > 0) { return; } /** * Evaluate the inputs against the validator chain. */ if (count((array) $validatorRule[self::FIELDS]) > 1) { if (!$validatorRule[self::VALIDATOR_CHAIN]->isValid($data)) { $this->_invalidMessages[$validatorRule[self::RULE]] = $validatorRule[self::VALIDATOR_CHAIN]->getMessages(); $this->_invalidErrors[$validatorRule[self::RULE]] = $validatorRule[self::VALIDATOR_CHAIN]->getErrors(); return; } } else { $failed = false; foreach ($data as $fieldKey => $field) { if (!is_array($field)) { $field = array($field); } foreach ($field as $value) { if (empty($value)) { if ($validatorRule[self::ALLOW_EMPTY] == true) { continue; } if ($validatorRule[self::VALIDATOR_CHAIN_COUNT] == 0) { $notEmptyValidator = $this->_getValidator('NotEmpty'); $notEmptyValidator->setMessage($this->_getNotEmptyMessage($validatorRule[self::RULE], $fieldKey)); $validatorRule[self::VALIDATOR_CHAIN]->addValidator($notEmptyValidator); } } if (!$validatorRule[self::VALIDATOR_CHAIN]->isValid($value)) { $this->_invalidMessages[$validatorRule[self::RULE]] = $validatorRule[self::VALIDATOR_CHAIN]->getMessages(); $this->_invalidErrors[$validatorRule[self::RULE]] = $validatorRule[self::VALIDATOR_CHAIN]->getErrors(); unset($this->_validFields[$fieldKey]); $failed = true; if ($validatorRule[self::BREAK_CHAIN]) { return; } } } } if ($failed) { return; } } /** * If we got this far, the inputs for this rule pass validation. */ foreach ((array) $validatorRule[self::FIELDS] as $field) { if (array_key_exists($field, $data)) { $this->_validFields[$field] = $data[$field]; } } } /** * @param mixed $classBaseName * @return Zend_Filter_Interface */ protected function _getFilter($classBaseName) { return $this->_getFilterOrValidator('Zend_Filter_Interface', $classBaseName); } /** * @param mixed $classBaseName * @return Zend_Validate_Interface */ protected function _getValidator($classBaseName) { return $this->_getFilterOrValidator('Zend_Validate_Interface', $classBaseName); } /** * @param string $interface * @param mixed $classBaseName * @return mixed object implementing Zend_Filter_Interface or Zend_Validate_Interface * @throws Zend_Filter_Exception */ protected function _getFilterOrValidator($interface, $classBaseName) { $args = array(); if (is_array($classBaseName)) { $args = $classBaseName; $classBaseName = array_shift($args); } foreach ($this->_namespaces as $namespace) { $className = $namespace . '_' . ucfirst($classBaseName); try { Zend_Loader::loadClass($className); $class = new ReflectionClass($className); if ($class->implementsInterface((string) $interface)) { if ($class->hasMethod('__construct')) { $object = $class->newInstanceArgs($args); } else { $object = $class->newInstance(); } return $object; } } catch (Zend_Exception $e) { // fallthrough and continue } } if (!class_exists($className, false)) { $msg = "Unable to find the implementation of the '$classBaseName' class"; } else { $msg = "Class based on basename '$classBaseName' must implement the '$interface' interface"; } /** * @see Zend_Filter_Exception */ require_once 'Zend/Filter/Exception.php'; throw new Zend_Filter_Exception($msg); } }