json-schema-php

?php

/*
 * MoXie ([email protected]) 2010-12-11 
 *
 * Copyright © 2010-2010 Zoeey.Org
 * Code license: GNU Lesser General Public License Version 3
 * http://www.gnu.org/licenses/lgpl-3.0.txt
 *
 * website:
 * http://code.google.com/p/json-schema-php/
 */

/**
 * JSON Schema generate/validate
 * 
 * @author moxie([email protected])
 */
class JsonSchema {

    /**
     * JSON
     *
     * @var string
     */
    private $json;
    /**
     * Last error
     * 
     * @var array
     */
    private $errors;
    /**
     * Extend types
     *
     * @var map
     */
    private $complexTypes;

    /**
     *
     * @param string $json
     */
    function __construct($json) {
        $this->errors = array();
        $this->complexTypes = array();
        $this->json = json_decode($json);
    }

    /**
     * Generate JSON Schema
     *
     * @return string JSON Schema
     */
    public function getSchema() {
        $schema = null;
        $schema = $this->genByType($this->json);
        return json_encode($schema);
    }

    /**
     * Generate JSON Schema by type
     * @param mixed $value
     * @return object
     */
    private function genByType($value) {
        $type = gettype($value);
        $schema = array();
        switch ($type) {
            case 'boolean':
                $schema['type'] = 'boolean';
                $schema['default'] = false;
                break;
            case 'integer':
                $schema['type'] = 'integer';
                $schema['default'] = 0;
                $schema['minimum'] = 0;
                $schema['maximum'] = PHP_INT_MAX;
                $schema['exclusiveMinimum'] = 0;
                $schema['exclusiveMaximum'] = PHP_INT_MAX;
                break;
            case 'double':
                $schema['type'] = 'number';
                $schema['default'] = 0;
                $schema['minimum'] = 0;
                $schema['maximum'] = PHP_INT_MAX;
                $schema['exclusiveMinimum'] = 0;
                $schema['exclusiveMaximum'] = PHP_INT_MAX;
                break;
            case 'string':
                $schema['type'] = 'string';
                $schema['format'] = 'regex';
                $schema['pattern'] = '/^[a-z0-9]+$/i';
                $schema['minLength'] = 0;
                $schema['maxLength'] = PHP_INT_MAX;
                break;
            case 'array':
                $schema['type'] = 'array';
                $schema['minItems'] = 0;
                $schema['maxItems'] = 20;
                $items = array();
                foreach ($value as $value) {
                    $items = $this->genByType($value);
                    break;
                }
                $schema['items'] = $items;
                break;
            case 'object':
                $schema['type'] = 'object';
                $items = array();
                $value = get_object_vars($value);
                foreach ($value as $key => $value) {
                    $items[$key] = $this->genByType($value);
                }
                $schema['properties'] = $items;
                break;
            case 'null': // any in union types
                $schema['type'] = 'null';
                break;
            default:
                break;
        }
        return $schema;
    }

    /**
     * Set type schema
     * @param string $typeSchema
     */
    public function addType($typeSchema) {
        if (empty($typeSchema)) {
            return;
        }
        $typeSchema = json_decode($typeSchema, true);
        if (is_array($typeSchema) && isset($typeSchema['id'])) {
            $this->complexTypes[$typeSchema['id']] = $typeSchema;
        }
    }

    /**
     * Get type schema
     *
     * @param string ref
     * @return string schema
     */
    private function getType($ref) {
        if (isset($this->complexTypes[$ref])) {
            return $this->complexTypes[$ref];
        }
        return null;
    }

    /**
     * Validate JSON
     *
     * @param string $schema JSON Schema
     * @return boolean
     */
    public function validate($schema) {
        $isVali = false;
        do {
            $schema = json_decode($schema, true);
            if (!is_array($schema) || !isset($schema['type'])) {
                $this->addError('schema parse error. (PHP 5 >= 5.3.0) see json_last_error(void).');
                break;
            }
            $isVali = $this->checkByType($this->json, $schema);
        } while (false);
        return $isVali;
    }

    /**
     * check type: string
     * http://tools.ietf.org/html/draft-zyp-json-schema-03#section-5.1
     * 
     * @param string $value
     * @param array $schema
     */
    private function checkString($value, $schema) {
        // string
        $isVali = false;
        do {

            if (!is_string($value)) {
                $this->addError(sprintf('value: "%s" is not a string.', $value));
                break;
            }
            $len = strlen($value);
            if (isset($schema['minLength'])) {
                if ($schema['minLength'] > $len) {
                    $this->addError(sprintf('value: "%s" is too short.', $value));
                    break;
                }
            }
            if (isset($schema['maxLength'])) {
                if ($schema['maxLength'] < $len) {
                    $this->addError(sprintf('value: "%s" is too long.', $value));
                    break;
                }
            }

            if (isset($schema['format'])) {
                switch ($schema['format']) {

                    case 'date-time':
                        /**
                         * date-time  This SHOULD be a date in ISO 8601 format of YYYY-MM-
                         * DDThh:mm:ssZ in UTC time.  This is the recommended form of date/
                         * timestamp.
                         */
                        break;
                    case 'date':
                        /**
                         * date  This SHOULD be a date in the format of YYYY-MM-DD.  It is
                         * recommended that you use the "date-time" format instead of "date"
                         * unless you need to transfer only the date part.
                         */
                        break;
                    case 'time':
                        /**
                         * time  This SHOULD be a time in the format of hh:mm:ss.  It is
                         * recommended that you use the "date-time" format instead of "time"
                         * unless you need to transfer only the time part.
                         */
                        break;
                    case 'utc-millisec':
                        /**
                         * utc-millisec  This SHOULD be the difference, measured in
                         * milliseconds, between the specified time and midnight, 00:00 of
                         * January 1, 1970 UTC.  The value SHOULD be a number (integer or
                         * float).
                         */
                        break;
                    case 'regex':
                        /**
                         * regex  A regular expression, following the regular expression
                         * specification from ECMA 262/Perl 5.
                         */
                        if (isset($schema['pattern'])) {
                            $pattern = $schema['pattern'];
                            if (preg_match($pattern, $value)) {
                                $isVali = true;
                            } else {
                                $this->addError(printf('"%s" does not match "%s"', $value, $pattern));
                            }
                        } else {
                            $this->addError('format-regex: pattern is undefined.');
                        }

                        break;
                    case 'color':
                        /**
                         * color  This is a CSS color (like "#FF0000" or "red"), based on CSS
                         * 2.1 [W3C.CR-CSS21-20070719].
                         */
                        break;
                    case 'style':
                        /**
                         * style  This is a CSS style definition (like "color: red; background-
                         * color:#FFF"), based on CSS 2.1 [W3C.CR-CSS21-20070719].
                         */
                        break;
                    case 'phone':
                        /**
                         * phone  This SHOULD be a phone number (format MAY follow E.123).
                         * http://en.wikipedia.org/wiki/E.123
                         */
                        if (preg_match("/^(\(0?[0-9]{2}\) \d{3,4}\s?\d{4}|\+\d{2} \d{2} \d{3,4}\s?\d{4})$/", $value)) {
                            $isVali = true;
                        } else {
                            $this->addError(sprintf('value: "%s" is not a phone number.', $value));
                        }
                        break;
                    case 'uri':
                        /**
                         * uri  This value SHOULD be a URI..
                         */
                        if (filter_var($value, FILTER_VALIDATE_URL, FILTER_FLAG_QUERY_REQUIRED)) {
                            $isVali = true;
                        } else {
                            $this->addError(sprintf('value: "%s" is not a URI.', $value));
                        }
                        break;
                    case 'email':
                        /**
                         *  email  This SHOULD be an email address.
                         */
                        if (filter_var($value, FILTER_VALIDATE_EMAIL)) {
                            $isVali = true;
                        } else {
                            $this->addError(sprintf('value: "%s" is not a email.', $value));
                        }
                        break;
                    case 'ip-address':
                        /**
                         *  ip-address  This SHOULD be an ip version 4 address.
                         */
                        if (filter_var($value, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)) {
                            $isVali = true;
                        } else {
                            $this->addError(sprintf('value: "%s" is not a ipv4 address.', $value));
                        }

                        break;
                    case 'ipv6':
                        /**
                         *  ipv6  This SHOULD be an ip version 6 address.
                         */
                        if (filter_var($value, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6)) {
                            $isVali = true;
                        } else {
                            $this->addError(sprintf('value: "%s" is not a ipv6 address.', $value));
                        }
                        break;
                    case 'host-name':
                        /**
                         *  host-name  This SHOULD be a host-name.
                         */
                        break;

                    default:
                        $this->addError(sprintf('format: "%s" is undefined.', $schema['format']));
                        break;
                }
                break;
            }

            $isVali = true;
        } while (false);
        return $isVali;
    }

    /**
     * check type: integer/double
     *
     * @param number $value
     * @param array $schema
     * @return boolean
     */
    private function checkNumber($value, $schema) {
        // number
        $isVali = false;
        do {

            if (!is_numeric($value)) {
                $this->addError($value . ' is not a number.');
                break;
            }
            if (isset($schema['minimum'])) {
                if ($schema['minimum'] > $value) {
                    $this->addError(sprintf('%s is less than %s.', $value, $schema['minimum']));
                    break;
                }
            }
            if (isset($schema['maximum'])) {
                if ($schema['maximum'] < $value) {
                    $this->addError(sprintf('%s is bigger than %s.', $value, $schema['maximum']));
                    break;
                }
            }
            if (isset($schema['exclusiveMinimum'])) {
                if ($schema['exclusiveMinimum'] >= $value) {
                    $this->addError(sprintf('%s is less or than %s or equal.', $value, $schema['exclusiveMinimum']));
                    break;
                }
            }
            if (isset($schema['exclusiveMaximum'])) {
                if ($schema['exclusiveMaximum'] <= $value) {
                    $this->addError(sprintf('%s is bigger than %s or equal.', $value, $schema['exclusiveMaximum']));
                    break;
                }
            }
            $isVali = true;
        } while (false);

        return $isVali;
    }

    /**
     * check type: integer
     *
     * @param integer $value
     * @param array $schema
     * @return boolean
     */
    private function checkInteger($value, $schema) {
        // integer
        if (!is_integer($value)) {
            $this->addError(sprintf('value:"%s" is not a integer.', $value));
            return false;
        }
        return $this->checkNumber($value, $schema);
    }

    /**
     * check type: boolean
     *
     * @param boolean $value
     * @param array $schema
     * @return boolean
     */
    private function checkBoolean($value, $schema) {
        // boolean
        if (!is_bool($value)) {
            $this->addError(sprintf('value: "%s" is not a boolean.', $value));
            return false;
        }
        return true;
    }

    /**
     * check type: object
     *
     * @param object $valueProp
     * @param array $schema
     * @return boolean
     */
    private function checkObject($value, $schema) {
        // object
        $isVali = false;
        do {
            if (!is_object($value)) {
                $this->addError(sprintf('value: "%s" is not an object.', $value));
                break;
            }
            if (isset($schema['properties'])
                    && !empty($schema['properties'])
            ) {
                $schemaProp = $schema['properties'];
                $valueProp = get_object_vars($value);
                $valueKeys = array_keys($valueProp);
                $schemaKeys = array_keys($schemaProp);
                $diffKeys = array_diff($valueKeys, $schemaKeys);
                if (!empty($diffKeys)) {
                    foreach ($diffKeys as $key) {
                        // property not defined / not optional
                        if (!isset($schemaProp[$key])
                                || !isset($schemaProp[$key]['optional'])
                                || !$schemaProp[$key]['optional']
                        ) {
                            $this->addError(sprintf('key: "%s" is not exist,And it\'s not a optional property.', $value));
                            break 2;
                        }
                    }
                }
                foreach ($schemaProp as $key => $sch) {
                    if (!isset($valueProp[$key])) {
                        continue;
                    }
                    if (!$this->checkByType($valueProp[$key], $sch)) {
                        break 2;
                    }
                }
            }
            $isVali = true;
        } while (false);
        return $isVali;
    }

    /**
     * check type: array
     *
     * @param array $value
     * @param array $schema
     * @return boolean 
     */
    private function checkArray($value, $schema) {
        $isVali = false;
        do {
            if (!is_array($value)) {
                $this->addError(sprintf('value: "%s" is not an array.', $value));
                break;
            }

            if (!isset($schema['items'])) {
                $this->addError('schema: items schema is undefined.');
                break;
            }
            $size = count($value);
            if (isset($schema['minItems'])) {
                if ($schema['minItems'] > $size) {
                    $this->addError(sprintf('array size: %s  is less than %s.', $size, $schema['minItems']));
                    break;
                }
            }
            if (isset($schema['maxItems'])) {
                if ($schema['maxItems'] < $size) {
                    $this->addError(sprintf('array size: %s is bigger than %s.', $size, $schema['maxItems']));
                    break;
                }
            }

            foreach ($value as $val) {
                if (!$this->checkByType($val, $schema['items'])) {
                    break 2;
                }
            }


            $isVali = true;
        } while (false);
        return $isVali;
    }

    /**
     * check value based on type
     *
     * @param mixed $value
     * @param array $schema
     * @return boolean
     */
    private function checkByType($value, $schema) {
        $isVali = false;
        if ($schema && isset($schema['type'])) {
            // union types
            if (is_array($schema['type'])) {
                $types = $schema['type'];
                foreach ($types as $type) {
                    $schema['type'] = $type;
                    $isVali = $this->checkByType($value, $schema);
                    if ($isVali) {
                        break;
                    }
                }
            } else {
                $type = $schema['type'];
                switch ($type) {
                    case 'boolean':
                        $isVali = $this->checkBoolean($value, $schema);
                        break;
                    case 'integer':
                        $isVali = $this->checkInteger($value, $schema);
                        break;
                    case 'number':
                        $isVali = $this->checkNumber($value, $schema);
                        break;
                    case 'string':
                        $isVali = $this->checkString($value, $schema);
                        break;
                    case 'array':
                        $isVali = $this->checkArray($value, $schema);
                        break;
                    case 'object':
                        $isVali = $this->checkObject($value, $schema);
                        break;
                    case 'enum':
                        $isVali = is_null($value);
                        break;
                    case 'null':
                        $isVali = is_null($value);
                        break;
                    case 'any':
                        $isVali = true;
                        break;
                    default:
                        $this->addError(sprintf('type_schema: "%s" is undefined.', $value));
                        break;
                }
            }
        }
        if (isset($schema['$ref'])) {
            $isVali = $this->checkByType($value, $this->getType($schema['$ref']));
        }
        return $isVali;
    }

    /**
     *  Get errors
     *
     * @return array errors
     */
    public function getErrors() {
        return $this->errors;
    }

    /**
     * add error message
     * @param string $msg
     */
    protected function addError($msg) {
        $this->errors[] = htmlentities($msg);
    }

}

?>
 

你可能感兴趣的:(PHP,json,css,perl,Gmail)