* @copyright 2007-2011 PrestaShop SA : 6 rue lacepede, 75005 PARIS * @version Release: $Revision: 9079 $ * @license http://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0) * International Registered Trademark & Property of PrestaShop SA **/ /** * This class works with the Twenga class to control params * used for Twenga API method. * * This class allow to : * - set fields with validators, the length of value, if field is required, * - check each fields if they saved according to params (see above), * - compare an array of values to check if required fields are set. * - use the twenga API method and throw exceptions if error or exceptions occurred. * - Transform the xml response into PHP values. * @version 1.3 */ abstract class TwengaFields { /** * This var must be filled by inherited constructor. * * Each item must look like : * item[(string)'name-of-the-field'] = array(#param1, #param2, #param3); * #param1 : (int) length of the field 0 for no restrictions * #param2 : (array) possibles validators. A validator could be a * - basic php function, * - Validate class method, * - Subclass method * #param3 : [optionnal](boolean) if is required or not. * @var array which saved possible fields with their specification * @see TwengaFields::checkFields() AND TwengaFields::checkFieldAttributs() */ protected $fields; /** * @var string Need to save the class name for used with validators * @see TwengaFields::isValidate(); */ protected $className; /** * @var array which save each required fields * @see TwengaFields::checkFields() for know how it's filled */ protected $requiredFields; /** * @var boolean */ protected $noRequiredFields = false; /** * @var array Each item is value for field which are defined by subclass. * Key of the item must be the name of the field. */ protected $params; /** * Constructor control each fields filled by subclass. * /!\ must be used by subclass after filled TwengaFields::$fields. * @see TwengaFields::checkFields() * @throws TwengaFieldsException throwing by TwengaFields::checkFields() */ public function __construct() { $this->className = get_class($this); try { $this->checkFields(); } catch (TwengaFieldsException $e) { throw $e; } } /** * Check fields attributs. * @throws TwengaFieldsException if TwengaFields::$fields was not filled. * @throws TwengaFieldsException throwing by TwengaFields::checkFieldAttributs() */ private function checkFields() { if (!is_array($this->fields)) throw new TwengaFieldsException('Some fields must be set'); // if $noRequiredFields is set to false, means that we don't need to check fields. if (!is_array($this->requiredFields) AND $this->noRequiredFields === false) { $this->requiredFields = array(); foreach ($this->fields as $name=>$attributs) { try { $this->checkFieldAttributs($name, $attributs); } catch (TwengaFieldsException $e) { throw $e; } } if (empty($this->requiredFields)) $this->noRequiredFields = true; } } /** * Check attributs validity and in case of field is required, * save it in (array)TwengaFields::$requiredFields. * @param string $name of the field * @param array field attributs * @throws TwengaFieldsException if an attribut of a field is not set correctly * (much more debugging rather than production). */ private function checkFieldAttributs ($name, $field_attributs) { // attribut at key 0 is the length of the required value if (!Validate::isInt($field_attributs[0])) throw new TwengaFieldsException(Tools::displayError('To add a field, you have to set a maximum length for checking the value. Error occurred for the value').' : '.$name); // attribut at key 1 is an array list of the validator if (!is_array($field_attributs[1])) throw new TwengaFieldsException(Tools::displayError('To add a field, you have to set validators for checking value. Error occurred for the value').' : '.$name); // Check if validators setted are valid. foreach ($field_attributs[1] as $validator) { if (!function_exists($validator) && !method_exists('Validate', $validator) && !method_exists($this->className, $validator)) throw new TwengaFieldsException (Tools::displayError('The Validator').' '.$validator.' '.Tools::displayError('does\'nt exist')); } // attribut at key 2 means that this fields is required or not if (isset($field_attributs[2]) AND Validate::isBool($field_attributs[2])) $this->requiredFields[$name] = false; } /** * Get a field by his name and return his atribut to check his validity. * @param string $name * @throws TwengaFieldsException if a TwengaFields::$field is an array, * @throws TwengaFieldsException if the choosen field is not set. * @return array field attributs */ public function getField($name) { if (!is_array($this->fields)) throw new TwengaFieldsException(Tools::displayError('To get a field you have to saved some fields !')); if (!key_exists($name, $this->fields)) throw new TwengaFieldsException(Tools::displayError('The field').' '.$name.' '.Tools::displayError('doesn\'t exist.')); return $this->fields[$name]; } /** * Check the value of one field by his name * @param string $key is field name to check. * @param string $value is field value. * @return string empty if it's ok, otherwise a string of errors. */ private function isValidate($key, $value) { $fieldValidate = $this->getField($key); $str_return = ''; // check the length if (strlen((string)$value) > $fieldValidate[0] AND $fieldValidate[0] !== 0) return Tools::displayError('Wrong length of the value. Must be set between 1 and ').$fieldValidate[0].'
'."\n"; // check each validators. foreach ($fieldValidate[1] as $validator) { $user_function = function_exists($validator) ? $validator : ''; $user_function = method_exists('Validate', $validator) ? array('Validate', $validator) : $user_function; $user_function = method_exists($this->className, $validator) ? array($this->className, $validator) : $user_function; try { $bool = call_user_func($user_function, $value); if (!$bool) $str_return .= Tools::displayError('Value don\'t respect the validator : ').''.$validator.'
'."\n";; } catch (TwengaFieldsException $e) { $str_return .= $e->getMessage(); } } return $str_return; } /** * Compare an array of values to each required fields. * * The method doesn't work if they are no required fields. * * No return value for this method, if compared array key * and TwengaFields::$requiredFields key match, * the TwengaFields::$requiredFields value become true. * * In this way if false value is found in TwengaFields::$requiredFields array, * it means that compared array is not filled enough. * @see TwengaFields::requiredFieldsAreSet() is the one using TwengaFields::compareFields(). */ private function compareFields() { if (is_array($this->requiredFields) AND !empty($this->requiredFields)) { $arr_compare = array_intersect_key($this->requiredFields, $this->params); foreach ($this->requiredFields as $key=>&$value) $value = key_exists($key, $arr_compare); } } /** * Check if each required fields are set * see TwengaFields::compareFields(). * Else the method throw an error message. * @return boolean if it's ok * @throws TwengaFieldsException in case of a required field missing. */ private function requiredFieldsAreSet() { if (is_array($this->requiredFields) AND !empty($this->requiredFields)) { $fields_not_set = array_keys($this->requiredFields, false); if (!empty($fields_not_set)) throw new TwengaFieldsException (Tools::displayError('Some fields must be set').' : '.implode(', ', $fields_not_set).''); } return true; } /** * Set values params * @param array $params * @return TwengaFields current instance for chainability */ public function setParams($params) { if (empty($params)) throw new TwengaFieldsException(Tools::displayError('Params must be filled')); $this->params = $params; return $this; } /** * Check params validity (length, validators, is required) * for each TwengaFields::$params value. * @see TwengaFields::compareFields() and TwengaFields::requiredFieldsAreSet() * to know required fields checking method. * @see TwengaFields::isValidate() to know the checking behaviour (length and Validators). * @throws TwengaFieldsException if TwengaFields::$params not set before. * @throws TwengaFieldsException with list of not validated values. * Using TwengaFields::isValidate() method to work. * @throws TwengaFieldsException occurred while compared * TwengaFields::$params and TwengaFields::$fields * * to check if required fields are filled. */ public function checkParams() { if (!is_array($this->params) || empty($this->params)) throw new TwengaFieldsException(Tools::displayError('Params must be setted before check it')); try { $this->compareFields(); $this->requiredFieldsAreSet(); } catch (TwengaFieldsException $e) { throw $e; } $str_message = ''; foreach ($this->params as $key=>$value) { $validate = $this->isValidate($key, $value); if ($validate !== '') $str_message .= Tools::displayError('The field').' '.$key.' '.Tools::displayError('is a wrong value, see details :').'
'."\n".$validate.''; } if ($str_message !== '') throw new TwengaFieldsException($str_message); } /** * @return array TwengaFields::$params */ public function getParams() { return $this->params; } /** * Get translation for a given module text * * @param string $string String to translate * @return string Translation */ public function l($string, $specific = false) { global $_MODULES, $_MODULE, $cookie; $id_lang = (!isset($cookie) OR !is_object($cookie)) ? (int)(Configuration::get('PS_LANG_DEFAULT')) : (int)($cookie->id_lang); $file = _PS_MODULE_DIR_.$this->name.'/'.Language::getIsoById($id_lang).'.php'; if (Tools::file_exists_cache($file) AND include_once($file)) $_MODULES = !empty($_MODULES) ? array_merge($_MODULES, $_MODULE) : $_MODULE; if (!is_array($_MODULES)) return (str_replace('"', '"', $string)); $source = Tools::strtolower($specific ? $specific : get_class($this)); $string2 = str_replace('\'', '\\\'', $string); $currentKey = '<{'.$this->name.'}'._THEME_NAME_.'>'.$source.'_'.md5($string2); $defaultKey = '<{'.$this->name.'}prestashop>'.$source.'_'.md5($string2); if (key_exists($currentKey, $_MODULES)) $ret = stripslashes($_MODULES[$currentKey]); elseif (key_exists($defaultKey, $_MODULES)) $ret = stripslashes($_MODULES[$defaultKey]); else $ret = $string; return str_replace('"', '"', $ret); } } class TwengaFieldsException extends Exception {}