diff --git a/src/JsonSchema/Validator.php b/src/JsonSchema/Validator.php index eb0f3a8..48e335e 100644 --- a/src/JsonSchema/Validator.php +++ b/src/JsonSchema/Validator.php @@ -8,9 +8,9 @@ All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. - * Neither the name of the Gradua Networks nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. + * Neither the name of the Gradua Networks nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. @@ -21,8 +21,8 @@ Usage: //JsonSchema::$checkMode = JsonSchema::CHECK_MODE_TYPE_CAST; $result = JsonSchema::validate( - $json, - $schema + $json, + $schema ); * */ @@ -30,402 +30,402 @@ namespace JsonSchema; class Validator { - private $errors = array(); - private $formatValidator; + private $errors = array(); + private $formatValidator; - const CHECK_MODE_NORMAL = 1; - const CHECK_MODE_TYPE_CAST = 2; - public $checkMode = self::CHECK_MODE_NORMAL; + const CHECK_MODE_NORMAL = 1; + const CHECK_MODE_TYPE_CAST = 2; + public $checkMode = self::CHECK_MODE_NORMAL; - /** - * Validates a php object against a schema. Both the php object and the schema - * are supposed to be a result of a json_decode call. - * The validation works as defined by the schema proposal in - * http://www.json.com/json-schema-proposal/ - * - * @param \stdClass $instance - * @param \stdClass $schema - * @param JsonFormatValidator $formatValidator an optional class that have methods to validate the format definitions. - * If this is null, so format validation will not be applied, but if its true, then the validation will throw - * an error if any format defined on the schema is not supported by the validator. - * @return unknown - */ - public function validate($instance, $schema = null, $formatValidator = null) { - $this->errors = array(); - $this->formatValidator = null; + /** + * Validates a php object against a schema. Both the php object and the schema + * are supposed to be a result of a json_decode call. + * The validation works as defined by the schema proposal in + * http://www.json.com/json-schema-proposal/ + * + * @param \stdClass $instance + * @param \stdClass $schema + * @param JsonFormatValidator $formatValidator an optional class that have methods to validate the format definitions. + * If this is null, so format validation will not be applied, but if its true, then the validation will throw + * an error if any format defined on the schema is not supported by the validator. + * @return unknown + */ + public function validate($instance, $schema = null, $formatValidator = null) { + $this->errors = array(); + $this->formatValidator = null; - if($formatValidator) $this->formatValidator = $formatValidator; - $res = $this->_validate($instance,$schema,false); - $this->formatValidator = null; - return $res; - } - - function _validate($instance,$schema = null,$_changing) { - // verify passed schema - if ($schema) { - $this->checkProp($instance,$schema,'','',$_changing); + if($formatValidator) $this->formatValidator = $formatValidator; + $res = $this->_validate($instance,$schema,false); + $this->formatValidator = null; + return $res; } - // verify "inline" schema - $propName = '$schema'; - if (!$_changing && isset($instance->$propName)) { - $this->checkProp($instance,$instance->$propName,'','',$_changing); - } - // show results - $obj = new \stdClass(); - $obj->valid = ! ((boolean)count($this->errors)); - $obj->errors = $this->errors; - return $obj; - } - function incrementPath($path,$i) { - if($path !== '') { - if(is_int($i)) { - $path .= '['.$i.']'; - } - elseif($i == '') { - $path .= ''; - } - else { - $path .= '.'.$i; - } + function _validate($instance,$schema = null,$_changing) { + // verify passed schema + if ($schema) { + $this->checkProp($instance,$schema,'','',$_changing); + } + // verify "inline" schema + $propName = '$schema'; + if (!$_changing && isset($instance->$propName)) { + $this->checkProp($instance,$instance->$propName,'','',$_changing); + } + // show results + $obj = new \stdClass(); + $obj->valid = ! ((boolean)count($this->errors)); + $obj->errors = $this->errors; + return $obj; } - else { - $path = $i; - } - return $path; - } - function checkArray($value,$schema,$path,$i,$_changing) { - //verify items - if(isset($schema->items)) { - //tuple typing - if(is_array($schema->items)) { - foreach($value as $k=>$v) { - if(array_key_exists($k,$schema->items)) { - $this->checkProp($v,$schema->items[$k],$path,$k,$_changing); - } - else { - // aditional array properties - if(array_key_exists('additionalProperties',$schema)) { - if($schema->additionalProperties === false) { - $this->adderror( - $path, - 'The item '.$i.'['.$k.'] is not defined in the objTypeDef and the objTypeDef does not allow additional properties' - ); - } - else { - $this->checkProp($v,$schema->additionalProperties,$path,$k,$_changing); - } + function incrementPath($path,$i) { + if($path !== '') { + if(is_int($i)) { + $path .= '['.$i.']'; + } + elseif($i == '') { + $path .= ''; + } + else { + $path .= '.'.$i; } - } - }//foreach($value as $k=>$v) { - // treat when we have more schema definitions than values - for($k = count($value); $k < count($schema->items); $k++) { - $this->checkProp( - new JsonSchemaUndefined(), - $schema->items[$k], $path, $k, $_changing - ); - } - } - // just one type definition for the whole array - else { - foreach($value as $k=>$v) { - $this->checkProp($v,$schema->items,$path,$k,$_changing); - } - } - } - // verify number of array items - if(isset($schema->minItems) && count($value) < $schema->minItems) { - $this->adderror($path,"There must be a minimum of " . $schema->minItems . " in the array"); - } - if(isset($schema->maxItems) && count($value) > $schema->maxItems) { - $this->adderror($path,"There must be a maximum of " . $schema->maxItems . " in the array"); - } - } - - function checkProp($value, $schema, $path, $i = '', $_changing = false) { - if (!is_object($schema)) { - return; - } - $path = $this->incrementPath($path,$i); - // verify readonly - if($_changing && $schema.readonly) { - $this->adderror($path,'is a readonly field, it can not be changed'); - } - // I think a schema cant be an array, only the items property - /*if(is_array($schema)) { - if(!is_array($value)) { - return array(array('property'=>$path,'message'=>'An array tuple is required')); - } - for($a = 0; $a < count($schema); $a++) { - $this->errors = array_merge( - $this->errors, - $this->checkProp($value->$a,$schema->$a,$path,$i,$_changing) - ); - return $this->errors; - } - }*/ - // if it extends another schema, it must pass that schema as well - if(isset($schema->extends)) { - $this->checkProp($value,$schema->extends,$path,$i,$_changing); - } - // verify optional values - if (is_object($value) && $value instanceOf JsonSchemaUndefined) { - if ( isset($schema->optional) ? !$schema->optional : true) { - $this->adderror($path,"is missing and it is not optional"); - } - } - // normal verifications - else { - $this->errors = array_merge( - $this->errors, - $this->checkType( isset($schema->type) ? $schema->type : null , $value, $path) - ); - } - if(array_key_exists('disallow',$schema)) { - $errorsBeforeDisallowCheck = $this->errors; - $response = $this->checkType($schema->disallow, $value, $path); - if( - ( count($errorsBeforeDisallowCheck) == count($this->errors) ) && - !count($response) - ) { - $this->adderror($path," disallowed value was matched"); - } - else { - $this->errors = $errorsBeforeDisallowCheck; - } - } - //verify the itens on an array and min and max number of items. - if(is_array($value)) { - if( - $this->checkMode == $this::CHECK_MODE_TYPE_CAST && - $schema->type == 'object' - ) { - $this->checkObj( - $value, - $schema->properties, - $path, - isset($schema->additionalProperties) ? $schema->additionalProperties : null, - $_changing - ); - } - $this->checkArray($value,$schema,$path,$i,$_changing); - } - ############ verificar! - elseif(isset($schema->properties) && is_object($value)) { - $this->checkObj( - $value, - $schema->properties, - $path, - isset($schema->additionalProperties) ? $schema->additionalProperties : null, - $_changing - ); - } - // verify a regex pattern - if( isset($schema->pattern) && is_string($value) && !preg_match('/'.$schema->pattern.'/',$value)) { - $this->adderror($path,"does not match the regex pattern " . $schema->pattern); - } - // verify maxLength, minLength, maximum and minimum values - if( isset($schema->maxLength) && is_string($value) && (strlen($value) > $schema->maxLength)) { - $this->adderror($path,"must be at most " . $schema->maxLength . " characters long"); - } - if( isset($schema->minLength) && is_string($value) && strlen($value) < $schema->minLength) { - $this->adderror($path,"must be at least " . $schema->minLength . " characters long"); - } - - if( - isset($schema->minimum) && - gettype($value) == gettype($schema->minimum) && - $value < $schema->minimum - ) { - $this->adderror($path,"must have a minimum value of " . $schema->minimum); - } - if( isset($schema->maximum) && gettype($value) == gettype($schema->maximum) && $value > $schema->maximum) { - $this->adderror($path,"must have a maximum value of " . $schema->maximum); - } - // verify enum values - if(isset($schema->enum)) { - $found = false; - foreach($schema->enum as $possibleValue) { - if($possibleValue == $value) { - $found = true; - break; - } - } - if(!$found) { - $this->adderror($path,"does not have a value in the enumeration " . implode(', ',$schema->enum)); - } - } - if( - isset($schema->maxDecimal) && - ( ($value * pow(10,$schema->maxDecimal)) != (int)($value * pow(10,$schema->maxDecimal)) ) - ) { - $this->adderror($path,"may only have " . $schema->maxDecimal . " digits of decimal places"); - } - if( isset($schema->format) && isset($this->formatValidator) ) { - $error = $this->formatValidator->validate($value,$schema->format); - if($error) { - $this->adderror($path,$error); - } - } - } - - function adderror($path,$message) { - $this->errors[] = array( - 'property'=>$path, - 'message'=>$message - ); - } - - /** - * Take Care: Value is being passed by ref to continue validation with proper format. - * @return array - */ - function checkType($type, &$value, $path) { - if($type) { - $wrongType = false; - if(is_string($type) && $type !== 'any') { - if($type == 'null') { - if (!is_null($value)) { - $wrongType = true; - } } else { - if($type == 'number') { - if($this->checkMode == $this::CHECK_MODE_TYPE_CAST) { - $wrongType = !$this->checkTypeCast($type,$value); + $path = $i; + } + return $path; + } + + function checkArray($value,$schema,$path,$i,$_changing) { + //verify items + if(isset($schema->items)) { + //tuple typing + if(is_array($schema->items)) { + foreach($value as $k=>$v) { + if(array_key_exists($k,$schema->items)) { + $this->checkProp($v,$schema->items[$k],$path,$k,$_changing); + } + else { + // aditional array properties + if(array_key_exists('additionalProperties',$schema)) { + if($schema->additionalProperties === false) { + $this->adderror( + $path, + 'The item '.$i.'['.$k.'] is not defined in the objTypeDef and the objTypeDef does not allow additional properties' + ); + } + else { + $this->checkProp($v,$schema->additionalProperties,$path,$k,$_changing); + } + } + } + }//foreach($value as $k=>$v) { + // treat when we have more schema definitions than values + for($k = count($value); $k < count($schema->items); $k++) { + $this->checkProp( + new JsonSchemaUndefined(), + $schema->items[$k], $path, $k, $_changing + ); + } } - elseif(!in_array(gettype($value),array('integer','double'))) { - $wrongType = true; - } - } else{ + // just one type definition for the whole array + else { + foreach($value as $k=>$v) { + $this->checkProp($v,$schema->items,$path,$k,$_changing); + } + } + } + // verify number of array items + if(isset($schema->minItems) && count($value) < $schema->minItems) { + $this->adderror($path,"There must be a minimum of " . $schema->minItems . " in the array"); + } + if(isset($schema->maxItems) && count($value) > $schema->maxItems) { + $this->adderror($path,"There must be a maximum of " . $schema->maxItems . " in the array"); + } + } + + function checkProp($value, $schema, $path, $i = '', $_changing = false) { + if (!is_object($schema)) { + return; + } + $path = $this->incrementPath($path,$i); + // verify readonly + if($_changing && $schema.readonly) { + $this->adderror($path,'is a readonly field, it can not be changed'); + } + // I think a schema cant be an array, only the items property + /*if(is_array($schema)) { + if(!is_array($value)) { + return array(array('property'=>$path,'message'=>'An array tuple is required')); + } + for($a = 0; $a < count($schema); $a++) { + $this->errors = array_merge( + $this->errors, + $this->checkProp($value->$a,$schema->$a,$path,$i,$_changing) + ); + return $this->errors; + } + }*/ + // if it extends another schema, it must pass that schema as well + if(isset($schema->extends)) { + $this->checkProp($value,$schema->extends,$path,$i,$_changing); + } + // verify optional values + if (is_object($value) && $value instanceOf JsonSchemaUndefined) { + if ( isset($schema->optional) ? !$schema->optional : true) { + $this->adderror($path,"is missing and it is not optional"); + } + } + // normal verifications + else { + $this->errors = array_merge( + $this->errors, + $this->checkType( isset($schema->type) ? $schema->type : null , $value, $path) + ); + } + if(array_key_exists('disallow',$schema)) { + $errorsBeforeDisallowCheck = $this->errors; + $response = $this->checkType($schema->disallow, $value, $path); if( - $this->checkMode == $this::CHECK_MODE_TYPE_CAST - && $type == 'integer' + ( count($errorsBeforeDisallowCheck) == count($this->errors) ) && + !count($response) ) { - $wrongType = !$this->checkTypeCast($type,$value); - } elseif ( - $this->checkMode == $this::CHECK_MODE_TYPE_CAST - && $type == 'object' && is_array($value) - ) { - $wrongType = false; - } elseif ($type !== gettype($value)) { - $wrongType = true; + $this->adderror($path," disallowed value was matched"); + } + else { + $this->errors = $errorsBeforeDisallowCheck; } - } } - } - if($wrongType) { - return array( - array( - 'property'=>$path, - 'message'=>gettype($value)." value found, but a ".$type." is required" - ) - ); - } - // Union Types :: for now, just return the message for the last expected type!! - if(is_array($type)) { - $validatedOneType = false; - $errors = array(); - foreach($type as $tp) { - $error = $this->checkType($tp,$value,$path); - if(!count($error)) { - $validatedOneType = true; - break; - } - else { - $errors[] = $error; - $errors = $error; - } + //verify the itens on an array and min and max number of items. + if(is_array($value)) { + if( + $this->checkMode == $this::CHECK_MODE_TYPE_CAST && + $schema->type == 'object' + ) { + $this->checkObj( + $value, + $schema->properties, + $path, + isset($schema->additionalProperties) ? $schema->additionalProperties : null, + $_changing + ); + } + $this->checkArray($value,$schema,$path,$i,$_changing); } - if(!$validatedOneType) { - return $errors; + ############ verificar! + elseif(isset($schema->properties) && is_object($value)) { + $this->checkObj( + $value, + $schema->properties, + $path, + isset($schema->additionalProperties) ? $schema->additionalProperties : null, + $_changing + ); + } + // verify a regex pattern + if( isset($schema->pattern) && is_string($value) && !preg_match('/'.$schema->pattern.'/',$value)) { + $this->adderror($path,"does not match the regex pattern " . $schema->pattern); + } + // verify maxLength, minLength, maximum and minimum values + if( isset($schema->maxLength) && is_string($value) && (strlen($value) > $schema->maxLength)) { + $this->adderror($path,"must be at most " . $schema->maxLength . " characters long"); + } + if( isset($schema->minLength) && is_string($value) && strlen($value) < $schema->minLength) { + $this->adderror($path,"must be at least " . $schema->minLength . " characters long"); + } + + if( + isset($schema->minimum) && + gettype($value) == gettype($schema->minimum) && + $value < $schema->minimum + ) { + $this->adderror($path,"must have a minimum value of " . $schema->minimum); + } + if( isset($schema->maximum) && gettype($value) == gettype($schema->maximum) && $value > $schema->maximum) { + $this->adderror($path,"must have a maximum value of " . $schema->maximum); + } + // verify enum values + if(isset($schema->enum)) { + $found = false; + foreach($schema->enum as $possibleValue) { + if($possibleValue == $value) { + $found = true; + break; + } + } + if(!$found) { + $this->adderror($path,"does not have a value in the enumeration " . implode(', ',$schema->enum)); + } + } + if( + isset($schema->maxDecimal) && + ( ($value * pow(10,$schema->maxDecimal)) != (int)($value * pow(10,$schema->maxDecimal)) ) + ) { + $this->adderror($path,"may only have " . $schema->maxDecimal . " digits of decimal places"); + } + if( isset($schema->format) && isset($this->formatValidator) ) { + $error = $this->formatValidator->validate($value,$schema->format); + if($error) { + $this->adderror($path,$error); + } } - } - elseif(is_object($type)) { - $this->checkProp($value,$type,$path); - } } - return array(); - } - /** - * Take Care: Value is being passed by ref to continue validation with proper format. - */ - function checkTypeCast($type,&$value) { - switch($type) { - case 'integer': - $castValue = (integer)$value; - break; - case 'number': - $castValue = (double)$value; - break; - default: - trigger_error('this method should only be called for the above supported types.'); - break; - } - if( (string)$value == (string)$castValue ) { - $res = true; - $value = $castValue; - } - else { - $res = false; - } - return $res; - } - - function checkObj($instance, $objTypeDef, $path, $additionalProp,$_changing) { - if($objTypeDef instanceOf \stdClass) { - if( ! (($instance instanceOf \stdClass) || is_array($instance)) ) { - $this->errors[] = array( - 'property'=>$path, - 'message'=>"an object is required" - ); - } - foreach($objTypeDef as $i=>$value) { - $value = - array_key_exists($i,$instance) ? - (is_array($instance) ? $instance[$i] : $instance->$i) : - new JsonSchemaUndefined(); - $propDef = $objTypeDef->$i; - $this->checkProp($value,$propDef,$path,$i,$_changing); - } - } - // additional properties and requires - foreach($instance as $i=>$value) { - // verify additional properties, when its not allowed - if( !isset($objTypeDef->$i) && ($additionalProp === false) && $i !== '$schema' ) { - $this->errors[] = array( - 'property'=>$path, - 'message'=>"The property " . $i . " is not defined in the objTypeDef and the objTypeDef does not allow additional properties" - ); - } - // verify requires - if($objTypeDef && isset($objTypeDef->$i) && isset($objTypeDef->$i->requires)) { - $requires = $objTypeDef->$i->requires; - if(!array_key_exists($requires,$instance)) { - $this->errors[] = array( - 'property'=>$path, - 'message'=>"the presence of the property " . $i . " requires that " . $requires . " also be present" - ); - } - } - $value = is_array($instance) ? $instance[$i] : $instance->$i; - - // To verify additional properties types. - if ($objTypeDef && is_object($objTypeDef) && !isset($objTypeDef->$i)) { - $this->checkProp($value,$additionalProp,$path,$i); - } - // Verify inner schema definitions - $schemaPropName = '$schema'; - if (!$_changing && $value && isset($value->$schemaPropName)) { - $this->errors = array_merge( - $this->errors, - checkProp($value,$value->$schemaPropname,$path,$i) + function adderror($path,$message) { + $this->errors[] = array( + 'property'=>$path, + 'message'=>$message ); - } } - return $this->errors; - } + + /** + * Take Care: Value is being passed by ref to continue validation with proper format. + * @return array + */ + function checkType($type, &$value, $path) { + if($type) { + $wrongType = false; + if(is_string($type) && $type !== 'any') { + if($type == 'null') { + if (!is_null($value)) { + $wrongType = true; + } + } + else { + if($type == 'number') { + if($this->checkMode == $this::CHECK_MODE_TYPE_CAST) { + $wrongType = !$this->checkTypeCast($type,$value); + } + elseif(!in_array(gettype($value),array('integer','double'))) { + $wrongType = true; + } + } else{ + if( + $this->checkMode == $this::CHECK_MODE_TYPE_CAST + && $type == 'integer' + ) { + $wrongType = !$this->checkTypeCast($type,$value); + } elseif ( + $this->checkMode == $this::CHECK_MODE_TYPE_CAST + && $type == 'object' && is_array($value) + ) { + $wrongType = false; + } elseif ($type !== gettype($value)) { + $wrongType = true; + } + } + } + } + if($wrongType) { + return array( + array( + 'property'=>$path, + 'message'=>gettype($value)." value found, but a ".$type." is required" + ) + ); + } + // Union Types :: for now, just return the message for the last expected type!! + if(is_array($type)) { + $validatedOneType = false; + $errors = array(); + foreach($type as $tp) { + $error = $this->checkType($tp,$value,$path); + if(!count($error)) { + $validatedOneType = true; + break; + } + else { + $errors[] = $error; + $errors = $error; + } + } + if(!$validatedOneType) { + return $errors; + } + } + elseif(is_object($type)) { + $this->checkProp($value,$type,$path); + } + } + return array(); + } + + /** + * Take Care: Value is being passed by ref to continue validation with proper format. + */ + function checkTypeCast($type,&$value) { + switch($type) { + case 'integer': + $castValue = (integer)$value; + break; + case 'number': + $castValue = (double)$value; + break; + default: + trigger_error('this method should only be called for the above supported types.'); + break; + } + if( (string)$value == (string)$castValue ) { + $res = true; + $value = $castValue; + } + else { + $res = false; + } + return $res; + } + + function checkObj($instance, $objTypeDef, $path, $additionalProp,$_changing) { + if($objTypeDef instanceOf \stdClass) { + if( ! (($instance instanceOf \stdClass) || is_array($instance)) ) { + $this->errors[] = array( + 'property'=>$path, + 'message'=>"an object is required" + ); + } + foreach($objTypeDef as $i=>$value) { + $value = + array_key_exists($i,$instance) ? + (is_array($instance) ? $instance[$i] : $instance->$i) : + new JsonSchemaUndefined(); + $propDef = $objTypeDef->$i; + $this->checkProp($value,$propDef,$path,$i,$_changing); + } + } + // additional properties and requires + foreach($instance as $i=>$value) { + // verify additional properties, when its not allowed + if( !isset($objTypeDef->$i) && ($additionalProp === false) && $i !== '$schema' ) { + $this->errors[] = array( + 'property'=>$path, + 'message'=>"The property " . $i . " is not defined in the objTypeDef and the objTypeDef does not allow additional properties" + ); + } + // verify requires + if($objTypeDef && isset($objTypeDef->$i) && isset($objTypeDef->$i->requires)) { + $requires = $objTypeDef->$i->requires; + if(!array_key_exists($requires,$instance)) { + $this->errors[] = array( + 'property'=>$path, + 'message'=>"the presence of the property " . $i . " requires that " . $requires . " also be present" + ); + } + } + $value = is_array($instance) ? $instance[$i] : $instance->$i; + + // To verify additional properties types. + if ($objTypeDef && is_object($objTypeDef) && !isset($objTypeDef->$i)) { + $this->checkProp($value,$additionalProp,$path,$i); + } + // Verify inner schema definitions + $schemaPropName = '$schema'; + if (!$_changing && $value && isset($value->$schemaPropName)) { + $this->errors = array_merge( + $this->errors, + checkProp($value,$value->$schemaPropname,$path,$i) + ); + } + } + return $this->errors; + } }