| @ -0,0 +1 @@ | |||||
| ../sshpk/bin/sshpk-conv | |||||
| @ -0,0 +1 @@ | |||||
| ../sshpk/bin/sshpk-sign | |||||
| @ -0,0 +1 @@ | |||||
| ../sshpk/bin/sshpk-verify | |||||
| @ -0,0 +1 @@ | |||||
| ../uuid/bin/uuid | |||||
| @ -0,0 +1,20 @@ | |||||
| var Ajv = require('ajv'); | |||||
| var ajv = new Ajv({allErrors: true}); | |||||
| var schema = { | |||||
| "properties": { | |||||
| "foo": { "type": "string" }, | |||||
| "bar": { "type": "number", "maximum": 3 } | |||||
| } | |||||
| }; | |||||
| var validate = ajv.compile(schema); | |||||
| test({"foo": "abc", "bar": 2}); | |||||
| test({"foo": 2, "bar": 4}); | |||||
| function test(data) { | |||||
| var valid = validate(data); | |||||
| if (valid) console.log('Valid!'); | |||||
| else console.log('Invalid: ' + ajv.errorsText(validate.errors)); | |||||
| } | |||||
| @ -0,0 +1,22 @@ | |||||
| The MIT License (MIT) | |||||
| Copyright (c) 2015-2017 Evgeny Poberezkin | |||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | |||||
| of this software and associated documentation files (the "Software"), to deal | |||||
| in the Software without restriction, including without limitation the rights | |||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |||||
| copies of the Software, and to permit persons to whom the Software is | |||||
| furnished to do so, subject to the following conditions: | |||||
| The above copyright notice and this permission notice shall be included in all | |||||
| copies or substantial portions of the Software. | |||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |||||
| SOFTWARE. | |||||
| @ -0,0 +1,395 @@ | |||||
| declare var ajv: { | |||||
| (options?: ajv.Options): ajv.Ajv; | |||||
| new(options?: ajv.Options): ajv.Ajv; | |||||
| ValidationError: typeof AjvErrors.ValidationError; | |||||
| MissingRefError: typeof AjvErrors.MissingRefError; | |||||
| $dataMetaSchema: object; | |||||
| } | |||||
| declare namespace AjvErrors { | |||||
| class ValidationError extends Error { | |||||
| constructor(errors: Array<ajv.ErrorObject>); | |||||
| message: string; | |||||
| errors: Array<ajv.ErrorObject>; | |||||
| ajv: true; | |||||
| validation: true; | |||||
| } | |||||
| class MissingRefError extends Error { | |||||
| constructor(baseId: string, ref: string, message?: string); | |||||
| static message: (baseId: string, ref: string) => string; | |||||
| message: string; | |||||
| missingRef: string; | |||||
| missingSchema: string; | |||||
| } | |||||
| } | |||||
| declare namespace ajv { | |||||
| type ValidationError = AjvErrors.ValidationError; | |||||
| type MissingRefError = AjvErrors.MissingRefError; | |||||
| interface Ajv { | |||||
| /** | |||||
| * Validate data using schema | |||||
| * Schema will be compiled and cached (using serialized JSON as key, [fast-json-stable-stringify](https://github.com/epoberezkin/fast-json-stable-stringify) is used to serialize by default). | |||||
| * @param {string|object|Boolean} schemaKeyRef key, ref or schema object | |||||
| * @param {Any} data to be validated | |||||
| * @return {Boolean} validation result. Errors from the last validation will be available in `ajv.errors` (and also in compiled schema: `schema.errors`). | |||||
| */ | |||||
| validate(schemaKeyRef: object | string | boolean, data: any): boolean | PromiseLike<any>; | |||||
| /** | |||||
| * Create validating function for passed schema. | |||||
| * @param {object|Boolean} schema schema object | |||||
| * @return {Function} validating function | |||||
| */ | |||||
| compile(schema: object | boolean): ValidateFunction; | |||||
| /** | |||||
| * Creates validating function for passed schema with asynchronous loading of missing schemas. | |||||
| * `loadSchema` option should be a function that accepts schema uri and node-style callback. | |||||
| * @this Ajv | |||||
| * @param {object|Boolean} schema schema object | |||||
| * @param {Boolean} meta optional true to compile meta-schema; this parameter can be skipped | |||||
| * @param {Function} callback optional node-style callback, it is always called with 2 parameters: error (or null) and validating function. | |||||
| * @return {PromiseLike<ValidateFunction>} validating function | |||||
| */ | |||||
| compileAsync(schema: object | boolean, meta?: Boolean, callback?: (err: Error, validate: ValidateFunction) => any): PromiseLike<ValidateFunction>; | |||||
| /** | |||||
| * Adds schema to the instance. | |||||
| * @param {object|Array} schema schema or array of schemas. If array is passed, `key` and other parameters will be ignored. | |||||
| * @param {string} key Optional schema key. Can be passed to `validate` method instead of schema object or id/ref. One schema per instance can have empty `id` and `key`. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| addSchema(schema: Array<object> | object, key?: string): Ajv; | |||||
| /** | |||||
| * Add schema that will be used to validate other schemas | |||||
| * options in META_IGNORE_OPTIONS are alway set to false | |||||
| * @param {object} schema schema object | |||||
| * @param {string} key optional schema key | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| addMetaSchema(schema: object, key?: string): Ajv; | |||||
| /** | |||||
| * Validate schema | |||||
| * @param {object|Boolean} schema schema to validate | |||||
| * @return {Boolean} true if schema is valid | |||||
| */ | |||||
| validateSchema(schema: object | boolean): boolean; | |||||
| /** | |||||
| * Get compiled schema from the instance by `key` or `ref`. | |||||
| * @param {string} keyRef `key` that was passed to `addSchema` or full schema reference (`schema.id` or resolved id). | |||||
| * @return {Function} schema validating function (with property `schema`). Returns undefined if keyRef can't be resolved to an existing schema. | |||||
| */ | |||||
| getSchema(keyRef: string): ValidateFunction | undefined; | |||||
| /** | |||||
| * Remove cached schema(s). | |||||
| * If no parameter is passed all schemas but meta-schemas are removed. | |||||
| * If RegExp is passed all schemas with key/id matching pattern but meta-schemas are removed. | |||||
| * Even if schema is referenced by other schemas it still can be removed as other schemas have local references. | |||||
| * @param {string|object|RegExp|Boolean} schemaKeyRef key, ref, pattern to match key/ref or schema object | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| removeSchema(schemaKeyRef?: object | string | RegExp | boolean): Ajv; | |||||
| /** | |||||
| * Add custom format | |||||
| * @param {string} name format name | |||||
| * @param {string|RegExp|Function} format string is converted to RegExp; function should return boolean (true when valid) | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| addFormat(name: string, format: FormatValidator | FormatDefinition): Ajv; | |||||
| /** | |||||
| * Define custom keyword | |||||
| * @this Ajv | |||||
| * @param {string} keyword custom keyword, should be a valid identifier, should be different from all standard, custom and macro keywords. | |||||
| * @param {object} definition keyword definition object with properties `type` (type(s) which the keyword applies to), `validate` or `compile`. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| addKeyword(keyword: string, definition: KeywordDefinition): Ajv; | |||||
| /** | |||||
| * Get keyword definition | |||||
| * @this Ajv | |||||
| * @param {string} keyword pre-defined or custom keyword. | |||||
| * @return {object|Boolean} custom keyword definition, `true` if it is a predefined keyword, `false` otherwise. | |||||
| */ | |||||
| getKeyword(keyword: string): object | boolean; | |||||
| /** | |||||
| * Remove keyword | |||||
| * @this Ajv | |||||
| * @param {string} keyword pre-defined or custom keyword. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| removeKeyword(keyword: string): Ajv; | |||||
| /** | |||||
| * Validate keyword | |||||
| * @this Ajv | |||||
| * @param {object} definition keyword definition object | |||||
| * @param {boolean} throwError true to throw exception if definition is invalid | |||||
| * @return {boolean} validation result | |||||
| */ | |||||
| validateKeyword(definition: KeywordDefinition, throwError: boolean): boolean; | |||||
| /** | |||||
| * Convert array of error message objects to string | |||||
| * @param {Array<object>} errors optional array of validation errors, if not passed errors from the instance are used. | |||||
| * @param {object} options optional options with properties `separator` and `dataVar`. | |||||
| * @return {string} human readable string with all errors descriptions | |||||
| */ | |||||
| errorsText(errors?: Array<ErrorObject> | null, options?: ErrorsTextOptions): string; | |||||
| errors?: Array<ErrorObject> | null; | |||||
| } | |||||
| interface CustomLogger { | |||||
| log(...args: any[]): any; | |||||
| warn(...args: any[]): any; | |||||
| error(...args: any[]): any; | |||||
| } | |||||
| interface ValidateFunction { | |||||
| ( | |||||
| data: any, | |||||
| dataPath?: string, | |||||
| parentData?: object | Array<any>, | |||||
| parentDataProperty?: string | number, | |||||
| rootData?: object | Array<any> | |||||
| ): boolean | PromiseLike<any>; | |||||
| schema?: object | boolean; | |||||
| errors?: null | Array<ErrorObject>; | |||||
| refs?: object; | |||||
| refVal?: Array<any>; | |||||
| root?: ValidateFunction | object; | |||||
| $async?: true; | |||||
| source?: object; | |||||
| } | |||||
| interface Options { | |||||
| $data?: boolean; | |||||
| allErrors?: boolean; | |||||
| verbose?: boolean; | |||||
| jsonPointers?: boolean; | |||||
| uniqueItems?: boolean; | |||||
| unicode?: boolean; | |||||
| format?: false | string; | |||||
| formats?: object; | |||||
| keywords?: object; | |||||
| unknownFormats?: true | string[] | 'ignore'; | |||||
| schemas?: Array<object> | object; | |||||
| schemaId?: '$id' | 'id' | 'auto'; | |||||
| missingRefs?: true | 'ignore' | 'fail'; | |||||
| extendRefs?: true | 'ignore' | 'fail'; | |||||
| loadSchema?: (uri: string, cb?: (err: Error, schema: object) => void) => PromiseLike<object | boolean>; | |||||
| removeAdditional?: boolean | 'all' | 'failing'; | |||||
| useDefaults?: boolean | 'empty' | 'shared'; | |||||
| coerceTypes?: boolean | 'array'; | |||||
| strictDefaults?: boolean | 'log'; | |||||
| strictKeywords?: boolean | 'log'; | |||||
| async?: boolean | string; | |||||
| transpile?: string | ((code: string) => string); | |||||
| meta?: boolean | object; | |||||
| validateSchema?: boolean | 'log'; | |||||
| addUsedSchema?: boolean; | |||||
| inlineRefs?: boolean | number; | |||||
| passContext?: boolean; | |||||
| loopRequired?: number; | |||||
| ownProperties?: boolean; | |||||
| multipleOfPrecision?: boolean | number; | |||||
| errorDataPath?: string, | |||||
| messages?: boolean; | |||||
| sourceCode?: boolean; | |||||
| processCode?: (code: string) => string; | |||||
| cache?: object; | |||||
| logger?: CustomLogger | false; | |||||
| nullable?: boolean; | |||||
| serialize?: ((schema: object | boolean) => any) | false; | |||||
| } | |||||
| type FormatValidator = string | RegExp | ((data: string) => boolean | PromiseLike<any>); | |||||
| type NumberFormatValidator = ((data: number) => boolean | PromiseLike<any>); | |||||
| interface NumberFormatDefinition { | |||||
| type: "number", | |||||
| validate: NumberFormatValidator; | |||||
| compare?: (data1: number, data2: number) => number; | |||||
| async?: boolean; | |||||
| } | |||||
| interface StringFormatDefinition { | |||||
| type?: "string", | |||||
| validate: FormatValidator; | |||||
| compare?: (data1: string, data2: string) => number; | |||||
| async?: boolean; | |||||
| } | |||||
| type FormatDefinition = NumberFormatDefinition | StringFormatDefinition; | |||||
| interface KeywordDefinition { | |||||
| type?: string | Array<string>; | |||||
| async?: boolean; | |||||
| $data?: boolean; | |||||
| errors?: boolean | string; | |||||
| metaSchema?: object; | |||||
| // schema: false makes validate not to expect schema (ValidateFunction) | |||||
| schema?: boolean; | |||||
| statements?: boolean; | |||||
| dependencies?: Array<string>; | |||||
| modifying?: boolean; | |||||
| valid?: boolean; | |||||
| // one and only one of the following properties should be present | |||||
| validate?: SchemaValidateFunction | ValidateFunction; | |||||
| compile?: (schema: any, parentSchema: object, it: CompilationContext) => ValidateFunction; | |||||
| macro?: (schema: any, parentSchema: object, it: CompilationContext) => object | boolean; | |||||
| inline?: (it: CompilationContext, keyword: string, schema: any, parentSchema: object) => string; | |||||
| } | |||||
| interface CompilationContext { | |||||
| level: number; | |||||
| dataLevel: number; | |||||
| dataPathArr: string[]; | |||||
| schema: any; | |||||
| schemaPath: string; | |||||
| baseId: string; | |||||
| async: boolean; | |||||
| opts: Options; | |||||
| formats: { | |||||
| [index: string]: FormatDefinition | undefined; | |||||
| }; | |||||
| keywords: { | |||||
| [index: string]: KeywordDefinition | undefined; | |||||
| }; | |||||
| compositeRule: boolean; | |||||
| validate: (schema: object) => boolean; | |||||
| util: { | |||||
| copy(obj: any, target?: any): any; | |||||
| toHash(source: string[]): { [index: string]: true | undefined }; | |||||
| equal(obj: any, target: any): boolean; | |||||
| getProperty(str: string): string; | |||||
| schemaHasRules(schema: object, rules: any): string; | |||||
| escapeQuotes(str: string): string; | |||||
| toQuotedString(str: string): string; | |||||
| getData(jsonPointer: string, dataLevel: number, paths: string[]): string; | |||||
| escapeJsonPointer(str: string): string; | |||||
| unescapeJsonPointer(str: string): string; | |||||
| escapeFragment(str: string): string; | |||||
| unescapeFragment(str: string): string; | |||||
| }; | |||||
| self: Ajv; | |||||
| } | |||||
| interface SchemaValidateFunction { | |||||
| ( | |||||
| schema: any, | |||||
| data: any, | |||||
| parentSchema?: object, | |||||
| dataPath?: string, | |||||
| parentData?: object | Array<any>, | |||||
| parentDataProperty?: string | number, | |||||
| rootData?: object | Array<any> | |||||
| ): boolean | PromiseLike<any>; | |||||
| errors?: Array<ErrorObject>; | |||||
| } | |||||
| interface ErrorsTextOptions { | |||||
| separator?: string; | |||||
| dataVar?: string; | |||||
| } | |||||
| interface ErrorObject { | |||||
| keyword: string; | |||||
| dataPath: string; | |||||
| schemaPath: string; | |||||
| params: ErrorParameters; | |||||
| // Added to validation errors of propertyNames keyword schema | |||||
| propertyName?: string; | |||||
| // Excluded if messages set to false. | |||||
| message?: string; | |||||
| // These are added with the `verbose` option. | |||||
| schema?: any; | |||||
| parentSchema?: object; | |||||
| data?: any; | |||||
| } | |||||
| type ErrorParameters = RefParams | LimitParams | AdditionalPropertiesParams | | |||||
| DependenciesParams | FormatParams | ComparisonParams | | |||||
| MultipleOfParams | PatternParams | RequiredParams | | |||||
| TypeParams | UniqueItemsParams | CustomParams | | |||||
| PatternRequiredParams | PropertyNamesParams | | |||||
| IfParams | SwitchParams | NoParams | EnumParams; | |||||
| interface RefParams { | |||||
| ref: string; | |||||
| } | |||||
| interface LimitParams { | |||||
| limit: number; | |||||
| } | |||||
| interface AdditionalPropertiesParams { | |||||
| additionalProperty: string; | |||||
| } | |||||
| interface DependenciesParams { | |||||
| property: string; | |||||
| missingProperty: string; | |||||
| depsCount: number; | |||||
| deps: string; | |||||
| } | |||||
| interface FormatParams { | |||||
| format: string | |||||
| } | |||||
| interface ComparisonParams { | |||||
| comparison: string; | |||||
| limit: number | string; | |||||
| exclusive: boolean; | |||||
| } | |||||
| interface MultipleOfParams { | |||||
| multipleOf: number; | |||||
| } | |||||
| interface PatternParams { | |||||
| pattern: string; | |||||
| } | |||||
| interface RequiredParams { | |||||
| missingProperty: string; | |||||
| } | |||||
| interface TypeParams { | |||||
| type: string; | |||||
| } | |||||
| interface UniqueItemsParams { | |||||
| i: number; | |||||
| j: number; | |||||
| } | |||||
| interface CustomParams { | |||||
| keyword: string; | |||||
| } | |||||
| interface PatternRequiredParams { | |||||
| missingPattern: string; | |||||
| } | |||||
| interface PropertyNamesParams { | |||||
| propertyName: string; | |||||
| } | |||||
| interface IfParams { | |||||
| failingKeyword: string; | |||||
| } | |||||
| interface SwitchParams { | |||||
| caseIndex: number; | |||||
| } | |||||
| interface NoParams { } | |||||
| interface EnumParams { | |||||
| allowedValues: Array<any>; | |||||
| } | |||||
| } | |||||
| export = ajv; | |||||
| @ -0,0 +1,506 @@ | |||||
| 'use strict'; | |||||
| var compileSchema = require('./compile') | |||||
| , resolve = require('./compile/resolve') | |||||
| , Cache = require('./cache') | |||||
| , SchemaObject = require('./compile/schema_obj') | |||||
| , stableStringify = require('fast-json-stable-stringify') | |||||
| , formats = require('./compile/formats') | |||||
| , rules = require('./compile/rules') | |||||
| , $dataMetaSchema = require('./data') | |||||
| , util = require('./compile/util'); | |||||
| module.exports = Ajv; | |||||
| Ajv.prototype.validate = validate; | |||||
| Ajv.prototype.compile = compile; | |||||
| Ajv.prototype.addSchema = addSchema; | |||||
| Ajv.prototype.addMetaSchema = addMetaSchema; | |||||
| Ajv.prototype.validateSchema = validateSchema; | |||||
| Ajv.prototype.getSchema = getSchema; | |||||
| Ajv.prototype.removeSchema = removeSchema; | |||||
| Ajv.prototype.addFormat = addFormat; | |||||
| Ajv.prototype.errorsText = errorsText; | |||||
| Ajv.prototype._addSchema = _addSchema; | |||||
| Ajv.prototype._compile = _compile; | |||||
| Ajv.prototype.compileAsync = require('./compile/async'); | |||||
| var customKeyword = require('./keyword'); | |||||
| Ajv.prototype.addKeyword = customKeyword.add; | |||||
| Ajv.prototype.getKeyword = customKeyword.get; | |||||
| Ajv.prototype.removeKeyword = customKeyword.remove; | |||||
| Ajv.prototype.validateKeyword = customKeyword.validate; | |||||
| var errorClasses = require('./compile/error_classes'); | |||||
| Ajv.ValidationError = errorClasses.Validation; | |||||
| Ajv.MissingRefError = errorClasses.MissingRef; | |||||
| Ajv.$dataMetaSchema = $dataMetaSchema; | |||||
| var META_SCHEMA_ID = 'http://json-schema.org/draft-07/schema'; | |||||
| var META_IGNORE_OPTIONS = [ 'removeAdditional', 'useDefaults', 'coerceTypes', 'strictDefaults' ]; | |||||
| var META_SUPPORT_DATA = ['/properties']; | |||||
| /** | |||||
| * Creates validator instance. | |||||
| * Usage: `Ajv(opts)` | |||||
| * @param {Object} opts optional options | |||||
| * @return {Object} ajv instance | |||||
| */ | |||||
| function Ajv(opts) { | |||||
| if (!(this instanceof Ajv)) return new Ajv(opts); | |||||
| opts = this._opts = util.copy(opts) || {}; | |||||
| setLogger(this); | |||||
| this._schemas = {}; | |||||
| this._refs = {}; | |||||
| this._fragments = {}; | |||||
| this._formats = formats(opts.format); | |||||
| this._cache = opts.cache || new Cache; | |||||
| this._loadingSchemas = {}; | |||||
| this._compilations = []; | |||||
| this.RULES = rules(); | |||||
| this._getId = chooseGetId(opts); | |||||
| opts.loopRequired = opts.loopRequired || Infinity; | |||||
| if (opts.errorDataPath == 'property') opts._errorDataPathProperty = true; | |||||
| if (opts.serialize === undefined) opts.serialize = stableStringify; | |||||
| this._metaOpts = getMetaSchemaOptions(this); | |||||
| if (opts.formats) addInitialFormats(this); | |||||
| if (opts.keywords) addInitialKeywords(this); | |||||
| addDefaultMetaSchema(this); | |||||
| if (typeof opts.meta == 'object') this.addMetaSchema(opts.meta); | |||||
| if (opts.nullable) this.addKeyword('nullable', {metaSchema: {type: 'boolean'}}); | |||||
| addInitialSchemas(this); | |||||
| } | |||||
| /** | |||||
| * Validate data using schema | |||||
| * Schema will be compiled and cached (using serialized JSON as key. [fast-json-stable-stringify](https://github.com/epoberezkin/fast-json-stable-stringify) is used to serialize. | |||||
| * @this Ajv | |||||
| * @param {String|Object} schemaKeyRef key, ref or schema object | |||||
| * @param {Any} data to be validated | |||||
| * @return {Boolean} validation result. Errors from the last validation will be available in `ajv.errors` (and also in compiled schema: `schema.errors`). | |||||
| */ | |||||
| function validate(schemaKeyRef, data) { | |||||
| var v; | |||||
| if (typeof schemaKeyRef == 'string') { | |||||
| v = this.getSchema(schemaKeyRef); | |||||
| if (!v) throw new Error('no schema with key or ref "' + schemaKeyRef + '"'); | |||||
| } else { | |||||
| var schemaObj = this._addSchema(schemaKeyRef); | |||||
| v = schemaObj.validate || this._compile(schemaObj); | |||||
| } | |||||
| var valid = v(data); | |||||
| if (v.$async !== true) this.errors = v.errors; | |||||
| return valid; | |||||
| } | |||||
| /** | |||||
| * Create validating function for passed schema. | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema object | |||||
| * @param {Boolean} _meta true if schema is a meta-schema. Used internally to compile meta schemas of custom keywords. | |||||
| * @return {Function} validating function | |||||
| */ | |||||
| function compile(schema, _meta) { | |||||
| var schemaObj = this._addSchema(schema, undefined, _meta); | |||||
| return schemaObj.validate || this._compile(schemaObj); | |||||
| } | |||||
| /** | |||||
| * Adds schema to the instance. | |||||
| * @this Ajv | |||||
| * @param {Object|Array} schema schema or array of schemas. If array is passed, `key` and other parameters will be ignored. | |||||
| * @param {String} key Optional schema key. Can be passed to `validate` method instead of schema object or id/ref. One schema per instance can have empty `id` and `key`. | |||||
| * @param {Boolean} _skipValidation true to skip schema validation. Used internally, option validateSchema should be used instead. | |||||
| * @param {Boolean} _meta true if schema is a meta-schema. Used internally, addMetaSchema should be used instead. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function addSchema(schema, key, _skipValidation, _meta) { | |||||
| if (Array.isArray(schema)){ | |||||
| for (var i=0; i<schema.length; i++) this.addSchema(schema[i], undefined, _skipValidation, _meta); | |||||
| return this; | |||||
| } | |||||
| var id = this._getId(schema); | |||||
| if (id !== undefined && typeof id != 'string') | |||||
| throw new Error('schema id must be string'); | |||||
| key = resolve.normalizeId(key || id); | |||||
| checkUnique(this, key); | |||||
| this._schemas[key] = this._addSchema(schema, _skipValidation, _meta, true); | |||||
| return this; | |||||
| } | |||||
| /** | |||||
| * Add schema that will be used to validate other schemas | |||||
| * options in META_IGNORE_OPTIONS are alway set to false | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema object | |||||
| * @param {String} key optional schema key | |||||
| * @param {Boolean} skipValidation true to skip schema validation, can be used to override validateSchema option for meta-schema | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function addMetaSchema(schema, key, skipValidation) { | |||||
| this.addSchema(schema, key, skipValidation, true); | |||||
| return this; | |||||
| } | |||||
| /** | |||||
| * Validate schema | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema to validate | |||||
| * @param {Boolean} throwOrLogError pass true to throw (or log) an error if invalid | |||||
| * @return {Boolean} true if schema is valid | |||||
| */ | |||||
| function validateSchema(schema, throwOrLogError) { | |||||
| var $schema = schema.$schema; | |||||
| if ($schema !== undefined && typeof $schema != 'string') | |||||
| throw new Error('$schema must be a string'); | |||||
| $schema = $schema || this._opts.defaultMeta || defaultMeta(this); | |||||
| if (!$schema) { | |||||
| this.logger.warn('meta-schema not available'); | |||||
| this.errors = null; | |||||
| return true; | |||||
| } | |||||
| var valid = this.validate($schema, schema); | |||||
| if (!valid && throwOrLogError) { | |||||
| var message = 'schema is invalid: ' + this.errorsText(); | |||||
| if (this._opts.validateSchema == 'log') this.logger.error(message); | |||||
| else throw new Error(message); | |||||
| } | |||||
| return valid; | |||||
| } | |||||
| function defaultMeta(self) { | |||||
| var meta = self._opts.meta; | |||||
| self._opts.defaultMeta = typeof meta == 'object' | |||||
| ? self._getId(meta) || meta | |||||
| : self.getSchema(META_SCHEMA_ID) | |||||
| ? META_SCHEMA_ID | |||||
| : undefined; | |||||
| return self._opts.defaultMeta; | |||||
| } | |||||
| /** | |||||
| * Get compiled schema from the instance by `key` or `ref`. | |||||
| * @this Ajv | |||||
| * @param {String} keyRef `key` that was passed to `addSchema` or full schema reference (`schema.id` or resolved id). | |||||
| * @return {Function} schema validating function (with property `schema`). | |||||
| */ | |||||
| function getSchema(keyRef) { | |||||
| var schemaObj = _getSchemaObj(this, keyRef); | |||||
| switch (typeof schemaObj) { | |||||
| case 'object': return schemaObj.validate || this._compile(schemaObj); | |||||
| case 'string': return this.getSchema(schemaObj); | |||||
| case 'undefined': return _getSchemaFragment(this, keyRef); | |||||
| } | |||||
| } | |||||
| function _getSchemaFragment(self, ref) { | |||||
| var res = resolve.schema.call(self, { schema: {} }, ref); | |||||
| if (res) { | |||||
| var schema = res.schema | |||||
| , root = res.root | |||||
| , baseId = res.baseId; | |||||
| var v = compileSchema.call(self, schema, root, undefined, baseId); | |||||
| self._fragments[ref] = new SchemaObject({ | |||||
| ref: ref, | |||||
| fragment: true, | |||||
| schema: schema, | |||||
| root: root, | |||||
| baseId: baseId, | |||||
| validate: v | |||||
| }); | |||||
| return v; | |||||
| } | |||||
| } | |||||
| function _getSchemaObj(self, keyRef) { | |||||
| keyRef = resolve.normalizeId(keyRef); | |||||
| return self._schemas[keyRef] || self._refs[keyRef] || self._fragments[keyRef]; | |||||
| } | |||||
| /** | |||||
| * Remove cached schema(s). | |||||
| * If no parameter is passed all schemas but meta-schemas are removed. | |||||
| * If RegExp is passed all schemas with key/id matching pattern but meta-schemas are removed. | |||||
| * Even if schema is referenced by other schemas it still can be removed as other schemas have local references. | |||||
| * @this Ajv | |||||
| * @param {String|Object|RegExp} schemaKeyRef key, ref, pattern to match key/ref or schema object | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function removeSchema(schemaKeyRef) { | |||||
| if (schemaKeyRef instanceof RegExp) { | |||||
| _removeAllSchemas(this, this._schemas, schemaKeyRef); | |||||
| _removeAllSchemas(this, this._refs, schemaKeyRef); | |||||
| return this; | |||||
| } | |||||
| switch (typeof schemaKeyRef) { | |||||
| case 'undefined': | |||||
| _removeAllSchemas(this, this._schemas); | |||||
| _removeAllSchemas(this, this._refs); | |||||
| this._cache.clear(); | |||||
| return this; | |||||
| case 'string': | |||||
| var schemaObj = _getSchemaObj(this, schemaKeyRef); | |||||
| if (schemaObj) this._cache.del(schemaObj.cacheKey); | |||||
| delete this._schemas[schemaKeyRef]; | |||||
| delete this._refs[schemaKeyRef]; | |||||
| return this; | |||||
| case 'object': | |||||
| var serialize = this._opts.serialize; | |||||
| var cacheKey = serialize ? serialize(schemaKeyRef) : schemaKeyRef; | |||||
| this._cache.del(cacheKey); | |||||
| var id = this._getId(schemaKeyRef); | |||||
| if (id) { | |||||
| id = resolve.normalizeId(id); | |||||
| delete this._schemas[id]; | |||||
| delete this._refs[id]; | |||||
| } | |||||
| } | |||||
| return this; | |||||
| } | |||||
| function _removeAllSchemas(self, schemas, regex) { | |||||
| for (var keyRef in schemas) { | |||||
| var schemaObj = schemas[keyRef]; | |||||
| if (!schemaObj.meta && (!regex || regex.test(keyRef))) { | |||||
| self._cache.del(schemaObj.cacheKey); | |||||
| delete schemas[keyRef]; | |||||
| } | |||||
| } | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function _addSchema(schema, skipValidation, meta, shouldAddSchema) { | |||||
| if (typeof schema != 'object' && typeof schema != 'boolean') | |||||
| throw new Error('schema should be object or boolean'); | |||||
| var serialize = this._opts.serialize; | |||||
| var cacheKey = serialize ? serialize(schema) : schema; | |||||
| var cached = this._cache.get(cacheKey); | |||||
| if (cached) return cached; | |||||
| shouldAddSchema = shouldAddSchema || this._opts.addUsedSchema !== false; | |||||
| var id = resolve.normalizeId(this._getId(schema)); | |||||
| if (id && shouldAddSchema) checkUnique(this, id); | |||||
| var willValidate = this._opts.validateSchema !== false && !skipValidation; | |||||
| var recursiveMeta; | |||||
| if (willValidate && !(recursiveMeta = id && id == resolve.normalizeId(schema.$schema))) | |||||
| this.validateSchema(schema, true); | |||||
| var localRefs = resolve.ids.call(this, schema); | |||||
| var schemaObj = new SchemaObject({ | |||||
| id: id, | |||||
| schema: schema, | |||||
| localRefs: localRefs, | |||||
| cacheKey: cacheKey, | |||||
| meta: meta | |||||
| }); | |||||
| if (id[0] != '#' && shouldAddSchema) this._refs[id] = schemaObj; | |||||
| this._cache.put(cacheKey, schemaObj); | |||||
| if (willValidate && recursiveMeta) this.validateSchema(schema, true); | |||||
| return schemaObj; | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function _compile(schemaObj, root) { | |||||
| if (schemaObj.compiling) { | |||||
| schemaObj.validate = callValidate; | |||||
| callValidate.schema = schemaObj.schema; | |||||
| callValidate.errors = null; | |||||
| callValidate.root = root ? root : callValidate; | |||||
| if (schemaObj.schema.$async === true) | |||||
| callValidate.$async = true; | |||||
| return callValidate; | |||||
| } | |||||
| schemaObj.compiling = true; | |||||
| var currentOpts; | |||||
| if (schemaObj.meta) { | |||||
| currentOpts = this._opts; | |||||
| this._opts = this._metaOpts; | |||||
| } | |||||
| var v; | |||||
| try { v = compileSchema.call(this, schemaObj.schema, root, schemaObj.localRefs); } | |||||
| catch(e) { | |||||
| delete schemaObj.validate; | |||||
| throw e; | |||||
| } | |||||
| finally { | |||||
| schemaObj.compiling = false; | |||||
| if (schemaObj.meta) this._opts = currentOpts; | |||||
| } | |||||
| schemaObj.validate = v; | |||||
| schemaObj.refs = v.refs; | |||||
| schemaObj.refVal = v.refVal; | |||||
| schemaObj.root = v.root; | |||||
| return v; | |||||
| /* @this {*} - custom context, see passContext option */ | |||||
| function callValidate() { | |||||
| /* jshint validthis: true */ | |||||
| var _validate = schemaObj.validate; | |||||
| var result = _validate.apply(this, arguments); | |||||
| callValidate.errors = _validate.errors; | |||||
| return result; | |||||
| } | |||||
| } | |||||
| function chooseGetId(opts) { | |||||
| switch (opts.schemaId) { | |||||
| case 'auto': return _get$IdOrId; | |||||
| case 'id': return _getId; | |||||
| default: return _get$Id; | |||||
| } | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function _getId(schema) { | |||||
| if (schema.$id) this.logger.warn('schema $id ignored', schema.$id); | |||||
| return schema.id; | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function _get$Id(schema) { | |||||
| if (schema.id) this.logger.warn('schema id ignored', schema.id); | |||||
| return schema.$id; | |||||
| } | |||||
| function _get$IdOrId(schema) { | |||||
| if (schema.$id && schema.id && schema.$id != schema.id) | |||||
| throw new Error('schema $id is different from id'); | |||||
| return schema.$id || schema.id; | |||||
| } | |||||
| /** | |||||
| * Convert array of error message objects to string | |||||
| * @this Ajv | |||||
| * @param {Array<Object>} errors optional array of validation errors, if not passed errors from the instance are used. | |||||
| * @param {Object} options optional options with properties `separator` and `dataVar`. | |||||
| * @return {String} human readable string with all errors descriptions | |||||
| */ | |||||
| function errorsText(errors, options) { | |||||
| errors = errors || this.errors; | |||||
| if (!errors) return 'No errors'; | |||||
| options = options || {}; | |||||
| var separator = options.separator === undefined ? ', ' : options.separator; | |||||
| var dataVar = options.dataVar === undefined ? 'data' : options.dataVar; | |||||
| var text = ''; | |||||
| for (var i=0; i<errors.length; i++) { | |||||
| var e = errors[i]; | |||||
| if (e) text += dataVar + e.dataPath + ' ' + e.message + separator; | |||||
| } | |||||
| return text.slice(0, -separator.length); | |||||
| } | |||||
| /** | |||||
| * Add custom format | |||||
| * @this Ajv | |||||
| * @param {String} name format name | |||||
| * @param {String|RegExp|Function} format string is converted to RegExp; function should return boolean (true when valid) | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function addFormat(name, format) { | |||||
| if (typeof format == 'string') format = new RegExp(format); | |||||
| this._formats[name] = format; | |||||
| return this; | |||||
| } | |||||
| function addDefaultMetaSchema(self) { | |||||
| var $dataSchema; | |||||
| if (self._opts.$data) { | |||||
| $dataSchema = require('./refs/data.json'); | |||||
| self.addMetaSchema($dataSchema, $dataSchema.$id, true); | |||||
| } | |||||
| if (self._opts.meta === false) return; | |||||
| var metaSchema = require('./refs/json-schema-draft-07.json'); | |||||
| if (self._opts.$data) metaSchema = $dataMetaSchema(metaSchema, META_SUPPORT_DATA); | |||||
| self.addMetaSchema(metaSchema, META_SCHEMA_ID, true); | |||||
| self._refs['http://json-schema.org/schema'] = META_SCHEMA_ID; | |||||
| } | |||||
| function addInitialSchemas(self) { | |||||
| var optsSchemas = self._opts.schemas; | |||||
| if (!optsSchemas) return; | |||||
| if (Array.isArray(optsSchemas)) self.addSchema(optsSchemas); | |||||
| else for (var key in optsSchemas) self.addSchema(optsSchemas[key], key); | |||||
| } | |||||
| function addInitialFormats(self) { | |||||
| for (var name in self._opts.formats) { | |||||
| var format = self._opts.formats[name]; | |||||
| self.addFormat(name, format); | |||||
| } | |||||
| } | |||||
| function addInitialKeywords(self) { | |||||
| for (var name in self._opts.keywords) { | |||||
| var keyword = self._opts.keywords[name]; | |||||
| self.addKeyword(name, keyword); | |||||
| } | |||||
| } | |||||
| function checkUnique(self, id) { | |||||
| if (self._schemas[id] || self._refs[id]) | |||||
| throw new Error('schema with key or id "' + id + '" already exists'); | |||||
| } | |||||
| function getMetaSchemaOptions(self) { | |||||
| var metaOpts = util.copy(self._opts); | |||||
| for (var i=0; i<META_IGNORE_OPTIONS.length; i++) | |||||
| delete metaOpts[META_IGNORE_OPTIONS[i]]; | |||||
| return metaOpts; | |||||
| } | |||||
| function setLogger(self) { | |||||
| var logger = self._opts.logger; | |||||
| if (logger === false) { | |||||
| self.logger = {log: noop, warn: noop, error: noop}; | |||||
| } else { | |||||
| if (logger === undefined) logger = console; | |||||
| if (!(typeof logger == 'object' && logger.log && logger.warn && logger.error)) | |||||
| throw new Error('logger must implement log, warn and error methods'); | |||||
| self.logger = logger; | |||||
| } | |||||
| } | |||||
| function noop() {} | |||||
| @ -0,0 +1,26 @@ | |||||
| 'use strict'; | |||||
| var Cache = module.exports = function Cache() { | |||||
| this._cache = {}; | |||||
| }; | |||||
| Cache.prototype.put = function Cache_put(key, value) { | |||||
| this._cache[key] = value; | |||||
| }; | |||||
| Cache.prototype.get = function Cache_get(key) { | |||||
| return this._cache[key]; | |||||
| }; | |||||
| Cache.prototype.del = function Cache_del(key) { | |||||
| delete this._cache[key]; | |||||
| }; | |||||
| Cache.prototype.clear = function Cache_clear() { | |||||
| this._cache = {}; | |||||
| }; | |||||
| @ -0,0 +1,90 @@ | |||||
| 'use strict'; | |||||
| var MissingRefError = require('./error_classes').MissingRef; | |||||
| module.exports = compileAsync; | |||||
| /** | |||||
| * Creates validating function for passed schema with asynchronous loading of missing schemas. | |||||
| * `loadSchema` option should be a function that accepts schema uri and returns promise that resolves with the schema. | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema object | |||||
| * @param {Boolean} meta optional true to compile meta-schema; this parameter can be skipped | |||||
| * @param {Function} callback an optional node-style callback, it is called with 2 parameters: error (or null) and validating function. | |||||
| * @return {Promise} promise that resolves with a validating function. | |||||
| */ | |||||
| function compileAsync(schema, meta, callback) { | |||||
| /* eslint no-shadow: 0 */ | |||||
| /* global Promise */ | |||||
| /* jshint validthis: true */ | |||||
| var self = this; | |||||
| if (typeof this._opts.loadSchema != 'function') | |||||
| throw new Error('options.loadSchema should be a function'); | |||||
| if (typeof meta == 'function') { | |||||
| callback = meta; | |||||
| meta = undefined; | |||||
| } | |||||
| var p = loadMetaSchemaOf(schema).then(function () { | |||||
| var schemaObj = self._addSchema(schema, undefined, meta); | |||||
| return schemaObj.validate || _compileAsync(schemaObj); | |||||
| }); | |||||
| if (callback) { | |||||
| p.then( | |||||
| function(v) { callback(null, v); }, | |||||
| callback | |||||
| ); | |||||
| } | |||||
| return p; | |||||
| function loadMetaSchemaOf(sch) { | |||||
| var $schema = sch.$schema; | |||||
| return $schema && !self.getSchema($schema) | |||||
| ? compileAsync.call(self, { $ref: $schema }, true) | |||||
| : Promise.resolve(); | |||||
| } | |||||
| function _compileAsync(schemaObj) { | |||||
| try { return self._compile(schemaObj); } | |||||
| catch(e) { | |||||
| if (e instanceof MissingRefError) return loadMissingSchema(e); | |||||
| throw e; | |||||
| } | |||||
| function loadMissingSchema(e) { | |||||
| var ref = e.missingSchema; | |||||
| if (added(ref)) throw new Error('Schema ' + ref + ' is loaded but ' + e.missingRef + ' cannot be resolved'); | |||||
| var schemaPromise = self._loadingSchemas[ref]; | |||||
| if (!schemaPromise) { | |||||
| schemaPromise = self._loadingSchemas[ref] = self._opts.loadSchema(ref); | |||||
| schemaPromise.then(removePromise, removePromise); | |||||
| } | |||||
| return schemaPromise.then(function (sch) { | |||||
| if (!added(ref)) { | |||||
| return loadMetaSchemaOf(sch).then(function () { | |||||
| if (!added(ref)) self.addSchema(sch, ref, undefined, meta); | |||||
| }); | |||||
| } | |||||
| }).then(function() { | |||||
| return _compileAsync(schemaObj); | |||||
| }); | |||||
| function removePromise() { | |||||
| delete self._loadingSchemas[ref]; | |||||
| } | |||||
| function added(ref) { | |||||
| return self._refs[ref] || self._schemas[ref]; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| @ -0,0 +1,5 @@ | |||||
| 'use strict'; | |||||
| // do NOT remove this file - it would break pre-compiled schemas | |||||
| // https://github.com/epoberezkin/ajv/issues/889 | |||||
| module.exports = require('fast-deep-equal'); | |||||
| @ -0,0 +1,34 @@ | |||||
| 'use strict'; | |||||
| var resolve = require('./resolve'); | |||||
| module.exports = { | |||||
| Validation: errorSubclass(ValidationError), | |||||
| MissingRef: errorSubclass(MissingRefError) | |||||
| }; | |||||
| function ValidationError(errors) { | |||||
| this.message = 'validation failed'; | |||||
| this.errors = errors; | |||||
| this.ajv = this.validation = true; | |||||
| } | |||||
| MissingRefError.message = function (baseId, ref) { | |||||
| return 'can\'t resolve reference ' + ref + ' from id ' + baseId; | |||||
| }; | |||||
| function MissingRefError(baseId, ref, message) { | |||||
| this.message = message || MissingRefError.message(baseId, ref); | |||||
| this.missingRef = resolve.url(baseId, ref); | |||||
| this.missingSchema = resolve.normalizeId(resolve.fullPath(this.missingRef)); | |||||
| } | |||||
| function errorSubclass(Subclass) { | |||||
| Subclass.prototype = Object.create(Error.prototype); | |||||
| Subclass.prototype.constructor = Subclass; | |||||
| return Subclass; | |||||
| } | |||||
| @ -0,0 +1,142 @@ | |||||
| 'use strict'; | |||||
| var util = require('./util'); | |||||
| var DATE = /^(\d\d\d\d)-(\d\d)-(\d\d)$/; | |||||
| var DAYS = [0,31,28,31,30,31,30,31,31,30,31,30,31]; | |||||
| var TIME = /^(\d\d):(\d\d):(\d\d)(\.\d+)?(z|[+-]\d\d(?::?\d\d)?)?$/i; | |||||
| var HOSTNAME = /^(?=.{1,253}\.?$)[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?(?:\.[a-z0-9](?:[-0-9a-z]{0,61}[0-9a-z])?)*\.?$/i; | |||||
| var URI = /^(?:[a-z][a-z0-9+\-.]*:)(?:\/?\/(?:(?:[a-z0-9\-._~!$&'()*+,;=:]|%[0-9a-f]{2})*@)?(?:\[(?:(?:(?:(?:[0-9a-f]{1,4}:){6}|::(?:[0-9a-f]{1,4}:){5}|(?:[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){4}|(?:(?:[0-9a-f]{1,4}:){0,1}[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){3}|(?:(?:[0-9a-f]{1,4}:){0,2}[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){2}|(?:(?:[0-9a-f]{1,4}:){0,3}[0-9a-f]{1,4})?::[0-9a-f]{1,4}:|(?:(?:[0-9a-f]{1,4}:){0,4}[0-9a-f]{1,4})?::)(?:[0-9a-f]{1,4}:[0-9a-f]{1,4}|(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?))|(?:(?:[0-9a-f]{1,4}:){0,5}[0-9a-f]{1,4})?::[0-9a-f]{1,4}|(?:(?:[0-9a-f]{1,4}:){0,6}[0-9a-f]{1,4})?::)|[Vv][0-9a-f]+\.[a-z0-9\-._~!$&'()*+,;=:]+)\]|(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?)|(?:[a-z0-9\-._~!$&'()*+,;=]|%[0-9a-f]{2})*)(?::\d*)?(?:\/(?:[a-z0-9\-._~!$&'()*+,;=:@]|%[0-9a-f]{2})*)*|\/(?:(?:[a-z0-9\-._~!$&'()*+,;=:@]|%[0-9a-f]{2})+(?:\/(?:[a-z0-9\-._~!$&'()*+,;=:@]|%[0-9a-f]{2})*)*)?|(?:[a-z0-9\-._~!$&'()*+,;=:@]|%[0-9a-f]{2})+(?:\/(?:[a-z0-9\-._~!$&'()*+,;=:@]|%[0-9a-f]{2})*)*)(?:\?(?:[a-z0-9\-._~!$&'()*+,;=:@/?]|%[0-9a-f]{2})*)?(?:#(?:[a-z0-9\-._~!$&'()*+,;=:@/?]|%[0-9a-f]{2})*)?$/i; | |||||
| var URIREF = /^(?:[a-z][a-z0-9+\-.]*:)?(?:\/?\/(?:(?:[a-z0-9\-._~!$&'()*+,;=:]|%[0-9a-f]{2})*@)?(?:\[(?:(?:(?:(?:[0-9a-f]{1,4}:){6}|::(?:[0-9a-f]{1,4}:){5}|(?:[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){4}|(?:(?:[0-9a-f]{1,4}:){0,1}[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){3}|(?:(?:[0-9a-f]{1,4}:){0,2}[0-9a-f]{1,4})?::(?:[0-9a-f]{1,4}:){2}|(?:(?:[0-9a-f]{1,4}:){0,3}[0-9a-f]{1,4})?::[0-9a-f]{1,4}:|(?:(?:[0-9a-f]{1,4}:){0,4}[0-9a-f]{1,4})?::)(?:[0-9a-f]{1,4}:[0-9a-f]{1,4}|(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?))|(?:(?:[0-9a-f]{1,4}:){0,5}[0-9a-f]{1,4})?::[0-9a-f]{1,4}|(?:(?:[0-9a-f]{1,4}:){0,6}[0-9a-f]{1,4})?::)|[Vv][0-9a-f]+\.[a-z0-9\-._~!$&'()*+,;=:]+)\]|(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?)|(?:[a-z0-9\-._~!$&'"()*+,;=]|%[0-9a-f]{2})*)(?::\d*)?(?:\/(?:[a-z0-9\-._~!$&'"()*+,;=:@]|%[0-9a-f]{2})*)*|\/(?:(?:[a-z0-9\-._~!$&'"()*+,;=:@]|%[0-9a-f]{2})+(?:\/(?:[a-z0-9\-._~!$&'"()*+,;=:@]|%[0-9a-f]{2})*)*)?|(?:[a-z0-9\-._~!$&'"()*+,;=:@]|%[0-9a-f]{2})+(?:\/(?:[a-z0-9\-._~!$&'"()*+,;=:@]|%[0-9a-f]{2})*)*)?(?:\?(?:[a-z0-9\-._~!$&'"()*+,;=:@/?]|%[0-9a-f]{2})*)?(?:#(?:[a-z0-9\-._~!$&'"()*+,;=:@/?]|%[0-9a-f]{2})*)?$/i; | |||||
| // uri-template: https://tools.ietf.org/html/rfc6570 | |||||
| var URITEMPLATE = /^(?:(?:[^\x00-\x20"'<>%\\^`{|}]|%[0-9a-f]{2})|\{[+#./;?&=,!@|]?(?:[a-z0-9_]|%[0-9a-f]{2})+(?::[1-9][0-9]{0,3}|\*)?(?:,(?:[a-z0-9_]|%[0-9a-f]{2})+(?::[1-9][0-9]{0,3}|\*)?)*\})*$/i; | |||||
| // For the source: https://gist.github.com/dperini/729294 | |||||
| // For test cases: https://mathiasbynens.be/demo/url-regex | |||||
| // @todo Delete current URL in favour of the commented out URL rule when this issue is fixed https://github.com/eslint/eslint/issues/7983. | |||||
| // var URL = /^(?:(?:https?|ftp):\/\/)(?:\S+(?::\S*)?@)?(?:(?!10(?:\.\d{1,3}){3})(?!127(?:\.\d{1,3}){3})(?!169\.254(?:\.\d{1,3}){2})(?!192\.168(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u{00a1}-\u{ffff}0-9]+-?)*[a-z\u{00a1}-\u{ffff}0-9]+)(?:\.(?:[a-z\u{00a1}-\u{ffff}0-9]+-?)*[a-z\u{00a1}-\u{ffff}0-9]+)*(?:\.(?:[a-z\u{00a1}-\u{ffff}]{2,})))(?::\d{2,5})?(?:\/[^\s]*)?$/iu; | |||||
| var URL = /^(?:(?:http[s\u017F]?|ftp):\/\/)(?:(?:[\0-\x08\x0E-\x1F!-\x9F\xA1-\u167F\u1681-\u1FFF\u200B-\u2027\u202A-\u202E\u2030-\u205E\u2060-\u2FFF\u3001-\uD7FF\uE000-\uFEFE\uFF00-\uFFFF]|[\uD800-\uDBFF][\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])+(?::(?:[\0-\x08\x0E-\x1F!-\x9F\xA1-\u167F\u1681-\u1FFF\u200B-\u2027\u202A-\u202E\u2030-\u205E\u2060-\u2FFF\u3001-\uD7FF\uE000-\uFEFE\uFF00-\uFFFF]|[\uD800-\uDBFF][\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])*)?@)?(?:(?!10(?:\.[0-9]{1,3}){3})(?!127(?:\.[0-9]{1,3}){3})(?!169\.254(?:\.[0-9]{1,3}){2})(?!192\.168(?:\.[0-9]{1,3}){2})(?!172\.(?:1[6-9]|2[0-9]|3[01])(?:\.[0-9]{1,3}){2})(?:[1-9][0-9]?|1[0-9][0-9]|2[01][0-9]|22[0-3])(?:\.(?:1?[0-9]{1,2}|2[0-4][0-9]|25[0-5])){2}(?:\.(?:[1-9][0-9]?|1[0-9][0-9]|2[0-4][0-9]|25[0-4]))|(?:(?:(?:[0-9KSa-z\xA1-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])+-?)*(?:[0-9KSa-z\xA1-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])+)(?:\.(?:(?:[0-9KSa-z\xA1-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])+-?)*(?:[0-9KSa-z\xA1-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])+)*(?:\.(?:(?:[KSa-z\xA1-\uD7FF\uE000-\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF]){2,})))(?::[0-9]{2,5})?(?:\/(?:[\0-\x08\x0E-\x1F!-\x9F\xA1-\u167F\u1681-\u1FFF\u200B-\u2027\u202A-\u202E\u2030-\u205E\u2060-\u2FFF\u3001-\uD7FF\uE000-\uFEFE\uFF00-\uFFFF]|[\uD800-\uDBFF][\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF])*)?$/i; | |||||
| var UUID = /^(?:urn:uuid:)?[0-9a-f]{8}-(?:[0-9a-f]{4}-){3}[0-9a-f]{12}$/i; | |||||
| var JSON_POINTER = /^(?:\/(?:[^~/]|~0|~1)*)*$/; | |||||
| var JSON_POINTER_URI_FRAGMENT = /^#(?:\/(?:[a-z0-9_\-.!$&'()*+,;:=@]|%[0-9a-f]{2}|~0|~1)*)*$/i; | |||||
| var RELATIVE_JSON_POINTER = /^(?:0|[1-9][0-9]*)(?:#|(?:\/(?:[^~/]|~0|~1)*)*)$/; | |||||
| module.exports = formats; | |||||
| function formats(mode) { | |||||
| mode = mode == 'full' ? 'full' : 'fast'; | |||||
| return util.copy(formats[mode]); | |||||
| } | |||||
| formats.fast = { | |||||
| // date: http://tools.ietf.org/html/rfc3339#section-5.6 | |||||
| date: /^\d\d\d\d-[0-1]\d-[0-3]\d$/, | |||||
| // date-time: http://tools.ietf.org/html/rfc3339#section-5.6 | |||||
| time: /^(?:[0-2]\d:[0-5]\d:[0-5]\d|23:59:60)(?:\.\d+)?(?:z|[+-]\d\d(?::?\d\d)?)?$/i, | |||||
| 'date-time': /^\d\d\d\d-[0-1]\d-[0-3]\d[t\s](?:[0-2]\d:[0-5]\d:[0-5]\d|23:59:60)(?:\.\d+)?(?:z|[+-]\d\d(?::?\d\d)?)$/i, | |||||
| // uri: https://github.com/mafintosh/is-my-json-valid/blob/master/formats.js | |||||
| uri: /^(?:[a-z][a-z0-9+-.]*:)(?:\/?\/)?[^\s]*$/i, | |||||
| 'uri-reference': /^(?:(?:[a-z][a-z0-9+-.]*:)?\/?\/)?(?:[^\\\s#][^\s#]*)?(?:#[^\\\s]*)?$/i, | |||||
| 'uri-template': URITEMPLATE, | |||||
| url: URL, | |||||
| // email (sources from jsen validator): | |||||
| // http://stackoverflow.com/questions/201323/using-a-regular-expression-to-validate-an-email-address#answer-8829363 | |||||
| // http://www.w3.org/TR/html5/forms.html#valid-e-mail-address (search for 'willful violation') | |||||
| email: /^[a-z0-9.!#$%&'*+/=?^_`{|}~-]+@[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?(?:\.[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?)*$/i, | |||||
| hostname: HOSTNAME, | |||||
| // optimized https://www.safaribooksonline.com/library/view/regular-expressions-cookbook/9780596802837/ch07s16.html | |||||
| ipv4: /^(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?)$/, | |||||
| // optimized http://stackoverflow.com/questions/53497/regular-expression-that-matches-valid-ipv6-addresses | |||||
| ipv6: /^\s*(?:(?:(?:[0-9a-f]{1,4}:){7}(?:[0-9a-f]{1,4}|:))|(?:(?:[0-9a-f]{1,4}:){6}(?::[0-9a-f]{1,4}|(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(?:(?:[0-9a-f]{1,4}:){5}(?:(?:(?::[0-9a-f]{1,4}){1,2})|:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(?:(?:[0-9a-f]{1,4}:){4}(?:(?:(?::[0-9a-f]{1,4}){1,3})|(?:(?::[0-9a-f]{1,4})?:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){3}(?:(?:(?::[0-9a-f]{1,4}){1,4})|(?:(?::[0-9a-f]{1,4}){0,2}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){2}(?:(?:(?::[0-9a-f]{1,4}){1,5})|(?:(?::[0-9a-f]{1,4}){0,3}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){1}(?:(?:(?::[0-9a-f]{1,4}){1,6})|(?:(?::[0-9a-f]{1,4}){0,4}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?::(?:(?:(?::[0-9a-f]{1,4}){1,7})|(?:(?::[0-9a-f]{1,4}){0,5}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(?:%.+)?\s*$/i, | |||||
| regex: regex, | |||||
| // uuid: http://tools.ietf.org/html/rfc4122 | |||||
| uuid: UUID, | |||||
| // JSON-pointer: https://tools.ietf.org/html/rfc6901 | |||||
| // uri fragment: https://tools.ietf.org/html/rfc3986#appendix-A | |||||
| 'json-pointer': JSON_POINTER, | |||||
| 'json-pointer-uri-fragment': JSON_POINTER_URI_FRAGMENT, | |||||
| // relative JSON-pointer: http://tools.ietf.org/html/draft-luff-relative-json-pointer-00 | |||||
| 'relative-json-pointer': RELATIVE_JSON_POINTER | |||||
| }; | |||||
| formats.full = { | |||||
| date: date, | |||||
| time: time, | |||||
| 'date-time': date_time, | |||||
| uri: uri, | |||||
| 'uri-reference': URIREF, | |||||
| 'uri-template': URITEMPLATE, | |||||
| url: URL, | |||||
| email: /^[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?$/i, | |||||
| hostname: HOSTNAME, | |||||
| ipv4: /^(?:(?:25[0-5]|2[0-4]\d|[01]?\d\d?)\.){3}(?:25[0-5]|2[0-4]\d|[01]?\d\d?)$/, | |||||
| ipv6: /^\s*(?:(?:(?:[0-9a-f]{1,4}:){7}(?:[0-9a-f]{1,4}|:))|(?:(?:[0-9a-f]{1,4}:){6}(?::[0-9a-f]{1,4}|(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(?:(?:[0-9a-f]{1,4}:){5}(?:(?:(?::[0-9a-f]{1,4}){1,2})|:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(?:(?:[0-9a-f]{1,4}:){4}(?:(?:(?::[0-9a-f]{1,4}){1,3})|(?:(?::[0-9a-f]{1,4})?:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){3}(?:(?:(?::[0-9a-f]{1,4}){1,4})|(?:(?::[0-9a-f]{1,4}){0,2}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){2}(?:(?:(?::[0-9a-f]{1,4}){1,5})|(?:(?::[0-9a-f]{1,4}){0,3}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?:(?:[0-9a-f]{1,4}:){1}(?:(?:(?::[0-9a-f]{1,4}){1,6})|(?:(?::[0-9a-f]{1,4}){0,4}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(?::(?:(?:(?::[0-9a-f]{1,4}){1,7})|(?:(?::[0-9a-f]{1,4}){0,5}:(?:(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(?:\.(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(?:%.+)?\s*$/i, | |||||
| regex: regex, | |||||
| uuid: UUID, | |||||
| 'json-pointer': JSON_POINTER, | |||||
| 'json-pointer-uri-fragment': JSON_POINTER_URI_FRAGMENT, | |||||
| 'relative-json-pointer': RELATIVE_JSON_POINTER | |||||
| }; | |||||
| function isLeapYear(year) { | |||||
| // https://tools.ietf.org/html/rfc3339#appendix-C | |||||
| return year % 4 === 0 && (year % 100 !== 0 || year % 400 === 0); | |||||
| } | |||||
| function date(str) { | |||||
| // full-date from http://tools.ietf.org/html/rfc3339#section-5.6 | |||||
| var matches = str.match(DATE); | |||||
| if (!matches) return false; | |||||
| var year = +matches[1]; | |||||
| var month = +matches[2]; | |||||
| var day = +matches[3]; | |||||
| return month >= 1 && month <= 12 && day >= 1 && | |||||
| day <= (month == 2 && isLeapYear(year) ? 29 : DAYS[month]); | |||||
| } | |||||
| function time(str, full) { | |||||
| var matches = str.match(TIME); | |||||
| if (!matches) return false; | |||||
| var hour = matches[1]; | |||||
| var minute = matches[2]; | |||||
| var second = matches[3]; | |||||
| var timeZone = matches[5]; | |||||
| return ((hour <= 23 && minute <= 59 && second <= 59) || | |||||
| (hour == 23 && minute == 59 && second == 60)) && | |||||
| (!full || timeZone); | |||||
| } | |||||
| var DATE_TIME_SEPARATOR = /t|\s/i; | |||||
| function date_time(str) { | |||||
| // http://tools.ietf.org/html/rfc3339#section-5.6 | |||||
| var dateTime = str.split(DATE_TIME_SEPARATOR); | |||||
| return dateTime.length == 2 && date(dateTime[0]) && time(dateTime[1], true); | |||||
| } | |||||
| var NOT_URI_FRAGMENT = /\/|:/; | |||||
| function uri(str) { | |||||
| // http://jmrware.com/articles/2009/uri_regexp/URI_regex.html + optional protocol + required "." | |||||
| return NOT_URI_FRAGMENT.test(str) && URI.test(str); | |||||
| } | |||||
| var Z_ANCHOR = /[^\\]\\Z/; | |||||
| function regex(str) { | |||||
| if (Z_ANCHOR.test(str)) return false; | |||||
| try { | |||||
| new RegExp(str); | |||||
| return true; | |||||
| } catch(e) { | |||||
| return false; | |||||
| } | |||||
| } | |||||
| @ -0,0 +1,387 @@ | |||||
| 'use strict'; | |||||
| var resolve = require('./resolve') | |||||
| , util = require('./util') | |||||
| , errorClasses = require('./error_classes') | |||||
| , stableStringify = require('fast-json-stable-stringify'); | |||||
| var validateGenerator = require('../dotjs/validate'); | |||||
| /** | |||||
| * Functions below are used inside compiled validations function | |||||
| */ | |||||
| var ucs2length = util.ucs2length; | |||||
| var equal = require('fast-deep-equal'); | |||||
| // this error is thrown by async schemas to return validation errors via exception | |||||
| var ValidationError = errorClasses.Validation; | |||||
| module.exports = compile; | |||||
| /** | |||||
| * Compiles schema to validation function | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema object | |||||
| * @param {Object} root object with information about the root schema for this schema | |||||
| * @param {Object} localRefs the hash of local references inside the schema (created by resolve.id), used for inline resolution | |||||
| * @param {String} baseId base ID for IDs in the schema | |||||
| * @return {Function} validation function | |||||
| */ | |||||
| function compile(schema, root, localRefs, baseId) { | |||||
| /* jshint validthis: true, evil: true */ | |||||
| /* eslint no-shadow: 0 */ | |||||
| var self = this | |||||
| , opts = this._opts | |||||
| , refVal = [ undefined ] | |||||
| , refs = {} | |||||
| , patterns = [] | |||||
| , patternsHash = {} | |||||
| , defaults = [] | |||||
| , defaultsHash = {} | |||||
| , customRules = []; | |||||
| root = root || { schema: schema, refVal: refVal, refs: refs }; | |||||
| var c = checkCompiling.call(this, schema, root, baseId); | |||||
| var compilation = this._compilations[c.index]; | |||||
| if (c.compiling) return (compilation.callValidate = callValidate); | |||||
| var formats = this._formats; | |||||
| var RULES = this.RULES; | |||||
| try { | |||||
| var v = localCompile(schema, root, localRefs, baseId); | |||||
| compilation.validate = v; | |||||
| var cv = compilation.callValidate; | |||||
| if (cv) { | |||||
| cv.schema = v.schema; | |||||
| cv.errors = null; | |||||
| cv.refs = v.refs; | |||||
| cv.refVal = v.refVal; | |||||
| cv.root = v.root; | |||||
| cv.$async = v.$async; | |||||
| if (opts.sourceCode) cv.source = v.source; | |||||
| } | |||||
| return v; | |||||
| } finally { | |||||
| endCompiling.call(this, schema, root, baseId); | |||||
| } | |||||
| /* @this {*} - custom context, see passContext option */ | |||||
| function callValidate() { | |||||
| /* jshint validthis: true */ | |||||
| var validate = compilation.validate; | |||||
| var result = validate.apply(this, arguments); | |||||
| callValidate.errors = validate.errors; | |||||
| return result; | |||||
| } | |||||
| function localCompile(_schema, _root, localRefs, baseId) { | |||||
| var isRoot = !_root || (_root && _root.schema == _schema); | |||||
| if (_root.schema != root.schema) | |||||
| return compile.call(self, _schema, _root, localRefs, baseId); | |||||
| var $async = _schema.$async === true; | |||||
| var sourceCode = validateGenerator({ | |||||
| isTop: true, | |||||
| schema: _schema, | |||||
| isRoot: isRoot, | |||||
| baseId: baseId, | |||||
| root: _root, | |||||
| schemaPath: '', | |||||
| errSchemaPath: '#', | |||||
| errorPath: '""', | |||||
| MissingRefError: errorClasses.MissingRef, | |||||
| RULES: RULES, | |||||
| validate: validateGenerator, | |||||
| util: util, | |||||
| resolve: resolve, | |||||
| resolveRef: resolveRef, | |||||
| usePattern: usePattern, | |||||
| useDefault: useDefault, | |||||
| useCustomRule: useCustomRule, | |||||
| opts: opts, | |||||
| formats: formats, | |||||
| logger: self.logger, | |||||
| self: self | |||||
| }); | |||||
| sourceCode = vars(refVal, refValCode) + vars(patterns, patternCode) | |||||
| + vars(defaults, defaultCode) + vars(customRules, customRuleCode) | |||||
| + sourceCode; | |||||
| if (opts.processCode) sourceCode = opts.processCode(sourceCode); | |||||
| // console.log('\n\n\n *** \n', JSON.stringify(sourceCode)); | |||||
| var validate; | |||||
| try { | |||||
| var makeValidate = new Function( | |||||
| 'self', | |||||
| 'RULES', | |||||
| 'formats', | |||||
| 'root', | |||||
| 'refVal', | |||||
| 'defaults', | |||||
| 'customRules', | |||||
| 'equal', | |||||
| 'ucs2length', | |||||
| 'ValidationError', | |||||
| sourceCode | |||||
| ); | |||||
| validate = makeValidate( | |||||
| self, | |||||
| RULES, | |||||
| formats, | |||||
| root, | |||||
| refVal, | |||||
| defaults, | |||||
| customRules, | |||||
| equal, | |||||
| ucs2length, | |||||
| ValidationError | |||||
| ); | |||||
| refVal[0] = validate; | |||||
| } catch(e) { | |||||
| self.logger.error('Error compiling schema, function code:', sourceCode); | |||||
| throw e; | |||||
| } | |||||
| validate.schema = _schema; | |||||
| validate.errors = null; | |||||
| validate.refs = refs; | |||||
| validate.refVal = refVal; | |||||
| validate.root = isRoot ? validate : _root; | |||||
| if ($async) validate.$async = true; | |||||
| if (opts.sourceCode === true) { | |||||
| validate.source = { | |||||
| code: sourceCode, | |||||
| patterns: patterns, | |||||
| defaults: defaults | |||||
| }; | |||||
| } | |||||
| return validate; | |||||
| } | |||||
| function resolveRef(baseId, ref, isRoot) { | |||||
| ref = resolve.url(baseId, ref); | |||||
| var refIndex = refs[ref]; | |||||
| var _refVal, refCode; | |||||
| if (refIndex !== undefined) { | |||||
| _refVal = refVal[refIndex]; | |||||
| refCode = 'refVal[' + refIndex + ']'; | |||||
| return resolvedRef(_refVal, refCode); | |||||
| } | |||||
| if (!isRoot && root.refs) { | |||||
| var rootRefId = root.refs[ref]; | |||||
| if (rootRefId !== undefined) { | |||||
| _refVal = root.refVal[rootRefId]; | |||||
| refCode = addLocalRef(ref, _refVal); | |||||
| return resolvedRef(_refVal, refCode); | |||||
| } | |||||
| } | |||||
| refCode = addLocalRef(ref); | |||||
| var v = resolve.call(self, localCompile, root, ref); | |||||
| if (v === undefined) { | |||||
| var localSchema = localRefs && localRefs[ref]; | |||||
| if (localSchema) { | |||||
| v = resolve.inlineRef(localSchema, opts.inlineRefs) | |||||
| ? localSchema | |||||
| : compile.call(self, localSchema, root, localRefs, baseId); | |||||
| } | |||||
| } | |||||
| if (v === undefined) { | |||||
| removeLocalRef(ref); | |||||
| } else { | |||||
| replaceLocalRef(ref, v); | |||||
| return resolvedRef(v, refCode); | |||||
| } | |||||
| } | |||||
| function addLocalRef(ref, v) { | |||||
| var refId = refVal.length; | |||||
| refVal[refId] = v; | |||||
| refs[ref] = refId; | |||||
| return 'refVal' + refId; | |||||
| } | |||||
| function removeLocalRef(ref) { | |||||
| delete refs[ref]; | |||||
| } | |||||
| function replaceLocalRef(ref, v) { | |||||
| var refId = refs[ref]; | |||||
| refVal[refId] = v; | |||||
| } | |||||
| function resolvedRef(refVal, code) { | |||||
| return typeof refVal == 'object' || typeof refVal == 'boolean' | |||||
| ? { code: code, schema: refVal, inline: true } | |||||
| : { code: code, $async: refVal && !!refVal.$async }; | |||||
| } | |||||
| function usePattern(regexStr) { | |||||
| var index = patternsHash[regexStr]; | |||||
| if (index === undefined) { | |||||
| index = patternsHash[regexStr] = patterns.length; | |||||
| patterns[index] = regexStr; | |||||
| } | |||||
| return 'pattern' + index; | |||||
| } | |||||
| function useDefault(value) { | |||||
| switch (typeof value) { | |||||
| case 'boolean': | |||||
| case 'number': | |||||
| return '' + value; | |||||
| case 'string': | |||||
| return util.toQuotedString(value); | |||||
| case 'object': | |||||
| if (value === null) return 'null'; | |||||
| var valueStr = stableStringify(value); | |||||
| var index = defaultsHash[valueStr]; | |||||
| if (index === undefined) { | |||||
| index = defaultsHash[valueStr] = defaults.length; | |||||
| defaults[index] = value; | |||||
| } | |||||
| return 'default' + index; | |||||
| } | |||||
| } | |||||
| function useCustomRule(rule, schema, parentSchema, it) { | |||||
| if (self._opts.validateSchema !== false) { | |||||
| var deps = rule.definition.dependencies; | |||||
| if (deps && !deps.every(function(keyword) { | |||||
| return Object.prototype.hasOwnProperty.call(parentSchema, keyword); | |||||
| })) | |||||
| throw new Error('parent schema must have all required keywords: ' + deps.join(',')); | |||||
| var validateSchema = rule.definition.validateSchema; | |||||
| if (validateSchema) { | |||||
| var valid = validateSchema(schema); | |||||
| if (!valid) { | |||||
| var message = 'keyword schema is invalid: ' + self.errorsText(validateSchema.errors); | |||||
| if (self._opts.validateSchema == 'log') self.logger.error(message); | |||||
| else throw new Error(message); | |||||
| } | |||||
| } | |||||
| } | |||||
| var compile = rule.definition.compile | |||||
| , inline = rule.definition.inline | |||||
| , macro = rule.definition.macro; | |||||
| var validate; | |||||
| if (compile) { | |||||
| validate = compile.call(self, schema, parentSchema, it); | |||||
| } else if (macro) { | |||||
| validate = macro.call(self, schema, parentSchema, it); | |||||
| if (opts.validateSchema !== false) self.validateSchema(validate, true); | |||||
| } else if (inline) { | |||||
| validate = inline.call(self, it, rule.keyword, schema, parentSchema); | |||||
| } else { | |||||
| validate = rule.definition.validate; | |||||
| if (!validate) return; | |||||
| } | |||||
| if (validate === undefined) | |||||
| throw new Error('custom keyword "' + rule.keyword + '"failed to compile'); | |||||
| var index = customRules.length; | |||||
| customRules[index] = validate; | |||||
| return { | |||||
| code: 'customRule' + index, | |||||
| validate: validate | |||||
| }; | |||||
| } | |||||
| } | |||||
| /** | |||||
| * Checks if the schema is currently compiled | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema to compile | |||||
| * @param {Object} root root object | |||||
| * @param {String} baseId base schema ID | |||||
| * @return {Object} object with properties "index" (compilation index) and "compiling" (boolean) | |||||
| */ | |||||
| function checkCompiling(schema, root, baseId) { | |||||
| /* jshint validthis: true */ | |||||
| var index = compIndex.call(this, schema, root, baseId); | |||||
| if (index >= 0) return { index: index, compiling: true }; | |||||
| index = this._compilations.length; | |||||
| this._compilations[index] = { | |||||
| schema: schema, | |||||
| root: root, | |||||
| baseId: baseId | |||||
| }; | |||||
| return { index: index, compiling: false }; | |||||
| } | |||||
| /** | |||||
| * Removes the schema from the currently compiled list | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema to compile | |||||
| * @param {Object} root root object | |||||
| * @param {String} baseId base schema ID | |||||
| */ | |||||
| function endCompiling(schema, root, baseId) { | |||||
| /* jshint validthis: true */ | |||||
| var i = compIndex.call(this, schema, root, baseId); | |||||
| if (i >= 0) this._compilations.splice(i, 1); | |||||
| } | |||||
| /** | |||||
| * Index of schema compilation in the currently compiled list | |||||
| * @this Ajv | |||||
| * @param {Object} schema schema to compile | |||||
| * @param {Object} root root object | |||||
| * @param {String} baseId base schema ID | |||||
| * @return {Integer} compilation index | |||||
| */ | |||||
| function compIndex(schema, root, baseId) { | |||||
| /* jshint validthis: true */ | |||||
| for (var i=0; i<this._compilations.length; i++) { | |||||
| var c = this._compilations[i]; | |||||
| if (c.schema == schema && c.root == root && c.baseId == baseId) return i; | |||||
| } | |||||
| return -1; | |||||
| } | |||||
| function patternCode(i, patterns) { | |||||
| return 'var pattern' + i + ' = new RegExp(' + util.toQuotedString(patterns[i]) + ');'; | |||||
| } | |||||
| function defaultCode(i) { | |||||
| return 'var default' + i + ' = defaults[' + i + '];'; | |||||
| } | |||||
| function refValCode(i, refVal) { | |||||
| return refVal[i] === undefined ? '' : 'var refVal' + i + ' = refVal[' + i + '];'; | |||||
| } | |||||
| function customRuleCode(i) { | |||||
| return 'var customRule' + i + ' = customRules[' + i + '];'; | |||||
| } | |||||
| function vars(arr, statement) { | |||||
| if (!arr.length) return ''; | |||||
| var code = ''; | |||||
| for (var i=0; i<arr.length; i++) | |||||
| code += statement(i, arr); | |||||
| return code; | |||||
| } | |||||
| @ -0,0 +1,270 @@ | |||||
| 'use strict'; | |||||
| var URI = require('uri-js') | |||||
| , equal = require('fast-deep-equal') | |||||
| , util = require('./util') | |||||
| , SchemaObject = require('./schema_obj') | |||||
| , traverse = require('json-schema-traverse'); | |||||
| module.exports = resolve; | |||||
| resolve.normalizeId = normalizeId; | |||||
| resolve.fullPath = getFullPath; | |||||
| resolve.url = resolveUrl; | |||||
| resolve.ids = resolveIds; | |||||
| resolve.inlineRef = inlineRef; | |||||
| resolve.schema = resolveSchema; | |||||
| /** | |||||
| * [resolve and compile the references ($ref)] | |||||
| * @this Ajv | |||||
| * @param {Function} compile reference to schema compilation funciton (localCompile) | |||||
| * @param {Object} root object with information about the root schema for the current schema | |||||
| * @param {String} ref reference to resolve | |||||
| * @return {Object|Function} schema object (if the schema can be inlined) or validation function | |||||
| */ | |||||
| function resolve(compile, root, ref) { | |||||
| /* jshint validthis: true */ | |||||
| var refVal = this._refs[ref]; | |||||
| if (typeof refVal == 'string') { | |||||
| if (this._refs[refVal]) refVal = this._refs[refVal]; | |||||
| else return resolve.call(this, compile, root, refVal); | |||||
| } | |||||
| refVal = refVal || this._schemas[ref]; | |||||
| if (refVal instanceof SchemaObject) { | |||||
| return inlineRef(refVal.schema, this._opts.inlineRefs) | |||||
| ? refVal.schema | |||||
| : refVal.validate || this._compile(refVal); | |||||
| } | |||||
| var res = resolveSchema.call(this, root, ref); | |||||
| var schema, v, baseId; | |||||
| if (res) { | |||||
| schema = res.schema; | |||||
| root = res.root; | |||||
| baseId = res.baseId; | |||||
| } | |||||
| if (schema instanceof SchemaObject) { | |||||
| v = schema.validate || compile.call(this, schema.schema, root, undefined, baseId); | |||||
| } else if (schema !== undefined) { | |||||
| v = inlineRef(schema, this._opts.inlineRefs) | |||||
| ? schema | |||||
| : compile.call(this, schema, root, undefined, baseId); | |||||
| } | |||||
| return v; | |||||
| } | |||||
| /** | |||||
| * Resolve schema, its root and baseId | |||||
| * @this Ajv | |||||
| * @param {Object} root root object with properties schema, refVal, refs | |||||
| * @param {String} ref reference to resolve | |||||
| * @return {Object} object with properties schema, root, baseId | |||||
| */ | |||||
| function resolveSchema(root, ref) { | |||||
| /* jshint validthis: true */ | |||||
| var p = URI.parse(ref) | |||||
| , refPath = _getFullPath(p) | |||||
| , baseId = getFullPath(this._getId(root.schema)); | |||||
| if (Object.keys(root.schema).length === 0 || refPath !== baseId) { | |||||
| var id = normalizeId(refPath); | |||||
| var refVal = this._refs[id]; | |||||
| if (typeof refVal == 'string') { | |||||
| return resolveRecursive.call(this, root, refVal, p); | |||||
| } else if (refVal instanceof SchemaObject) { | |||||
| if (!refVal.validate) this._compile(refVal); | |||||
| root = refVal; | |||||
| } else { | |||||
| refVal = this._schemas[id]; | |||||
| if (refVal instanceof SchemaObject) { | |||||
| if (!refVal.validate) this._compile(refVal); | |||||
| if (id == normalizeId(ref)) | |||||
| return { schema: refVal, root: root, baseId: baseId }; | |||||
| root = refVal; | |||||
| } else { | |||||
| return; | |||||
| } | |||||
| } | |||||
| if (!root.schema) return; | |||||
| baseId = getFullPath(this._getId(root.schema)); | |||||
| } | |||||
| return getJsonPointer.call(this, p, baseId, root.schema, root); | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function resolveRecursive(root, ref, parsedRef) { | |||||
| /* jshint validthis: true */ | |||||
| var res = resolveSchema.call(this, root, ref); | |||||
| if (res) { | |||||
| var schema = res.schema; | |||||
| var baseId = res.baseId; | |||||
| root = res.root; | |||||
| var id = this._getId(schema); | |||||
| if (id) baseId = resolveUrl(baseId, id); | |||||
| return getJsonPointer.call(this, parsedRef, baseId, schema, root); | |||||
| } | |||||
| } | |||||
| var PREVENT_SCOPE_CHANGE = util.toHash(['properties', 'patternProperties', 'enum', 'dependencies', 'definitions']); | |||||
| /* @this Ajv */ | |||||
| function getJsonPointer(parsedRef, baseId, schema, root) { | |||||
| /* jshint validthis: true */ | |||||
| parsedRef.fragment = parsedRef.fragment || ''; | |||||
| if (parsedRef.fragment.slice(0,1) != '/') return; | |||||
| var parts = parsedRef.fragment.split('/'); | |||||
| for (var i = 1; i < parts.length; i++) { | |||||
| var part = parts[i]; | |||||
| if (part) { | |||||
| part = util.unescapeFragment(part); | |||||
| schema = schema[part]; | |||||
| if (schema === undefined) break; | |||||
| var id; | |||||
| if (!PREVENT_SCOPE_CHANGE[part]) { | |||||
| id = this._getId(schema); | |||||
| if (id) baseId = resolveUrl(baseId, id); | |||||
| if (schema.$ref) { | |||||
| var $ref = resolveUrl(baseId, schema.$ref); | |||||
| var res = resolveSchema.call(this, root, $ref); | |||||
| if (res) { | |||||
| schema = res.schema; | |||||
| root = res.root; | |||||
| baseId = res.baseId; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if (schema !== undefined && schema !== root.schema) | |||||
| return { schema: schema, root: root, baseId: baseId }; | |||||
| } | |||||
| var SIMPLE_INLINED = util.toHash([ | |||||
| 'type', 'format', 'pattern', | |||||
| 'maxLength', 'minLength', | |||||
| 'maxProperties', 'minProperties', | |||||
| 'maxItems', 'minItems', | |||||
| 'maximum', 'minimum', | |||||
| 'uniqueItems', 'multipleOf', | |||||
| 'required', 'enum' | |||||
| ]); | |||||
| function inlineRef(schema, limit) { | |||||
| if (limit === false) return false; | |||||
| if (limit === undefined || limit === true) return checkNoRef(schema); | |||||
| else if (limit) return countKeys(schema) <= limit; | |||||
| } | |||||
| function checkNoRef(schema) { | |||||
| var item; | |||||
| if (Array.isArray(schema)) { | |||||
| for (var i=0; i<schema.length; i++) { | |||||
| item = schema[i]; | |||||
| if (typeof item == 'object' && !checkNoRef(item)) return false; | |||||
| } | |||||
| } else { | |||||
| for (var key in schema) { | |||||
| if (key == '$ref') return false; | |||||
| item = schema[key]; | |||||
| if (typeof item == 'object' && !checkNoRef(item)) return false; | |||||
| } | |||||
| } | |||||
| return true; | |||||
| } | |||||
| function countKeys(schema) { | |||||
| var count = 0, item; | |||||
| if (Array.isArray(schema)) { | |||||
| for (var i=0; i<schema.length; i++) { | |||||
| item = schema[i]; | |||||
| if (typeof item == 'object') count += countKeys(item); | |||||
| if (count == Infinity) return Infinity; | |||||
| } | |||||
| } else { | |||||
| for (var key in schema) { | |||||
| if (key == '$ref') return Infinity; | |||||
| if (SIMPLE_INLINED[key]) { | |||||
| count++; | |||||
| } else { | |||||
| item = schema[key]; | |||||
| if (typeof item == 'object') count += countKeys(item) + 1; | |||||
| if (count == Infinity) return Infinity; | |||||
| } | |||||
| } | |||||
| } | |||||
| return count; | |||||
| } | |||||
| function getFullPath(id, normalize) { | |||||
| if (normalize !== false) id = normalizeId(id); | |||||
| var p = URI.parse(id); | |||||
| return _getFullPath(p); | |||||
| } | |||||
| function _getFullPath(p) { | |||||
| return URI.serialize(p).split('#')[0] + '#'; | |||||
| } | |||||
| var TRAILING_SLASH_HASH = /#\/?$/; | |||||
| function normalizeId(id) { | |||||
| return id ? id.replace(TRAILING_SLASH_HASH, '') : ''; | |||||
| } | |||||
| function resolveUrl(baseId, id) { | |||||
| id = normalizeId(id); | |||||
| return URI.resolve(baseId, id); | |||||
| } | |||||
| /* @this Ajv */ | |||||
| function resolveIds(schema) { | |||||
| var schemaId = normalizeId(this._getId(schema)); | |||||
| var baseIds = {'': schemaId}; | |||||
| var fullPaths = {'': getFullPath(schemaId, false)}; | |||||
| var localRefs = {}; | |||||
| var self = this; | |||||
| traverse(schema, {allKeys: true}, function(sch, jsonPtr, rootSchema, parentJsonPtr, parentKeyword, parentSchema, keyIndex) { | |||||
| if (jsonPtr === '') return; | |||||
| var id = self._getId(sch); | |||||
| var baseId = baseIds[parentJsonPtr]; | |||||
| var fullPath = fullPaths[parentJsonPtr] + '/' + parentKeyword; | |||||
| if (keyIndex !== undefined) | |||||
| fullPath += '/' + (typeof keyIndex == 'number' ? keyIndex : util.escapeFragment(keyIndex)); | |||||
| if (typeof id == 'string') { | |||||
| id = baseId = normalizeId(baseId ? URI.resolve(baseId, id) : id); | |||||
| var refVal = self._refs[id]; | |||||
| if (typeof refVal == 'string') refVal = self._refs[refVal]; | |||||
| if (refVal && refVal.schema) { | |||||
| if (!equal(sch, refVal.schema)) | |||||
| throw new Error('id "' + id + '" resolves to more than one schema'); | |||||
| } else if (id != normalizeId(fullPath)) { | |||||
| if (id[0] == '#') { | |||||
| if (localRefs[id] && !equal(sch, localRefs[id])) | |||||
| throw new Error('id "' + id + '" resolves to more than one schema'); | |||||
| localRefs[id] = sch; | |||||
| } else { | |||||
| self._refs[id] = fullPath; | |||||
| } | |||||
| } | |||||
| } | |||||
| baseIds[jsonPtr] = baseId; | |||||
| fullPaths[jsonPtr] = fullPath; | |||||
| }); | |||||
| return localRefs; | |||||
| } | |||||
| @ -0,0 +1,66 @@ | |||||
| 'use strict'; | |||||
| var ruleModules = require('../dotjs') | |||||
| , toHash = require('./util').toHash; | |||||
| module.exports = function rules() { | |||||
| var RULES = [ | |||||
| { type: 'number', | |||||
| rules: [ { 'maximum': ['exclusiveMaximum'] }, | |||||
| { 'minimum': ['exclusiveMinimum'] }, 'multipleOf', 'format'] }, | |||||
| { type: 'string', | |||||
| rules: [ 'maxLength', 'minLength', 'pattern', 'format' ] }, | |||||
| { type: 'array', | |||||
| rules: [ 'maxItems', 'minItems', 'items', 'contains', 'uniqueItems' ] }, | |||||
| { type: 'object', | |||||
| rules: [ 'maxProperties', 'minProperties', 'required', 'dependencies', 'propertyNames', | |||||
| { 'properties': ['additionalProperties', 'patternProperties'] } ] }, | |||||
| { rules: [ '$ref', 'const', 'enum', 'not', 'anyOf', 'oneOf', 'allOf', 'if' ] } | |||||
| ]; | |||||
| var ALL = [ 'type', '$comment' ]; | |||||
| var KEYWORDS = [ | |||||
| '$schema', '$id', 'id', '$data', '$async', 'title', | |||||
| 'description', 'default', 'definitions', | |||||
| 'examples', 'readOnly', 'writeOnly', | |||||
| 'contentMediaType', 'contentEncoding', | |||||
| 'additionalItems', 'then', 'else' | |||||
| ]; | |||||
| var TYPES = [ 'number', 'integer', 'string', 'array', 'object', 'boolean', 'null' ]; | |||||
| RULES.all = toHash(ALL); | |||||
| RULES.types = toHash(TYPES); | |||||
| RULES.forEach(function (group) { | |||||
| group.rules = group.rules.map(function (keyword) { | |||||
| var implKeywords; | |||||
| if (typeof keyword == 'object') { | |||||
| var key = Object.keys(keyword)[0]; | |||||
| implKeywords = keyword[key]; | |||||
| keyword = key; | |||||
| implKeywords.forEach(function (k) { | |||||
| ALL.push(k); | |||||
| RULES.all[k] = true; | |||||
| }); | |||||
| } | |||||
| ALL.push(keyword); | |||||
| var rule = RULES.all[keyword] = { | |||||
| keyword: keyword, | |||||
| code: ruleModules[keyword], | |||||
| implements: implKeywords | |||||
| }; | |||||
| return rule; | |||||
| }); | |||||
| RULES.all.$comment = { | |||||
| keyword: '$comment', | |||||
| code: ruleModules.$comment | |||||
| }; | |||||
| if (group.type) RULES.types[group.type] = group; | |||||
| }); | |||||
| RULES.keywords = toHash(ALL.concat(KEYWORDS)); | |||||
| RULES.custom = {}; | |||||
| return RULES; | |||||
| }; | |||||
| @ -0,0 +1,9 @@ | |||||
| 'use strict'; | |||||
| var util = require('./util'); | |||||
| module.exports = SchemaObject; | |||||
| function SchemaObject(obj) { | |||||
| util.copy(obj, this); | |||||
| } | |||||
| @ -0,0 +1,20 @@ | |||||
| 'use strict'; | |||||
| // https://mathiasbynens.be/notes/javascript-encoding | |||||
| // https://github.com/bestiejs/punycode.js - punycode.ucs2.decode | |||||
| module.exports = function ucs2length(str) { | |||||
| var length = 0 | |||||
| , len = str.length | |||||
| , pos = 0 | |||||
| , value; | |||||
| while (pos < len) { | |||||
| length++; | |||||
| value = str.charCodeAt(pos++); | |||||
| if (value >= 0xD800 && value <= 0xDBFF && pos < len) { | |||||
| // high surrogate, and there is a next character | |||||
| value = str.charCodeAt(pos); | |||||
| if ((value & 0xFC00) == 0xDC00) pos++; // low surrogate | |||||
| } | |||||
| } | |||||
| return length; | |||||
| }; | |||||
| @ -0,0 +1,274 @@ | |||||
| 'use strict'; | |||||
| module.exports = { | |||||
| copy: copy, | |||||
| checkDataType: checkDataType, | |||||
| checkDataTypes: checkDataTypes, | |||||
| coerceToTypes: coerceToTypes, | |||||
| toHash: toHash, | |||||
| getProperty: getProperty, | |||||
| escapeQuotes: escapeQuotes, | |||||
| equal: require('fast-deep-equal'), | |||||
| ucs2length: require('./ucs2length'), | |||||
| varOccurences: varOccurences, | |||||
| varReplace: varReplace, | |||||
| cleanUpCode: cleanUpCode, | |||||
| finalCleanUpCode: finalCleanUpCode, | |||||
| schemaHasRules: schemaHasRules, | |||||
| schemaHasRulesExcept: schemaHasRulesExcept, | |||||
| schemaUnknownRules: schemaUnknownRules, | |||||
| toQuotedString: toQuotedString, | |||||
| getPathExpr: getPathExpr, | |||||
| getPath: getPath, | |||||
| getData: getData, | |||||
| unescapeFragment: unescapeFragment, | |||||
| unescapeJsonPointer: unescapeJsonPointer, | |||||
| escapeFragment: escapeFragment, | |||||
| escapeJsonPointer: escapeJsonPointer | |||||
| }; | |||||
| function copy(o, to) { | |||||
| to = to || {}; | |||||
| for (var key in o) to[key] = o[key]; | |||||
| return to; | |||||
| } | |||||
| function checkDataType(dataType, data, negate) { | |||||
| var EQUAL = negate ? ' !== ' : ' === ' | |||||
| , AND = negate ? ' || ' : ' && ' | |||||
| , OK = negate ? '!' : '' | |||||
| , NOT = negate ? '' : '!'; | |||||
| switch (dataType) { | |||||
| case 'null': return data + EQUAL + 'null'; | |||||
| case 'array': return OK + 'Array.isArray(' + data + ')'; | |||||
| case 'object': return '(' + OK + data + AND + | |||||
| 'typeof ' + data + EQUAL + '"object"' + AND + | |||||
| NOT + 'Array.isArray(' + data + '))'; | |||||
| case 'integer': return '(typeof ' + data + EQUAL + '"number"' + AND + | |||||
| NOT + '(' + data + ' % 1)' + | |||||
| AND + data + EQUAL + data + ')'; | |||||
| default: return 'typeof ' + data + EQUAL + '"' + dataType + '"'; | |||||
| } | |||||
| } | |||||
| function checkDataTypes(dataTypes, data) { | |||||
| switch (dataTypes.length) { | |||||
| case 1: return checkDataType(dataTypes[0], data, true); | |||||
| default: | |||||
| var code = ''; | |||||
| var types = toHash(dataTypes); | |||||
| if (types.array && types.object) { | |||||
| code = types.null ? '(': '(!' + data + ' || '; | |||||
| code += 'typeof ' + data + ' !== "object")'; | |||||
| delete types.null; | |||||
| delete types.array; | |||||
| delete types.object; | |||||
| } | |||||
| if (types.number) delete types.integer; | |||||
| for (var t in types) | |||||
| code += (code ? ' && ' : '' ) + checkDataType(t, data, true); | |||||
| return code; | |||||
| } | |||||
| } | |||||
| var COERCE_TO_TYPES = toHash([ 'string', 'number', 'integer', 'boolean', 'null' ]); | |||||
| function coerceToTypes(optionCoerceTypes, dataTypes) { | |||||
| if (Array.isArray(dataTypes)) { | |||||
| var types = []; | |||||
| for (var i=0; i<dataTypes.length; i++) { | |||||
| var t = dataTypes[i]; | |||||
| if (COERCE_TO_TYPES[t]) types[types.length] = t; | |||||
| else if (optionCoerceTypes === 'array' && t === 'array') types[types.length] = t; | |||||
| } | |||||
| if (types.length) return types; | |||||
| } else if (COERCE_TO_TYPES[dataTypes]) { | |||||
| return [dataTypes]; | |||||
| } else if (optionCoerceTypes === 'array' && dataTypes === 'array') { | |||||
| return ['array']; | |||||
| } | |||||
| } | |||||
| function toHash(arr) { | |||||
| var hash = {}; | |||||
| for (var i=0; i<arr.length; i++) hash[arr[i]] = true; | |||||
| return hash; | |||||
| } | |||||
| var IDENTIFIER = /^[a-z$_][a-z$_0-9]*$/i; | |||||
| var SINGLE_QUOTE = /'|\\/g; | |||||
| function getProperty(key) { | |||||
| return typeof key == 'number' | |||||
| ? '[' + key + ']' | |||||
| : IDENTIFIER.test(key) | |||||
| ? '.' + key | |||||
| : "['" + escapeQuotes(key) + "']"; | |||||
| } | |||||
| function escapeQuotes(str) { | |||||
| return str.replace(SINGLE_QUOTE, '\\$&') | |||||
| .replace(/\n/g, '\\n') | |||||
| .replace(/\r/g, '\\r') | |||||
| .replace(/\f/g, '\\f') | |||||
| .replace(/\t/g, '\\t'); | |||||
| } | |||||
| function varOccurences(str, dataVar) { | |||||
| dataVar += '[^0-9]'; | |||||
| var matches = str.match(new RegExp(dataVar, 'g')); | |||||
| return matches ? matches.length : 0; | |||||
| } | |||||
| function varReplace(str, dataVar, expr) { | |||||
| dataVar += '([^0-9])'; | |||||
| expr = expr.replace(/\$/g, '$$$$'); | |||||
| return str.replace(new RegExp(dataVar, 'g'), expr + '$1'); | |||||
| } | |||||
| var EMPTY_ELSE = /else\s*{\s*}/g | |||||
| , EMPTY_IF_NO_ELSE = /if\s*\([^)]+\)\s*\{\s*\}(?!\s*else)/g | |||||
| , EMPTY_IF_WITH_ELSE = /if\s*\(([^)]+)\)\s*\{\s*\}\s*else(?!\s*if)/g; | |||||
| function cleanUpCode(out) { | |||||
| return out.replace(EMPTY_ELSE, '') | |||||
| .replace(EMPTY_IF_NO_ELSE, '') | |||||
| .replace(EMPTY_IF_WITH_ELSE, 'if (!($1))'); | |||||
| } | |||||
| var ERRORS_REGEXP = /[^v.]errors/g | |||||
| , REMOVE_ERRORS = /var errors = 0;|var vErrors = null;|validate.errors = vErrors;/g | |||||
| , REMOVE_ERRORS_ASYNC = /var errors = 0;|var vErrors = null;/g | |||||
| , RETURN_VALID = 'return errors === 0;' | |||||
| , RETURN_TRUE = 'validate.errors = null; return true;' | |||||
| , RETURN_ASYNC = /if \(errors === 0\) return data;\s*else throw new ValidationError\(vErrors\);/ | |||||
| , RETURN_DATA_ASYNC = 'return data;' | |||||
| , ROOTDATA_REGEXP = /[^A-Za-z_$]rootData[^A-Za-z0-9_$]/g | |||||
| , REMOVE_ROOTDATA = /if \(rootData === undefined\) rootData = data;/; | |||||
| function finalCleanUpCode(out, async) { | |||||
| var matches = out.match(ERRORS_REGEXP); | |||||
| if (matches && matches.length == 2) { | |||||
| out = async | |||||
| ? out.replace(REMOVE_ERRORS_ASYNC, '') | |||||
| .replace(RETURN_ASYNC, RETURN_DATA_ASYNC) | |||||
| : out.replace(REMOVE_ERRORS, '') | |||||
| .replace(RETURN_VALID, RETURN_TRUE); | |||||
| } | |||||
| matches = out.match(ROOTDATA_REGEXP); | |||||
| if (!matches || matches.length !== 3) return out; | |||||
| return out.replace(REMOVE_ROOTDATA, ''); | |||||
| } | |||||
| function schemaHasRules(schema, rules) { | |||||
| if (typeof schema == 'boolean') return !schema; | |||||
| for (var key in schema) if (rules[key]) return true; | |||||
| } | |||||
| function schemaHasRulesExcept(schema, rules, exceptKeyword) { | |||||
| if (typeof schema == 'boolean') return !schema && exceptKeyword != 'not'; | |||||
| for (var key in schema) if (key != exceptKeyword && rules[key]) return true; | |||||
| } | |||||
| function schemaUnknownRules(schema, rules) { | |||||
| if (typeof schema == 'boolean') return; | |||||
| for (var key in schema) if (!rules[key]) return key; | |||||
| } | |||||
| function toQuotedString(str) { | |||||
| return '\'' + escapeQuotes(str) + '\''; | |||||
| } | |||||
| function getPathExpr(currentPath, expr, jsonPointers, isNumber) { | |||||
| var path = jsonPointers // false by default | |||||
| ? '\'/\' + ' + expr + (isNumber ? '' : '.replace(/~/g, \'~0\').replace(/\\//g, \'~1\')') | |||||
| : (isNumber ? '\'[\' + ' + expr + ' + \']\'' : '\'[\\\'\' + ' + expr + ' + \'\\\']\''); | |||||
| return joinPaths(currentPath, path); | |||||
| } | |||||
| function getPath(currentPath, prop, jsonPointers) { | |||||
| var path = jsonPointers // false by default | |||||
| ? toQuotedString('/' + escapeJsonPointer(prop)) | |||||
| : toQuotedString(getProperty(prop)); | |||||
| return joinPaths(currentPath, path); | |||||
| } | |||||
| var JSON_POINTER = /^\/(?:[^~]|~0|~1)*$/; | |||||
| var RELATIVE_JSON_POINTER = /^([0-9]+)(#|\/(?:[^~]|~0|~1)*)?$/; | |||||
| function getData($data, lvl, paths) { | |||||
| var up, jsonPointer, data, matches; | |||||
| if ($data === '') return 'rootData'; | |||||
| if ($data[0] == '/') { | |||||
| if (!JSON_POINTER.test($data)) throw new Error('Invalid JSON-pointer: ' + $data); | |||||
| jsonPointer = $data; | |||||
| data = 'rootData'; | |||||
| } else { | |||||
| matches = $data.match(RELATIVE_JSON_POINTER); | |||||
| if (!matches) throw new Error('Invalid JSON-pointer: ' + $data); | |||||
| up = +matches[1]; | |||||
| jsonPointer = matches[2]; | |||||
| if (jsonPointer == '#') { | |||||
| if (up >= lvl) throw new Error('Cannot access property/index ' + up + ' levels up, current level is ' + lvl); | |||||
| return paths[lvl - up]; | |||||
| } | |||||
| if (up > lvl) throw new Error('Cannot access data ' + up + ' levels up, current level is ' + lvl); | |||||
| data = 'data' + ((lvl - up) || ''); | |||||
| if (!jsonPointer) return data; | |||||
| } | |||||
| var expr = data; | |||||
| var segments = jsonPointer.split('/'); | |||||
| for (var i=0; i<segments.length; i++) { | |||||
| var segment = segments[i]; | |||||
| if (segment) { | |||||
| data += getProperty(unescapeJsonPointer(segment)); | |||||
| expr += ' && ' + data; | |||||
| } | |||||
| } | |||||
| return expr; | |||||
| } | |||||
| function joinPaths (a, b) { | |||||
| if (a == '""') return b; | |||||
| return (a + ' + ' + b).replace(/' \+ '/g, ''); | |||||
| } | |||||
| function unescapeFragment(str) { | |||||
| return unescapeJsonPointer(decodeURIComponent(str)); | |||||
| } | |||||
| function escapeFragment(str) { | |||||
| return encodeURIComponent(escapeJsonPointer(str)); | |||||
| } | |||||
| function escapeJsonPointer(str) { | |||||
| return str.replace(/~/g, '~0').replace(/\//g, '~1'); | |||||
| } | |||||
| function unescapeJsonPointer(str) { | |||||
| return str.replace(/~1/g, '/').replace(/~0/g, '~'); | |||||
| } | |||||
| @ -0,0 +1,49 @@ | |||||
| 'use strict'; | |||||
| var KEYWORDS = [ | |||||
| 'multipleOf', | |||||
| 'maximum', | |||||
| 'exclusiveMaximum', | |||||
| 'minimum', | |||||
| 'exclusiveMinimum', | |||||
| 'maxLength', | |||||
| 'minLength', | |||||
| 'pattern', | |||||
| 'additionalItems', | |||||
| 'maxItems', | |||||
| 'minItems', | |||||
| 'uniqueItems', | |||||
| 'maxProperties', | |||||
| 'minProperties', | |||||
| 'required', | |||||
| 'additionalProperties', | |||||
| 'enum', | |||||
| 'format', | |||||
| 'const' | |||||
| ]; | |||||
| module.exports = function (metaSchema, keywordsJsonPointers) { | |||||
| for (var i=0; i<keywordsJsonPointers.length; i++) { | |||||
| metaSchema = JSON.parse(JSON.stringify(metaSchema)); | |||||
| var segments = keywordsJsonPointers[i].split('/'); | |||||
| var keywords = metaSchema; | |||||
| var j; | |||||
| for (j=1; j<segments.length; j++) | |||||
| keywords = keywords[segments[j]]; | |||||
| for (j=0; j<KEYWORDS.length; j++) { | |||||
| var key = KEYWORDS[j]; | |||||
| var schema = keywords[key]; | |||||
| if (schema) { | |||||
| keywords[key] = { | |||||
| anyOf: [ | |||||
| schema, | |||||
| { $ref: 'https://raw.githubusercontent.com/epoberezkin/ajv/master/lib/refs/data.json#' } | |||||
| ] | |||||
| }; | |||||
| } | |||||
| } | |||||
| } | |||||
| return metaSchema; | |||||
| }; | |||||
| @ -0,0 +1,37 @@ | |||||
| 'use strict'; | |||||
| var metaSchema = require('./refs/json-schema-draft-07.json'); | |||||
| module.exports = { | |||||
| $id: 'https://github.com/epoberezkin/ajv/blob/master/lib/definition_schema.js', | |||||
| definitions: { | |||||
| simpleTypes: metaSchema.definitions.simpleTypes | |||||
| }, | |||||
| type: 'object', | |||||
| dependencies: { | |||||
| schema: ['validate'], | |||||
| $data: ['validate'], | |||||
| statements: ['inline'], | |||||
| valid: {not: {required: ['macro']}} | |||||
| }, | |||||
| properties: { | |||||
| type: metaSchema.properties.type, | |||||
| schema: {type: 'boolean'}, | |||||
| statements: {type: 'boolean'}, | |||||
| dependencies: { | |||||
| type: 'array', | |||||
| items: {type: 'string'} | |||||
| }, | |||||
| metaSchema: {type: 'object'}, | |||||
| modifying: {type: 'boolean'}, | |||||
| valid: {type: 'boolean'}, | |||||
| $data: {type: 'boolean'}, | |||||
| async: {type: 'boolean'}, | |||||
| errors: { | |||||
| anyOf: [ | |||||
| {type: 'boolean'}, | |||||
| {const: 'full'} | |||||
| ] | |||||
| } | |||||
| } | |||||
| }; | |||||
| @ -0,0 +1,104 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{## def.setExclusiveLimit: | |||||
| $exclusive = true; | |||||
| $errorKeyword = $exclusiveKeyword; | |||||
| $errSchemaPath = it.errSchemaPath + '/' + $exclusiveKeyword; | |||||
| #}} | |||||
| {{ | |||||
| var $isMax = $keyword == 'maximum' | |||||
| , $exclusiveKeyword = $isMax ? 'exclusiveMaximum' : 'exclusiveMinimum' | |||||
| , $schemaExcl = it.schema[$exclusiveKeyword] | |||||
| , $isDataExcl = it.opts.$data && $schemaExcl && $schemaExcl.$data | |||||
| , $op = $isMax ? '<' : '>' | |||||
| , $notOp = $isMax ? '>' : '<' | |||||
| , $errorKeyword = undefined; | |||||
| }} | |||||
| {{? $isDataExcl }} | |||||
| {{ | |||||
| var $schemaValueExcl = it.util.getData($schemaExcl.$data, $dataLvl, it.dataPathArr) | |||||
| , $exclusive = 'exclusive' + $lvl | |||||
| , $exclType = 'exclType' + $lvl | |||||
| , $exclIsNumber = 'exclIsNumber' + $lvl | |||||
| , $opExpr = 'op' + $lvl | |||||
| , $opStr = '\' + ' + $opExpr + ' + \''; | |||||
| }} | |||||
| var schemaExcl{{=$lvl}} = {{=$schemaValueExcl}}; | |||||
| {{ $schemaValueExcl = 'schemaExcl' + $lvl; }} | |||||
| var {{=$exclusive}}; | |||||
| var {{=$exclType}} = typeof {{=$schemaValueExcl}}; | |||||
| if ({{=$exclType}} != 'boolean' && {{=$exclType}} != 'undefined' && {{=$exclType}} != 'number') { | |||||
| {{ var $errorKeyword = $exclusiveKeyword; }} | |||||
| {{# def.error:'_exclusiveLimit' }} | |||||
| } else if ({{# def.$dataNotType:'number' }} | |||||
| {{=$exclType}} == 'number' | |||||
| ? ( | |||||
| ({{=$exclusive}} = {{=$schemaValue}} === undefined || {{=$schemaValueExcl}} {{=$op}}= {{=$schemaValue}}) | |||||
| ? {{=$data}} {{=$notOp}}= {{=$schemaValueExcl}} | |||||
| : {{=$data}} {{=$notOp}} {{=$schemaValue}} | |||||
| ) | |||||
| : ( | |||||
| ({{=$exclusive}} = {{=$schemaValueExcl}} === true) | |||||
| ? {{=$data}} {{=$notOp}}= {{=$schemaValue}} | |||||
| : {{=$data}} {{=$notOp}} {{=$schemaValue}} | |||||
| ) | |||||
| || {{=$data}} !== {{=$data}}) { | |||||
| var op{{=$lvl}} = {{=$exclusive}} ? '{{=$op}}' : '{{=$op}}='; | |||||
| {{ | |||||
| if ($schema === undefined) { | |||||
| $errorKeyword = $exclusiveKeyword; | |||||
| $errSchemaPath = it.errSchemaPath + '/' + $exclusiveKeyword; | |||||
| $schemaValue = $schemaValueExcl; | |||||
| $isData = $isDataExcl; | |||||
| } | |||||
| }} | |||||
| {{??}} | |||||
| {{ | |||||
| var $exclIsNumber = typeof $schemaExcl == 'number' | |||||
| , $opStr = $op; /*used in error*/ | |||||
| }} | |||||
| {{? $exclIsNumber && $isData }} | |||||
| {{ var $opExpr = '\'' + $opStr + '\''; /*used in error*/ }} | |||||
| if ({{# def.$dataNotType:'number' }} | |||||
| ( {{=$schemaValue}} === undefined | |||||
| || {{=$schemaExcl}} {{=$op}}= {{=$schemaValue}} | |||||
| ? {{=$data}} {{=$notOp}}= {{=$schemaExcl}} | |||||
| : {{=$data}} {{=$notOp}} {{=$schemaValue}} ) | |||||
| || {{=$data}} !== {{=$data}}) { | |||||
| {{??}} | |||||
| {{ | |||||
| if ($exclIsNumber && $schema === undefined) { | |||||
| {{# def.setExclusiveLimit }} | |||||
| $schemaValue = $schemaExcl; | |||||
| $notOp += '='; | |||||
| } else { | |||||
| if ($exclIsNumber) | |||||
| $schemaValue = Math[$isMax ? 'min' : 'max']($schemaExcl, $schema); | |||||
| if ($schemaExcl === ($exclIsNumber ? $schemaValue : true)) { | |||||
| {{# def.setExclusiveLimit }} | |||||
| $notOp += '='; | |||||
| } else { | |||||
| $exclusive = false; | |||||
| $opStr += '='; | |||||
| } | |||||
| } | |||||
| var $opExpr = '\'' + $opStr + '\''; /*used in error*/ | |||||
| }} | |||||
| if ({{# def.$dataNotType:'number' }} | |||||
| {{=$data}} {{=$notOp}} {{=$schemaValue}} | |||||
| || {{=$data}} !== {{=$data}}) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{ $errorKeyword = $errorKeyword || $keyword; }} | |||||
| {{# def.error:'_limit' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,10 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ var $op = $keyword == 'maxItems' ? '>' : '<'; }} | |||||
| if ({{# def.$dataNotType:'number' }} {{=$data}}.length {{=$op}} {{=$schemaValue}}) { | |||||
| {{ var $errorKeyword = $keyword; }} | |||||
| {{# def.error:'_limitItems' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,10 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ var $op = $keyword == 'maxLength' ? '>' : '<'; }} | |||||
| if ({{# def.$dataNotType:'number' }} {{# def.strLength }} {{=$op}} {{=$schemaValue}}) { | |||||
| {{ var $errorKeyword = $keyword; }} | |||||
| {{# def.error:'_limitLength' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,10 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ var $op = $keyword == 'maxProperties' ? '>' : '<'; }} | |||||
| if ({{# def.$dataNotType:'number' }} Object.keys({{=$data}}).length {{=$op}} {{=$schemaValue}}) { | |||||
| {{ var $errorKeyword = $keyword; }} | |||||
| {{# def.error:'_limitProperties' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,34 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| var $currentBaseId = $it.baseId | |||||
| , $allSchemasEmpty = true; | |||||
| }} | |||||
| {{~ $schema:$sch:$i }} | |||||
| {{? {{# def.nonEmptySchema:$sch }} }} | |||||
| {{ | |||||
| $allSchemasEmpty = false; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{? $breakOnError }} | |||||
| {{? $allSchemasEmpty }} | |||||
| if (true) { | |||||
| {{??}} | |||||
| {{= $closingBraces.slice(0,-1) }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,48 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| var $noEmptySchema = $schema.every(function($sch) { | |||||
| return {{# def.nonEmptySchema:$sch }}; | |||||
| }); | |||||
| }} | |||||
| {{? $noEmptySchema }} | |||||
| {{ var $currentBaseId = $it.baseId; }} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$valid}} = false; | |||||
| {{# def.setCompositeRule }} | |||||
| {{~ $schema:$sch:$i }} | |||||
| {{ | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| {{=$valid}} = {{=$valid}} || {{=$nextValid}}; | |||||
| if (!{{=$valid}}) { | |||||
| {{ $closingBraces += '}'; }} | |||||
| {{~}} | |||||
| {{# def.resetCompositeRule }} | |||||
| {{= $closingBraces }} | |||||
| if (!{{=$valid}}) { | |||||
| {{# def.extraError:'anyOf' }} | |||||
| } else { | |||||
| {{# def.resetErrors }} | |||||
| {{? it.opts.allErrors }} } {{?}} | |||||
| {{# def.cleanUp }} | |||||
| {{??}} | |||||
| {{? $breakOnError }} | |||||
| if (true) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,61 @@ | |||||
| {{## def.coerceType: | |||||
| {{ | |||||
| var $dataType = 'dataType' + $lvl | |||||
| , $coerced = 'coerced' + $lvl; | |||||
| }} | |||||
| var {{=$dataType}} = typeof {{=$data}}; | |||||
| {{? it.opts.coerceTypes == 'array'}} | |||||
| if ({{=$dataType}} == 'object' && Array.isArray({{=$data}})) {{=$dataType}} = 'array'; | |||||
| {{?}} | |||||
| var {{=$coerced}} = undefined; | |||||
| {{ var $bracesCoercion = ''; }} | |||||
| {{~ $coerceToTypes:$type:$i }} | |||||
| {{? $i }} | |||||
| if ({{=$coerced}} === undefined) { | |||||
| {{ $bracesCoercion += '}'; }} | |||||
| {{?}} | |||||
| {{? it.opts.coerceTypes == 'array' && $type != 'array' }} | |||||
| if ({{=$dataType}} == 'array' && {{=$data}}.length == 1) { | |||||
| {{=$coerced}} = {{=$data}} = {{=$data}}[0]; | |||||
| {{=$dataType}} = typeof {{=$data}}; | |||||
| /*if ({{=$dataType}} == 'object' && Array.isArray({{=$data}})) {{=$dataType}} = 'array';*/ | |||||
| } | |||||
| {{?}} | |||||
| {{? $type == 'string' }} | |||||
| if ({{=$dataType}} == 'number' || {{=$dataType}} == 'boolean') | |||||
| {{=$coerced}} = '' + {{=$data}}; | |||||
| else if ({{=$data}} === null) {{=$coerced}} = ''; | |||||
| {{?? $type == 'number' || $type == 'integer' }} | |||||
| if ({{=$dataType}} == 'boolean' || {{=$data}} === null | |||||
| || ({{=$dataType}} == 'string' && {{=$data}} && {{=$data}} == +{{=$data}} | |||||
| {{? $type == 'integer' }} && !({{=$data}} % 1){{?}})) | |||||
| {{=$coerced}} = +{{=$data}}; | |||||
| {{?? $type == 'boolean' }} | |||||
| if ({{=$data}} === 'false' || {{=$data}} === 0 || {{=$data}} === null) | |||||
| {{=$coerced}} = false; | |||||
| else if ({{=$data}} === 'true' || {{=$data}} === 1) | |||||
| {{=$coerced}} = true; | |||||
| {{?? $type == 'null' }} | |||||
| if ({{=$data}} === '' || {{=$data}} === 0 || {{=$data}} === false) | |||||
| {{=$coerced}} = null; | |||||
| {{?? it.opts.coerceTypes == 'array' && $type == 'array' }} | |||||
| if ({{=$dataType}} == 'string' || {{=$dataType}} == 'number' || {{=$dataType}} == 'boolean' || {{=$data}} == null) | |||||
| {{=$coerced}} = [{{=$data}}]; | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{= $bracesCoercion }} | |||||
| if ({{=$coerced}} === undefined) { | |||||
| {{# def.error:'type' }} | |||||
| } else { | |||||
| {{# def.setParentData }} | |||||
| {{=$data}} = {{=$coerced}}; | |||||
| {{? !$dataLvl }}if ({{=$parentData}} !== undefined){{?}} | |||||
| {{=$parentData}}[{{=$parentDataProperty}}] = {{=$coerced}}; | |||||
| } | |||||
| #}} | |||||
| @ -0,0 +1,9 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.setupKeyword }} | |||||
| {{ var $comment = it.util.toQuotedString($schema); }} | |||||
| {{? it.opts.$comment === true }} | |||||
| console.log({{=$comment}}); | |||||
| {{?? typeof it.opts.$comment == 'function' }} | |||||
| self._opts.$comment({{=$comment}}, {{=it.util.toQuotedString($errSchemaPath)}}, validate.root.schema); | |||||
| {{?}} | |||||
| @ -0,0 +1,11 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{? !$isData }} | |||||
| var schema{{=$lvl}} = validate.schema{{=$schemaPath}}; | |||||
| {{?}} | |||||
| var {{=$valid}} = equal({{=$data}}, schema{{=$lvl}}); | |||||
| {{# def.checkError:'const' }} | |||||
| {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,57 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| var $idx = 'i' + $lvl | |||||
| , $dataNxt = $it.dataLevel = it.dataLevel + 1 | |||||
| , $nextData = 'data' + $dataNxt | |||||
| , $currentBaseId = it.baseId | |||||
| , $nonEmptySchema = {{# def.nonEmptySchema:$schema }}; | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$valid}}; | |||||
| {{? $nonEmptySchema }} | |||||
| {{# def.setCompositeRule }} | |||||
| {{ | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| }} | |||||
| var {{=$nextValid}} = false; | |||||
| for (var {{=$idx}} = 0; {{=$idx}} < {{=$data}}.length; {{=$idx}}++) { | |||||
| {{ | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $idx, it.opts.jsonPointers, true); | |||||
| var $passData = $data + '[' + $idx + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $idx; | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| if ({{=$nextValid}}) break; | |||||
| } | |||||
| {{# def.resetCompositeRule }} | |||||
| {{= $closingBraces }} | |||||
| if (!{{=$nextValid}}) { | |||||
| {{??}} | |||||
| if ({{=$data}}.length == 0) { | |||||
| {{?}} | |||||
| {{# def.error:'contains' }} | |||||
| } else { | |||||
| {{? $nonEmptySchema }} | |||||
| {{# def.resetErrors }} | |||||
| {{?}} | |||||
| {{? it.opts.allErrors }} } {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,191 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ | |||||
| var $rule = this | |||||
| , $definition = 'definition' + $lvl | |||||
| , $rDef = $rule.definition | |||||
| , $closingBraces = ''; | |||||
| var $validate = $rDef.validate; | |||||
| var $compile, $inline, $macro, $ruleValidate, $validateCode; | |||||
| }} | |||||
| {{? $isData && $rDef.$data }} | |||||
| {{ | |||||
| $validateCode = 'keywordValidate' + $lvl; | |||||
| var $validateSchema = $rDef.validateSchema; | |||||
| }} | |||||
| var {{=$definition}} = RULES.custom['{{=$keyword}}'].definition; | |||||
| var {{=$validateCode}} = {{=$definition}}.validate; | |||||
| {{??}} | |||||
| {{ | |||||
| $ruleValidate = it.useCustomRule($rule, $schema, it.schema, it); | |||||
| if (!$ruleValidate) return; | |||||
| $schemaValue = 'validate.schema' + $schemaPath; | |||||
| $validateCode = $ruleValidate.code; | |||||
| $compile = $rDef.compile; | |||||
| $inline = $rDef.inline; | |||||
| $macro = $rDef.macro; | |||||
| }} | |||||
| {{?}} | |||||
| {{ | |||||
| var $ruleErrs = $validateCode + '.errors' | |||||
| , $i = 'i' + $lvl | |||||
| , $ruleErr = 'ruleErr' + $lvl | |||||
| , $asyncKeyword = $rDef.async; | |||||
| if ($asyncKeyword && !it.async) | |||||
| throw new Error('async keyword in sync schema'); | |||||
| }} | |||||
| {{? !($inline || $macro) }}{{=$ruleErrs}} = null;{{?}} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$valid}}; | |||||
| {{## def.callRuleValidate: | |||||
| {{=$validateCode}}.call( | |||||
| {{? it.opts.passContext }}this{{??}}self{{?}} | |||||
| {{? $compile || $rDef.schema === false }} | |||||
| , {{=$data}} | |||||
| {{??}} | |||||
| , {{=$schemaValue}} | |||||
| , {{=$data}} | |||||
| , validate.schema{{=it.schemaPath}} | |||||
| {{?}} | |||||
| , {{# def.dataPath }} | |||||
| {{# def.passParentData }} | |||||
| , rootData | |||||
| ) | |||||
| #}} | |||||
| {{## def.extendErrors:_inline: | |||||
| for (var {{=$i}}={{=$errs}}; {{=$i}}<errors; {{=$i}}++) { | |||||
| var {{=$ruleErr}} = vErrors[{{=$i}}]; | |||||
| if ({{=$ruleErr}}.dataPath === undefined) | |||||
| {{=$ruleErr}}.dataPath = (dataPath || '') + {{= it.errorPath }}; | |||||
| {{# _inline ? 'if (\{\{=$ruleErr\}\}.schemaPath === undefined) {' : '' }} | |||||
| {{=$ruleErr}}.schemaPath = "{{=$errSchemaPath}}"; | |||||
| {{# _inline ? '}' : '' }} | |||||
| {{? it.opts.verbose }} | |||||
| {{=$ruleErr}}.schema = {{=$schemaValue}}; | |||||
| {{=$ruleErr}}.data = {{=$data}}; | |||||
| {{?}} | |||||
| } | |||||
| #}} | |||||
| {{? $isData && $rDef.$data }} | |||||
| {{ $closingBraces += '}'; }} | |||||
| if ({{=$schemaValue}} === undefined) { | |||||
| {{=$valid}} = true; | |||||
| } else { | |||||
| {{? $validateSchema }} | |||||
| {{ $closingBraces += '}'; }} | |||||
| {{=$valid}} = {{=$definition}}.validateSchema({{=$schemaValue}}); | |||||
| if ({{=$valid}}) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $inline }} | |||||
| {{? $rDef.statements }} | |||||
| {{= $ruleValidate.validate }} | |||||
| {{??}} | |||||
| {{=$valid}} = {{= $ruleValidate.validate }}; | |||||
| {{?}} | |||||
| {{?? $macro }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| $it.schema = $ruleValidate.validate; | |||||
| $it.schemaPath = ''; | |||||
| }} | |||||
| {{# def.setCompositeRule }} | |||||
| {{ var $code = it.validate($it).replace(/validate\.schema/g, $validateCode); }} | |||||
| {{# def.resetCompositeRule }} | |||||
| {{= $code }} | |||||
| {{??}} | |||||
| {{# def.beginDefOut}} | |||||
| {{# def.callRuleValidate }} | |||||
| {{# def.storeDefOut:def_callRuleValidate }} | |||||
| {{? $rDef.errors === false }} | |||||
| {{=$valid}} = {{? $asyncKeyword }}await {{?}}{{= def_callRuleValidate }}; | |||||
| {{??}} | |||||
| {{? $asyncKeyword }} | |||||
| {{ $ruleErrs = 'customErrors' + $lvl; }} | |||||
| var {{=$ruleErrs}} = null; | |||||
| try { | |||||
| {{=$valid}} = await {{= def_callRuleValidate }}; | |||||
| } catch (e) { | |||||
| {{=$valid}} = false; | |||||
| if (e instanceof ValidationError) {{=$ruleErrs}} = e.errors; | |||||
| else throw e; | |||||
| } | |||||
| {{??}} | |||||
| {{=$ruleErrs}} = null; | |||||
| {{=$valid}} = {{= def_callRuleValidate }}; | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $rDef.modifying }} | |||||
| if ({{=$parentData}}) {{=$data}} = {{=$parentData}}[{{=$parentDataProperty}}]; | |||||
| {{?}} | |||||
| {{= $closingBraces }} | |||||
| {{## def.notValidationResult: | |||||
| {{? $rDef.valid === undefined }} | |||||
| !{{? $macro }}{{=$nextValid}}{{??}}{{=$valid}}{{?}} | |||||
| {{??}} | |||||
| {{= !$rDef.valid }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{? $rDef.valid }} | |||||
| {{? $breakOnError }} if (true) { {{?}} | |||||
| {{??}} | |||||
| if ({{# def.notValidationResult }}) { | |||||
| {{ $errorKeyword = $rule.keyword; }} | |||||
| {{# def.beginDefOut}} | |||||
| {{# def.error:'custom' }} | |||||
| {{# def.storeDefOut:def_customError }} | |||||
| {{? $inline }} | |||||
| {{? $rDef.errors }} | |||||
| {{? $rDef.errors != 'full' }} | |||||
| {{# def.extendErrors:true }} | |||||
| {{?}} | |||||
| {{??}} | |||||
| {{? $rDef.errors === false}} | |||||
| {{= def_customError }} | |||||
| {{??}} | |||||
| if ({{=$errs}} == errors) { | |||||
| {{= def_customError }} | |||||
| } else { | |||||
| {{# def.extendErrors:true }} | |||||
| } | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{?? $macro }} | |||||
| {{# def.extraError:'custom' }} | |||||
| {{??}} | |||||
| {{? $rDef.errors === false}} | |||||
| {{= def_customError }} | |||||
| {{??}} | |||||
| if (Array.isArray({{=$ruleErrs}})) { | |||||
| if (vErrors === null) vErrors = {{=$ruleErrs}}; | |||||
| else vErrors = vErrors.concat({{=$ruleErrs}}); | |||||
| errors = vErrors.length; | |||||
| {{# def.extendErrors:false }} | |||||
| } else { | |||||
| {{= def_customError }} | |||||
| } | |||||
| {{?}} | |||||
| {{?}} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,47 @@ | |||||
| {{## def.assignDefault: | |||||
| {{? it.compositeRule }} | |||||
| {{ | |||||
| if (it.opts.strictDefaults) { | |||||
| var $defaultMsg = 'default is ignored for: ' + $passData; | |||||
| if (it.opts.strictDefaults === 'log') it.logger.warn($defaultMsg); | |||||
| else throw new Error($defaultMsg); | |||||
| } | |||||
| }} | |||||
| {{??}} | |||||
| if ({{=$passData}} === undefined | |||||
| {{? it.opts.useDefaults == 'empty' }} | |||||
| || {{=$passData}} === null | |||||
| || {{=$passData}} === '' | |||||
| {{?}} | |||||
| ) | |||||
| {{=$passData}} = {{? it.opts.useDefaults == 'shared' }} | |||||
| {{= it.useDefault($sch.default) }} | |||||
| {{??}} | |||||
| {{= JSON.stringify($sch.default) }} | |||||
| {{?}}; | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.defaultProperties: | |||||
| {{ | |||||
| var $schema = it.schema.properties | |||||
| , $schemaKeys = Object.keys($schema); }} | |||||
| {{~ $schemaKeys:$propertyKey }} | |||||
| {{ var $sch = $schema[$propertyKey]; }} | |||||
| {{? $sch.default !== undefined }} | |||||
| {{ var $passData = $data + it.util.getProperty($propertyKey); }} | |||||
| {{# def.assignDefault }} | |||||
| {{?}} | |||||
| {{~}} | |||||
| #}} | |||||
| {{## def.defaultItems: | |||||
| {{~ it.schema.items:$sch:$i }} | |||||
| {{? $sch.default !== undefined }} | |||||
| {{ var $passData = $data + '[' + $i + ']'; }} | |||||
| {{# def.assignDefault }} | |||||
| {{?}} | |||||
| {{~}} | |||||
| #}} | |||||
| @ -0,0 +1,201 @@ | |||||
| {{## def.setupKeyword: | |||||
| {{ | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| }} | |||||
| #}} | |||||
| {{## def.setCompositeRule: | |||||
| {{ | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| }} | |||||
| #}} | |||||
| {{## def.resetCompositeRule: | |||||
| {{ it.compositeRule = $it.compositeRule = $wasComposite; }} | |||||
| #}} | |||||
| {{## def.setupNextLevel: | |||||
| {{ | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| }} | |||||
| #}} | |||||
| {{## def.ifValid: | |||||
| {{? $breakOnError }} | |||||
| if ({{=$valid}}) { | |||||
| {{ $closingBraces += '}'; }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.ifResultValid: | |||||
| {{? $breakOnError }} | |||||
| if ({{=$nextValid}}) { | |||||
| {{ $closingBraces += '}'; }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.elseIfValid: | |||||
| {{? $breakOnError }} | |||||
| {{ $closingBraces += '}'; }} | |||||
| else { | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.nonEmptySchema:_schema: | |||||
| (it.opts.strictKeywords | |||||
| ? typeof _schema == 'object' && Object.keys(_schema).length > 0 | |||||
| : it.util.schemaHasRules(_schema, it.RULES.all)) | |||||
| #}} | |||||
| {{## def.strLength: | |||||
| {{? it.opts.unicode === false }} | |||||
| {{=$data}}.length | |||||
| {{??}} | |||||
| ucs2length({{=$data}}) | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.willOptimize: | |||||
| it.util.varOccurences($code, $nextData) < 2 | |||||
| #}} | |||||
| {{## def.generateSubschemaCode: | |||||
| {{ | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| }} | |||||
| #}} | |||||
| {{## def.insertSubschemaCode: | |||||
| {{= it.validate($it) }} | |||||
| {{ $it.baseId = $currentBaseId; }} | |||||
| #}} | |||||
| {{## def._optimizeValidate: | |||||
| it.util.varReplace($code, $nextData, $passData) | |||||
| #}} | |||||
| {{## def.optimizeValidate: | |||||
| {{? {{# def.willOptimize}} }} | |||||
| {{= {{# def._optimizeValidate }} }} | |||||
| {{??}} | |||||
| var {{=$nextData}} = {{=$passData}}; | |||||
| {{= $code }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.cleanUp: {{ out = it.util.cleanUpCode(out); }} #}} | |||||
| {{## def.finalCleanUp: {{ out = it.util.finalCleanUpCode(out, $async); }} #}} | |||||
| {{## def.$data: | |||||
| {{ | |||||
| var $isData = it.opts.$data && $schema && $schema.$data | |||||
| , $schemaValue; | |||||
| }} | |||||
| {{? $isData }} | |||||
| var schema{{=$lvl}} = {{= it.util.getData($schema.$data, $dataLvl, it.dataPathArr) }}; | |||||
| {{ $schemaValue = 'schema' + $lvl; }} | |||||
| {{??}} | |||||
| {{ $schemaValue = $schema; }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.$dataNotType:_type: | |||||
| {{?$isData}} ({{=$schemaValue}} !== undefined && typeof {{=$schemaValue}} != _type) || {{?}} | |||||
| #}} | |||||
| {{## def.check$dataIsArray: | |||||
| if (schema{{=$lvl}} === undefined) {{=$valid}} = true; | |||||
| else if (!Array.isArray(schema{{=$lvl}})) {{=$valid}} = false; | |||||
| else { | |||||
| #}} | |||||
| {{## def.beginDefOut: | |||||
| {{ | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; | |||||
| }} | |||||
| #}} | |||||
| {{## def.storeDefOut:_variable: | |||||
| {{ | |||||
| var _variable = out; | |||||
| out = $$outStack.pop(); | |||||
| }} | |||||
| #}} | |||||
| {{## def.dataPath:(dataPath || ''){{? it.errorPath != '""'}} + {{= it.errorPath }}{{?}}#}} | |||||
| {{## def.setParentData: | |||||
| {{ | |||||
| var $parentData = $dataLvl ? 'data' + (($dataLvl-1)||'') : 'parentData' | |||||
| , $parentDataProperty = $dataLvl ? it.dataPathArr[$dataLvl] : 'parentDataProperty'; | |||||
| }} | |||||
| #}} | |||||
| {{## def.passParentData: | |||||
| {{# def.setParentData }} | |||||
| , {{= $parentData }} | |||||
| , {{= $parentDataProperty }} | |||||
| #}} | |||||
| {{## def.iterateProperties: | |||||
| {{? $ownProperties }} | |||||
| {{=$dataProperties}} = {{=$dataProperties}} || Object.keys({{=$data}}); | |||||
| for (var {{=$idx}}=0; {{=$idx}}<{{=$dataProperties}}.length; {{=$idx}}++) { | |||||
| var {{=$key}} = {{=$dataProperties}}[{{=$idx}}]; | |||||
| {{??}} | |||||
| for (var {{=$key}} in {{=$data}}) { | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.noPropertyInData: | |||||
| {{=$useData}} === undefined | |||||
| {{? $ownProperties }} | |||||
| || !{{# def.isOwnProperty }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.isOwnProperty: | |||||
| Object.prototype.hasOwnProperty.call({{=$data}}, '{{=it.util.escapeQuotes($propertyKey)}}') | |||||
| #}} | |||||
| @ -0,0 +1,80 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.missing }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{## def.propertyInData: | |||||
| {{=$data}}{{= it.util.getProperty($property) }} !== undefined | |||||
| {{? $ownProperties }} | |||||
| && Object.prototype.hasOwnProperty.call({{=$data}}, '{{=it.util.escapeQuotes($property)}}') | |||||
| {{?}} | |||||
| #}} | |||||
| {{ | |||||
| var $schemaDeps = {} | |||||
| , $propertyDeps = {} | |||||
| , $ownProperties = it.opts.ownProperties; | |||||
| for ($property in $schema) { | |||||
| var $sch = $schema[$property]; | |||||
| var $deps = Array.isArray($sch) ? $propertyDeps : $schemaDeps; | |||||
| $deps[$property] = $sch; | |||||
| } | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| {{ var $currentErrorPath = it.errorPath; }} | |||||
| var missing{{=$lvl}}; | |||||
| {{ for (var $property in $propertyDeps) { }} | |||||
| {{ $deps = $propertyDeps[$property]; }} | |||||
| {{? $deps.length }} | |||||
| if ({{# def.propertyInData }} | |||||
| {{? $breakOnError }} | |||||
| && ({{# def.checkMissingProperty:$deps }})) { | |||||
| {{# def.errorMissingProperty:'dependencies' }} | |||||
| {{??}} | |||||
| ) { | |||||
| {{~ $deps:$propertyKey }} | |||||
| {{# def.allErrorsMissingProperty:'dependencies' }} | |||||
| {{~}} | |||||
| {{?}} | |||||
| } {{# def.elseIfValid }} | |||||
| {{?}} | |||||
| {{ } }} | |||||
| {{ | |||||
| it.errorPath = $currentErrorPath; | |||||
| var $currentBaseId = $it.baseId; | |||||
| }} | |||||
| {{ for (var $property in $schemaDeps) { }} | |||||
| {{ var $sch = $schemaDeps[$property]; }} | |||||
| {{? {{# def.nonEmptySchema:$sch }} }} | |||||
| {{=$nextValid}} = true; | |||||
| if ({{# def.propertyInData }}) { | |||||
| {{ | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + it.util.getProperty($property); | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + it.util.escapeFragment($property); | |||||
| }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| } | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} | |||||
| {{ } }} | |||||
| {{? $breakOnError }} | |||||
| {{= $closingBraces }} | |||||
| if ({{=$errs}} == errors) { | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,30 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ | |||||
| var $i = 'i' + $lvl | |||||
| , $vSchema = 'schema' + $lvl; | |||||
| }} | |||||
| {{? !$isData }} | |||||
| var {{=$vSchema}} = validate.schema{{=$schemaPath}}; | |||||
| {{?}} | |||||
| var {{=$valid}}; | |||||
| {{?$isData}}{{# def.check$dataIsArray }}{{?}} | |||||
| {{=$valid}} = false; | |||||
| for (var {{=$i}}=0; {{=$i}}<{{=$vSchema}}.length; {{=$i}}++) | |||||
| if (equal({{=$data}}, {{=$vSchema}}[{{=$i}}])) { | |||||
| {{=$valid}} = true; | |||||
| break; | |||||
| } | |||||
| {{? $isData }} } {{?}} | |||||
| {{# def.checkError:'enum' }} | |||||
| {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,194 @@ | |||||
| {{# def.definitions }} | |||||
| {{## def._error:_rule: | |||||
| {{ 'istanbul ignore else'; }} | |||||
| {{? it.createErrors !== false }} | |||||
| { | |||||
| keyword: '{{= $errorKeyword || _rule }}' | |||||
| , dataPath: (dataPath || '') + {{= it.errorPath }} | |||||
| , schemaPath: {{=it.util.toQuotedString($errSchemaPath)}} | |||||
| , params: {{# def._errorParams[_rule] }} | |||||
| {{? it.opts.messages !== false }} | |||||
| , message: {{# def._errorMessages[_rule] }} | |||||
| {{?}} | |||||
| {{? it.opts.verbose }} | |||||
| , schema: {{# def._errorSchemas[_rule] }} | |||||
| , parentSchema: validate.schema{{=it.schemaPath}} | |||||
| , data: {{=$data}} | |||||
| {{?}} | |||||
| } | |||||
| {{??}} | |||||
| {} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def._addError:_rule: | |||||
| if (vErrors === null) vErrors = [err]; | |||||
| else vErrors.push(err); | |||||
| errors++; | |||||
| #}} | |||||
| {{## def.addError:_rule: | |||||
| var err = {{# def._error:_rule }}; | |||||
| {{# def._addError:_rule }} | |||||
| #}} | |||||
| {{## def.error:_rule: | |||||
| {{# def.beginDefOut}} | |||||
| {{# def._error:_rule }} | |||||
| {{# def.storeDefOut:__err }} | |||||
| {{? !it.compositeRule && $breakOnError }} | |||||
| {{ 'istanbul ignore if'; }} | |||||
| {{? it.async }} | |||||
| throw new ValidationError([{{=__err}}]); | |||||
| {{??}} | |||||
| validate.errors = [{{=__err}}]; | |||||
| return false; | |||||
| {{?}} | |||||
| {{??}} | |||||
| var err = {{=__err}}; | |||||
| {{# def._addError:_rule }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.extraError:_rule: | |||||
| {{# def.addError:_rule}} | |||||
| {{? !it.compositeRule && $breakOnError }} | |||||
| {{ 'istanbul ignore if'; }} | |||||
| {{? it.async }} | |||||
| throw new ValidationError(vErrors); | |||||
| {{??}} | |||||
| validate.errors = vErrors; | |||||
| return false; | |||||
| {{?}} | |||||
| {{?}} | |||||
| #}} | |||||
| {{## def.checkError:_rule: | |||||
| if (!{{=$valid}}) { | |||||
| {{# def.error:_rule }} | |||||
| } | |||||
| #}} | |||||
| {{## def.resetErrors: | |||||
| errors = {{=$errs}}; | |||||
| if (vErrors !== null) { | |||||
| if ({{=$errs}}) vErrors.length = {{=$errs}}; | |||||
| else vErrors = null; | |||||
| } | |||||
| #}} | |||||
| {{## def.concatSchema:{{?$isData}}' + {{=$schemaValue}} + '{{??}}{{=$schema}}{{?}}#}} | |||||
| {{## def.appendSchema:{{?$isData}}' + {{=$schemaValue}}{{??}}{{=$schemaValue}}'{{?}}#}} | |||||
| {{## def.concatSchemaEQ:{{?$isData}}' + {{=$schemaValue}} + '{{??}}{{=it.util.escapeQuotes($schema)}}{{?}}#}} | |||||
| {{## def._errorMessages = { | |||||
| 'false schema': "'boolean schema is false'", | |||||
| $ref: "'can\\\'t resolve reference {{=it.util.escapeQuotes($schema)}}'", | |||||
| additionalItems: "'should NOT have more than {{=$schema.length}} items'", | |||||
| additionalProperties: "'{{? it.opts._errorDataPathProperty }}is an invalid additional property{{??}}should NOT have additional properties{{?}}'", | |||||
| anyOf: "'should match some schema in anyOf'", | |||||
| const: "'should be equal to constant'", | |||||
| contains: "'should contain a valid item'", | |||||
| dependencies: "'should have {{? $deps.length == 1 }}property {{= it.util.escapeQuotes($deps[0]) }}{{??}}properties {{= it.util.escapeQuotes($deps.join(\", \")) }}{{?}} when property {{= it.util.escapeQuotes($property) }} is present'", | |||||
| 'enum': "'should be equal to one of the allowed values'", | |||||
| format: "'should match format \"{{#def.concatSchemaEQ}}\"'", | |||||
| 'if': "'should match \"' + {{=$ifClause}} + '\" schema'", | |||||
| _limit: "'should be {{=$opStr}} {{#def.appendSchema}}", | |||||
| _exclusiveLimit: "'{{=$exclusiveKeyword}} should be boolean'", | |||||
| _limitItems: "'should NOT have {{?$keyword=='maxItems'}}more{{??}}fewer{{?}} than {{#def.concatSchema}} items'", | |||||
| _limitLength: "'should NOT be {{?$keyword=='maxLength'}}longer{{??}}shorter{{?}} than {{#def.concatSchema}} characters'", | |||||
| _limitProperties:"'should NOT have {{?$keyword=='maxProperties'}}more{{??}}fewer{{?}} than {{#def.concatSchema}} properties'", | |||||
| multipleOf: "'should be multiple of {{#def.appendSchema}}", | |||||
| not: "'should NOT be valid'", | |||||
| oneOf: "'should match exactly one schema in oneOf'", | |||||
| pattern: "'should match pattern \"{{#def.concatSchemaEQ}}\"'", | |||||
| propertyNames: "'property name \\'{{=$invalidName}}\\' is invalid'", | |||||
| required: "'{{? it.opts._errorDataPathProperty }}is a required property{{??}}should have required property \\'{{=$missingProperty}}\\'{{?}}'", | |||||
| type: "'should be {{? $typeIsArray }}{{= $typeSchema.join(\",\") }}{{??}}{{=$typeSchema}}{{?}}'", | |||||
| uniqueItems: "'should NOT have duplicate items (items ## ' + j + ' and ' + i + ' are identical)'", | |||||
| custom: "'should pass \"{{=$rule.keyword}}\" keyword validation'", | |||||
| patternRequired: "'should have property matching pattern \\'{{=$missingPattern}}\\''", | |||||
| switch: "'should pass \"switch\" keyword validation'", | |||||
| _formatLimit: "'should be {{=$opStr}} \"{{#def.concatSchemaEQ}}\"'", | |||||
| _formatExclusiveLimit: "'{{=$exclusiveKeyword}} should be boolean'" | |||||
| } #}} | |||||
| {{## def.schemaRefOrVal: {{?$isData}}validate.schema{{=$schemaPath}}{{??}}{{=$schema}}{{?}} #}} | |||||
| {{## def.schemaRefOrQS: {{?$isData}}validate.schema{{=$schemaPath}}{{??}}{{=it.util.toQuotedString($schema)}}{{?}} #}} | |||||
| {{## def._errorSchemas = { | |||||
| 'false schema': "false", | |||||
| $ref: "{{=it.util.toQuotedString($schema)}}", | |||||
| additionalItems: "false", | |||||
| additionalProperties: "false", | |||||
| anyOf: "validate.schema{{=$schemaPath}}", | |||||
| const: "validate.schema{{=$schemaPath}}", | |||||
| contains: "validate.schema{{=$schemaPath}}", | |||||
| dependencies: "validate.schema{{=$schemaPath}}", | |||||
| 'enum': "validate.schema{{=$schemaPath}}", | |||||
| format: "{{#def.schemaRefOrQS}}", | |||||
| 'if': "validate.schema{{=$schemaPath}}", | |||||
| _limit: "{{#def.schemaRefOrVal}}", | |||||
| _exclusiveLimit: "validate.schema{{=$schemaPath}}", | |||||
| _limitItems: "{{#def.schemaRefOrVal}}", | |||||
| _limitLength: "{{#def.schemaRefOrVal}}", | |||||
| _limitProperties:"{{#def.schemaRefOrVal}}", | |||||
| multipleOf: "{{#def.schemaRefOrVal}}", | |||||
| not: "validate.schema{{=$schemaPath}}", | |||||
| oneOf: "validate.schema{{=$schemaPath}}", | |||||
| pattern: "{{#def.schemaRefOrQS}}", | |||||
| propertyNames: "validate.schema{{=$schemaPath}}", | |||||
| required: "validate.schema{{=$schemaPath}}", | |||||
| type: "validate.schema{{=$schemaPath}}", | |||||
| uniqueItems: "{{#def.schemaRefOrVal}}", | |||||
| custom: "validate.schema{{=$schemaPath}}", | |||||
| patternRequired: "validate.schema{{=$schemaPath}}", | |||||
| switch: "validate.schema{{=$schemaPath}}", | |||||
| _formatLimit: "{{#def.schemaRefOrQS}}", | |||||
| _formatExclusiveLimit: "validate.schema{{=$schemaPath}}" | |||||
| } #}} | |||||
| {{## def.schemaValueQS: {{?$isData}}{{=$schemaValue}}{{??}}{{=it.util.toQuotedString($schema)}}{{?}} #}} | |||||
| {{## def._errorParams = { | |||||
| 'false schema': "{}", | |||||
| $ref: "{ ref: '{{=it.util.escapeQuotes($schema)}}' }", | |||||
| additionalItems: "{ limit: {{=$schema.length}} }", | |||||
| additionalProperties: "{ additionalProperty: '{{=$additionalProperty}}' }", | |||||
| anyOf: "{}", | |||||
| const: "{ allowedValue: schema{{=$lvl}} }", | |||||
| contains: "{}", | |||||
| dependencies: "{ property: '{{= it.util.escapeQuotes($property) }}', missingProperty: '{{=$missingProperty}}', depsCount: {{=$deps.length}}, deps: '{{= it.util.escapeQuotes($deps.length==1 ? $deps[0] : $deps.join(\", \")) }}' }", | |||||
| 'enum': "{ allowedValues: schema{{=$lvl}} }", | |||||
| format: "{ format: {{#def.schemaValueQS}} }", | |||||
| 'if': "{ failingKeyword: {{=$ifClause}} }", | |||||
| _limit: "{ comparison: {{=$opExpr}}, limit: {{=$schemaValue}}, exclusive: {{=$exclusive}} }", | |||||
| _exclusiveLimit: "{}", | |||||
| _limitItems: "{ limit: {{=$schemaValue}} }", | |||||
| _limitLength: "{ limit: {{=$schemaValue}} }", | |||||
| _limitProperties:"{ limit: {{=$schemaValue}} }", | |||||
| multipleOf: "{ multipleOf: {{=$schemaValue}} }", | |||||
| not: "{}", | |||||
| oneOf: "{ passingSchemas: {{=$passingSchemas}} }", | |||||
| pattern: "{ pattern: {{#def.schemaValueQS}} }", | |||||
| propertyNames: "{ propertyName: '{{=$invalidName}}' }", | |||||
| required: "{ missingProperty: '{{=$missingProperty}}' }", | |||||
| type: "{ type: '{{? $typeIsArray }}{{= $typeSchema.join(\",\") }}{{??}}{{=$typeSchema}}{{?}}' }", | |||||
| uniqueItems: "{ i: i, j: j }", | |||||
| custom: "{ keyword: '{{=$rule.keyword}}' }", | |||||
| patternRequired: "{ missingPattern: '{{=$missingPattern}}' }", | |||||
| switch: "{ caseIndex: {{=$caseIndex}} }", | |||||
| _formatLimit: "{ comparison: {{=$opExpr}}, limit: {{#def.schemaValueQS}}, exclusive: {{=$exclusive}} }", | |||||
| _formatExclusiveLimit: "{}" | |||||
| } #}} | |||||
| @ -0,0 +1,106 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{## def.skipFormat: | |||||
| {{? $breakOnError }} if (true) { {{?}} | |||||
| {{ return out; }} | |||||
| #}} | |||||
| {{? it.opts.format === false }}{{# def.skipFormat }}{{?}} | |||||
| {{# def.$data }} | |||||
| {{## def.$dataCheckFormat: | |||||
| {{# def.$dataNotType:'string' }} | |||||
| ({{? $unknownFormats != 'ignore' }} | |||||
| ({{=$schemaValue}} && !{{=$format}} | |||||
| {{? $allowUnknown }} | |||||
| && self._opts.unknownFormats.indexOf({{=$schemaValue}}) == -1 | |||||
| {{?}}) || | |||||
| {{?}} | |||||
| ({{=$format}} && {{=$formatType}} == '{{=$ruleType}}' | |||||
| && !(typeof {{=$format}} == 'function' | |||||
| ? {{? it.async}} | |||||
| (async{{=$lvl}} ? await {{=$format}}({{=$data}}) : {{=$format}}({{=$data}})) | |||||
| {{??}} | |||||
| {{=$format}}({{=$data}}) | |||||
| {{?}} | |||||
| : {{=$format}}.test({{=$data}})))) | |||||
| #}} | |||||
| {{## def.checkFormat: | |||||
| {{ | |||||
| var $formatRef = 'formats' + it.util.getProperty($schema); | |||||
| if ($isObject) $formatRef += '.validate'; | |||||
| }} | |||||
| {{? typeof $format == 'function' }} | |||||
| {{=$formatRef}}({{=$data}}) | |||||
| {{??}} | |||||
| {{=$formatRef}}.test({{=$data}}) | |||||
| {{?}} | |||||
| #}} | |||||
| {{ | |||||
| var $unknownFormats = it.opts.unknownFormats | |||||
| , $allowUnknown = Array.isArray($unknownFormats); | |||||
| }} | |||||
| {{? $isData }} | |||||
| {{ | |||||
| var $format = 'format' + $lvl | |||||
| , $isObject = 'isObject' + $lvl | |||||
| , $formatType = 'formatType' + $lvl; | |||||
| }} | |||||
| var {{=$format}} = formats[{{=$schemaValue}}]; | |||||
| var {{=$isObject}} = typeof {{=$format}} == 'object' | |||||
| && !({{=$format}} instanceof RegExp) | |||||
| && {{=$format}}.validate; | |||||
| var {{=$formatType}} = {{=$isObject}} && {{=$format}}.type || 'string'; | |||||
| if ({{=$isObject}}) { | |||||
| {{? it.async}} | |||||
| var async{{=$lvl}} = {{=$format}}.async; | |||||
| {{?}} | |||||
| {{=$format}} = {{=$format}}.validate; | |||||
| } | |||||
| if ({{# def.$dataCheckFormat }}) { | |||||
| {{??}} | |||||
| {{ var $format = it.formats[$schema]; }} | |||||
| {{? !$format }} | |||||
| {{? $unknownFormats == 'ignore' }} | |||||
| {{ it.logger.warn('unknown format "' + $schema + '" ignored in schema at path "' + it.errSchemaPath + '"'); }} | |||||
| {{# def.skipFormat }} | |||||
| {{?? $allowUnknown && $unknownFormats.indexOf($schema) >= 0 }} | |||||
| {{# def.skipFormat }} | |||||
| {{??}} | |||||
| {{ throw new Error('unknown format "' + $schema + '" is used in schema at path "' + it.errSchemaPath + '"'); }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{ | |||||
| var $isObject = typeof $format == 'object' | |||||
| && !($format instanceof RegExp) | |||||
| && $format.validate; | |||||
| var $formatType = $isObject && $format.type || 'string'; | |||||
| if ($isObject) { | |||||
| var $async = $format.async === true; | |||||
| $format = $format.validate; | |||||
| } | |||||
| }} | |||||
| {{? $formatType != $ruleType }} | |||||
| {{# def.skipFormat }} | |||||
| {{?}} | |||||
| {{? $async }} | |||||
| {{ | |||||
| if (!it.async) throw new Error('async format in sync schema'); | |||||
| var $formatRef = 'formats' + it.util.getProperty($schema) + '.validate'; | |||||
| }} | |||||
| if (!(await {{=$formatRef}}({{=$data}}))) { | |||||
| {{??}} | |||||
| if (!{{# def.checkFormat }}) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{# def.error:'format' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,75 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{## def.validateIfClause:_clause: | |||||
| {{ | |||||
| $it.schema = it.schema['_clause']; | |||||
| $it.schemaPath = it.schemaPath + '._clause'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/_clause'; | |||||
| }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| {{=$valid}} = {{=$nextValid}}; | |||||
| {{? $thenPresent && $elsePresent }} | |||||
| {{ $ifClause = 'ifClause' + $lvl; }} | |||||
| var {{=$ifClause}} = '_clause'; | |||||
| {{??}} | |||||
| {{ $ifClause = '\'_clause\''; }} | |||||
| {{?}} | |||||
| #}} | |||||
| {{ | |||||
| var $thenSch = it.schema['then'] | |||||
| , $elseSch = it.schema['else'] | |||||
| , $thenPresent = $thenSch !== undefined && {{# def.nonEmptySchema:$thenSch }} | |||||
| , $elsePresent = $elseSch !== undefined && {{# def.nonEmptySchema:$elseSch }} | |||||
| , $currentBaseId = $it.baseId; | |||||
| }} | |||||
| {{? $thenPresent || $elsePresent }} | |||||
| {{ | |||||
| var $ifClause; | |||||
| $it.createErrors = false; | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$valid}} = true; | |||||
| {{# def.setCompositeRule }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| {{ $it.createErrors = true; }} | |||||
| {{# def.resetErrors }} | |||||
| {{# def.resetCompositeRule }} | |||||
| {{? $thenPresent }} | |||||
| if ({{=$nextValid}}) { | |||||
| {{# def.validateIfClause:then }} | |||||
| } | |||||
| {{? $elsePresent }} | |||||
| else { | |||||
| {{?}} | |||||
| {{??}} | |||||
| if (!{{=$nextValid}}) { | |||||
| {{?}} | |||||
| {{? $elsePresent }} | |||||
| {{# def.validateIfClause:else }} | |||||
| } | |||||
| {{?}} | |||||
| if (!{{=$valid}}) { | |||||
| {{# def.extraError:'if' }} | |||||
| } | |||||
| {{? $breakOnError }} else { {{?}} | |||||
| {{# def.cleanUp }} | |||||
| {{??}} | |||||
| {{? $breakOnError }} | |||||
| if (true) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,100 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{## def.validateItems:startFrom: | |||||
| for (var {{=$idx}} = {{=startFrom}}; {{=$idx}} < {{=$data}}.length; {{=$idx}}++) { | |||||
| {{ | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $idx, it.opts.jsonPointers, true); | |||||
| var $passData = $data + '[' + $idx + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $idx; | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| {{? $breakOnError }} | |||||
| if (!{{=$nextValid}}) break; | |||||
| {{?}} | |||||
| } | |||||
| #}} | |||||
| {{ | |||||
| var $idx = 'i' + $lvl | |||||
| , $dataNxt = $it.dataLevel = it.dataLevel + 1 | |||||
| , $nextData = 'data' + $dataNxt | |||||
| , $currentBaseId = it.baseId; | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$valid}}; | |||||
| {{? Array.isArray($schema) }} | |||||
| {{ /* 'items' is an array of schemas */}} | |||||
| {{ var $additionalItems = it.schema.additionalItems; }} | |||||
| {{? $additionalItems === false }} | |||||
| {{=$valid}} = {{=$data}}.length <= {{= $schema.length }}; | |||||
| {{ | |||||
| var $currErrSchemaPath = $errSchemaPath; | |||||
| $errSchemaPath = it.errSchemaPath + '/additionalItems'; | |||||
| }} | |||||
| {{# def.checkError:'additionalItems' }} | |||||
| {{ $errSchemaPath = $currErrSchemaPath; }} | |||||
| {{# def.elseIfValid}} | |||||
| {{?}} | |||||
| {{~ $schema:$sch:$i }} | |||||
| {{? {{# def.nonEmptySchema:$sch }} }} | |||||
| {{=$nextValid}} = true; | |||||
| if ({{=$data}}.length > {{=$i}}) { | |||||
| {{ | |||||
| var $passData = $data + '[' + $i + ']'; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $i, it.opts.jsonPointers, true); | |||||
| $it.dataPathArr[$dataNxt] = $i; | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| } | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{? typeof $additionalItems == 'object' && {{# def.nonEmptySchema:$additionalItems }} }} | |||||
| {{ | |||||
| $it.schema = $additionalItems; | |||||
| $it.schemaPath = it.schemaPath + '.additionalItems'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/additionalItems'; | |||||
| }} | |||||
| {{=$nextValid}} = true; | |||||
| if ({{=$data}}.length > {{= $schema.length }}) { | |||||
| {{# def.validateItems: $schema.length }} | |||||
| } | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} | |||||
| {{?? {{# def.nonEmptySchema:$schema }} }} | |||||
| {{ /* 'items' is a single schema */}} | |||||
| {{ | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| }} | |||||
| {{# def.validateItems: 0 }} | |||||
| {{?}} | |||||
| {{? $breakOnError }} | |||||
| {{= $closingBraces }} | |||||
| if ({{=$errs}} == errors) { | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,39 @@ | |||||
| {{## def.checkMissingProperty:_properties: | |||||
| {{~ _properties:$propertyKey:$i }} | |||||
| {{?$i}} || {{?}} | |||||
| {{ | |||||
| var $prop = it.util.getProperty($propertyKey) | |||||
| , $useData = $data + $prop; | |||||
| }} | |||||
| ( ({{# def.noPropertyInData }}) && (missing{{=$lvl}} = {{= it.util.toQuotedString(it.opts.jsonPointers ? $propertyKey : $prop) }}) ) | |||||
| {{~}} | |||||
| #}} | |||||
| {{## def.errorMissingProperty:_error: | |||||
| {{ | |||||
| var $propertyPath = 'missing' + $lvl | |||||
| , $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.opts.jsonPointers | |||||
| ? it.util.getPathExpr($currentErrorPath, $propertyPath, true) | |||||
| : $currentErrorPath + ' + ' + $propertyPath; | |||||
| } | |||||
| }} | |||||
| {{# def.error:_error }} | |||||
| #}} | |||||
| {{## def.allErrorsMissingProperty:_error: | |||||
| {{ | |||||
| var $prop = it.util.getProperty($propertyKey) | |||||
| , $missingProperty = it.util.escapeQuotes($propertyKey) | |||||
| , $useData = $data + $prop; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPath($currentErrorPath, $propertyKey, it.opts.jsonPointers); | |||||
| } | |||||
| }} | |||||
| if ({{# def.noPropertyInData }}) { | |||||
| {{# def.addError:_error }} | |||||
| } | |||||
| #}} | |||||
| @ -0,0 +1,20 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| var division{{=$lvl}}; | |||||
| if ({{?$isData}} | |||||
| {{=$schemaValue}} !== undefined && ( | |||||
| typeof {{=$schemaValue}} != 'number' || | |||||
| {{?}} | |||||
| (division{{=$lvl}} = {{=$data}} / {{=$schemaValue}}, | |||||
| {{? it.opts.multipleOfPrecision }} | |||||
| Math.abs(Math.round(division{{=$lvl}}) - division{{=$lvl}}) > 1e-{{=it.opts.multipleOfPrecision}} | |||||
| {{??}} | |||||
| division{{=$lvl}} !== parseInt(division{{=$lvl}}) | |||||
| {{?}} | |||||
| ) | |||||
| {{?$isData}} ) {{?}} ) { | |||||
| {{# def.error:'multipleOf' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,43 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{? {{# def.nonEmptySchema:$schema }} }} | |||||
| {{ | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| {{# def.setCompositeRule }} | |||||
| {{ | |||||
| $it.createErrors = false; | |||||
| var $allErrorsOption; | |||||
| if ($it.opts.allErrors) { | |||||
| $allErrorsOption = $it.opts.allErrors; | |||||
| $it.opts.allErrors = false; | |||||
| } | |||||
| }} | |||||
| {{= it.validate($it) }} | |||||
| {{ | |||||
| $it.createErrors = true; | |||||
| if ($allErrorsOption) $it.opts.allErrors = $allErrorsOption; | |||||
| }} | |||||
| {{# def.resetCompositeRule }} | |||||
| if ({{=$nextValid}}) { | |||||
| {{# def.error:'not' }} | |||||
| } else { | |||||
| {{# def.resetErrors }} | |||||
| {{? it.opts.allErrors }} } {{?}} | |||||
| {{??}} | |||||
| {{# def.addError:'not' }} | |||||
| {{? $breakOnError}} | |||||
| if (false) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,54 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| var $currentBaseId = $it.baseId | |||||
| , $prevValid = 'prevValid' + $lvl | |||||
| , $passingSchemas = 'passingSchemas' + $lvl; | |||||
| }} | |||||
| var {{=$errs}} = errors | |||||
| , {{=$prevValid}} = false | |||||
| , {{=$valid}} = false | |||||
| , {{=$passingSchemas}} = null; | |||||
| {{# def.setCompositeRule }} | |||||
| {{~ $schema:$sch:$i }} | |||||
| {{? {{# def.nonEmptySchema:$sch }} }} | |||||
| {{ | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| }} | |||||
| {{# def.insertSubschemaCode }} | |||||
| {{??}} | |||||
| var {{=$nextValid}} = true; | |||||
| {{?}} | |||||
| {{? $i }} | |||||
| if ({{=$nextValid}} && {{=$prevValid}}) { | |||||
| {{=$valid}} = false; | |||||
| {{=$passingSchemas}} = [{{=$passingSchemas}}, {{=$i}}]; | |||||
| } else { | |||||
| {{ $closingBraces += '}'; }} | |||||
| {{?}} | |||||
| if ({{=$nextValid}}) { | |||||
| {{=$valid}} = {{=$prevValid}} = true; | |||||
| {{=$passingSchemas}} = {{=$i}}; | |||||
| } | |||||
| {{~}} | |||||
| {{# def.resetCompositeRule }} | |||||
| {{= $closingBraces }} | |||||
| if (!{{=$valid}}) { | |||||
| {{# def.extraError:'oneOf' }} | |||||
| } else { | |||||
| {{# def.resetErrors }} | |||||
| {{? it.opts.allErrors }} } {{?}} | |||||
| @ -0,0 +1,14 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ | |||||
| var $regexp = $isData | |||||
| ? '(new RegExp(' + $schemaValue + '))' | |||||
| : it.usePattern($schema); | |||||
| }} | |||||
| if ({{# def.$dataNotType:'string' }} !{{=$regexp}}.test({{=$data}}) ) { | |||||
| {{# def.error:'pattern' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| @ -0,0 +1,244 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{## def.validateAdditional: | |||||
| {{ /* additionalProperties is schema */ | |||||
| $it.schema = $aProperties; | |||||
| $it.schemaPath = it.schemaPath + '.additionalProperties'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/additionalProperties'; | |||||
| $it.errorPath = it.opts._errorDataPathProperty | |||||
| ? it.errorPath | |||||
| : it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| var $passData = $data + '[' + $key + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $key; | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| #}} | |||||
| {{ | |||||
| var $key = 'key' + $lvl | |||||
| , $idx = 'idx' + $lvl | |||||
| , $dataNxt = $it.dataLevel = it.dataLevel + 1 | |||||
| , $nextData = 'data' + $dataNxt | |||||
| , $dataProperties = 'dataProperties' + $lvl; | |||||
| var $schemaKeys = Object.keys($schema || {}) | |||||
| , $pProperties = it.schema.patternProperties || {} | |||||
| , $pPropertyKeys = Object.keys($pProperties) | |||||
| , $aProperties = it.schema.additionalProperties | |||||
| , $someProperties = $schemaKeys.length || $pPropertyKeys.length | |||||
| , $noAdditional = $aProperties === false | |||||
| , $additionalIsSchema = typeof $aProperties == 'object' | |||||
| && Object.keys($aProperties).length | |||||
| , $removeAdditional = it.opts.removeAdditional | |||||
| , $checkAdditional = $noAdditional || $additionalIsSchema || $removeAdditional | |||||
| , $ownProperties = it.opts.ownProperties | |||||
| , $currentBaseId = it.baseId; | |||||
| var $required = it.schema.required; | |||||
| if ($required && !(it.opts.$data && $required.$data) && $required.length < it.opts.loopRequired) | |||||
| var $requiredHash = it.util.toHash($required); | |||||
| }} | |||||
| var {{=$errs}} = errors; | |||||
| var {{=$nextValid}} = true; | |||||
| {{? $ownProperties }} | |||||
| var {{=$dataProperties}} = undefined; | |||||
| {{?}} | |||||
| {{? $checkAdditional }} | |||||
| {{# def.iterateProperties }} | |||||
| {{? $someProperties }} | |||||
| var isAdditional{{=$lvl}} = !(false | |||||
| {{? $schemaKeys.length }} | |||||
| {{? $schemaKeys.length > 8 }} | |||||
| || validate.schema{{=$schemaPath}}.hasOwnProperty({{=$key}}) | |||||
| {{??}} | |||||
| {{~ $schemaKeys:$propertyKey }} | |||||
| || {{=$key}} == {{= it.util.toQuotedString($propertyKey) }} | |||||
| {{~}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $pPropertyKeys.length }} | |||||
| {{~ $pPropertyKeys:$pProperty:$i }} | |||||
| || {{= it.usePattern($pProperty) }}.test({{=$key}}) | |||||
| {{~}} | |||||
| {{?}} | |||||
| ); | |||||
| if (isAdditional{{=$lvl}}) { | |||||
| {{?}} | |||||
| {{? $removeAdditional == 'all' }} | |||||
| delete {{=$data}}[{{=$key}}]; | |||||
| {{??}} | |||||
| {{ | |||||
| var $currentErrorPath = it.errorPath; | |||||
| var $additionalProperty = '\' + ' + $key + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| } | |||||
| }} | |||||
| {{? $noAdditional }} | |||||
| {{? $removeAdditional }} | |||||
| delete {{=$data}}[{{=$key}}]; | |||||
| {{??}} | |||||
| {{=$nextValid}} = false; | |||||
| {{ | |||||
| var $currErrSchemaPath = $errSchemaPath; | |||||
| $errSchemaPath = it.errSchemaPath + '/additionalProperties'; | |||||
| }} | |||||
| {{# def.error:'additionalProperties' }} | |||||
| {{ $errSchemaPath = $currErrSchemaPath; }} | |||||
| {{? $breakOnError }} break; {{?}} | |||||
| {{?}} | |||||
| {{?? $additionalIsSchema }} | |||||
| {{? $removeAdditional == 'failing' }} | |||||
| var {{=$errs}} = errors; | |||||
| {{# def.setCompositeRule }} | |||||
| {{# def.validateAdditional }} | |||||
| if (!{{=$nextValid}}) { | |||||
| errors = {{=$errs}}; | |||||
| if (validate.errors !== null) { | |||||
| if (errors) validate.errors.length = errors; | |||||
| else validate.errors = null; | |||||
| } | |||||
| delete {{=$data}}[{{=$key}}]; | |||||
| } | |||||
| {{# def.resetCompositeRule }} | |||||
| {{??}} | |||||
| {{# def.validateAdditional }} | |||||
| {{? $breakOnError }} if (!{{=$nextValid}}) break; {{?}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{ it.errorPath = $currentErrorPath; }} | |||||
| {{?}} | |||||
| {{? $someProperties }} | |||||
| } | |||||
| {{?}} | |||||
| } | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} | |||||
| {{ var $useDefaults = it.opts.useDefaults && !it.compositeRule; }} | |||||
| {{? $schemaKeys.length }} | |||||
| {{~ $schemaKeys:$propertyKey }} | |||||
| {{ var $sch = $schema[$propertyKey]; }} | |||||
| {{? {{# def.nonEmptySchema:$sch}} }} | |||||
| {{ | |||||
| var $prop = it.util.getProperty($propertyKey) | |||||
| , $passData = $data + $prop | |||||
| , $hasDefault = $useDefaults && $sch.default !== undefined; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + $prop; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + it.util.escapeFragment($propertyKey); | |||||
| $it.errorPath = it.util.getPath(it.errorPath, $propertyKey, it.opts.jsonPointers); | |||||
| $it.dataPathArr[$dataNxt] = it.util.toQuotedString($propertyKey); | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{? {{# def.willOptimize }} }} | |||||
| {{ | |||||
| $code = {{# def._optimizeValidate }}; | |||||
| var $useData = $passData; | |||||
| }} | |||||
| {{??}} | |||||
| {{ var $useData = $nextData; }} | |||||
| var {{=$nextData}} = {{=$passData}}; | |||||
| {{?}} | |||||
| {{? $hasDefault }} | |||||
| {{= $code }} | |||||
| {{??}} | |||||
| {{? $requiredHash && $requiredHash[$propertyKey] }} | |||||
| if ({{# def.noPropertyInData }}) { | |||||
| {{=$nextValid}} = false; | |||||
| {{ | |||||
| var $currentErrorPath = it.errorPath | |||||
| , $currErrSchemaPath = $errSchemaPath | |||||
| , $missingProperty = it.util.escapeQuotes($propertyKey); | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPath($currentErrorPath, $propertyKey, it.opts.jsonPointers); | |||||
| } | |||||
| $errSchemaPath = it.errSchemaPath + '/required'; | |||||
| }} | |||||
| {{# def.error:'required' }} | |||||
| {{ $errSchemaPath = $currErrSchemaPath; }} | |||||
| {{ it.errorPath = $currentErrorPath; }} | |||||
| } else { | |||||
| {{??}} | |||||
| {{? $breakOnError }} | |||||
| if ({{# def.noPropertyInData }}) { | |||||
| {{=$nextValid}} = true; | |||||
| } else { | |||||
| {{??}} | |||||
| if ({{=$useData}} !== undefined | |||||
| {{? $ownProperties }} | |||||
| && {{# def.isOwnProperty }} | |||||
| {{?}} | |||||
| ) { | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{= $code }} | |||||
| } | |||||
| {{?}} {{ /* $hasDefault */ }} | |||||
| {{?}} {{ /* def.nonEmptySchema */ }} | |||||
| {{# def.ifResultValid }} | |||||
| {{~}} | |||||
| {{?}} | |||||
| {{? $pPropertyKeys.length }} | |||||
| {{~ $pPropertyKeys:$pProperty }} | |||||
| {{ var $sch = $pProperties[$pProperty]; }} | |||||
| {{? {{# def.nonEmptySchema:$sch}} }} | |||||
| {{ | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = it.schemaPath + '.patternProperties' + it.util.getProperty($pProperty); | |||||
| $it.errSchemaPath = it.errSchemaPath + '/patternProperties/' | |||||
| + it.util.escapeFragment($pProperty); | |||||
| }} | |||||
| {{# def.iterateProperties }} | |||||
| if ({{= it.usePattern($pProperty) }}.test({{=$key}})) { | |||||
| {{ | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| var $passData = $data + '[' + $key + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $key; | |||||
| }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| {{? $breakOnError }} if (!{{=$nextValid}}) break; {{?}} | |||||
| } | |||||
| {{? $breakOnError }} else {{=$nextValid}} = true; {{?}} | |||||
| } | |||||
| {{# def.ifResultValid }} | |||||
| {{?}} {{ /* def.nonEmptySchema */ }} | |||||
| {{~}} | |||||
| {{?}} | |||||
| {{? $breakOnError }} | |||||
| {{= $closingBraces }} | |||||
| if ({{=$errs}} == errors) { | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,54 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.setupNextLevel }} | |||||
| var {{=$errs}} = errors; | |||||
| {{? {{# def.nonEmptySchema:$schema }} }} | |||||
| {{ | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| }} | |||||
| {{ | |||||
| var $key = 'key' + $lvl | |||||
| , $idx = 'idx' + $lvl | |||||
| , $i = 'i' + $lvl | |||||
| , $invalidName = '\' + ' + $key + ' + \'' | |||||
| , $dataNxt = $it.dataLevel = it.dataLevel + 1 | |||||
| , $nextData = 'data' + $dataNxt | |||||
| , $dataProperties = 'dataProperties' + $lvl | |||||
| , $ownProperties = it.opts.ownProperties | |||||
| , $currentBaseId = it.baseId; | |||||
| }} | |||||
| {{? $ownProperties }} | |||||
| var {{=$dataProperties}} = undefined; | |||||
| {{?}} | |||||
| {{# def.iterateProperties }} | |||||
| var startErrs{{=$lvl}} = errors; | |||||
| {{ var $passData = $key; }} | |||||
| {{# def.setCompositeRule }} | |||||
| {{# def.generateSubschemaCode }} | |||||
| {{# def.optimizeValidate }} | |||||
| {{# def.resetCompositeRule }} | |||||
| if (!{{=$nextValid}}) { | |||||
| for (var {{=$i}}=startErrs{{=$lvl}}; {{=$i}}<errors; {{=$i}}++) { | |||||
| vErrors[{{=$i}}].propertyName = {{=$key}}; | |||||
| } | |||||
| {{# def.extraError:'propertyNames' }} | |||||
| {{? $breakOnError }} break; {{?}} | |||||
| } | |||||
| } | |||||
| {{?}} | |||||
| {{? $breakOnError }} | |||||
| {{= $closingBraces }} | |||||
| if ({{=$errs}} == errors) { | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| @ -0,0 +1,85 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{## def._validateRef:_v: | |||||
| {{? it.opts.passContext }} | |||||
| {{=_v}}.call(this, | |||||
| {{??}} | |||||
| {{=_v}}( | |||||
| {{?}} | |||||
| {{=$data}}, {{# def.dataPath }}{{# def.passParentData }}, rootData) | |||||
| #}} | |||||
| {{ var $async, $refCode; }} | |||||
| {{? $schema == '#' || $schema == '#/' }} | |||||
| {{ | |||||
| if (it.isRoot) { | |||||
| $async = it.async; | |||||
| $refCode = 'validate'; | |||||
| } else { | |||||
| $async = it.root.schema.$async === true; | |||||
| $refCode = 'root.refVal[0]'; | |||||
| } | |||||
| }} | |||||
| {{??}} | |||||
| {{ var $refVal = it.resolveRef(it.baseId, $schema, it.isRoot); }} | |||||
| {{? $refVal === undefined }} | |||||
| {{ var $message = it.MissingRefError.message(it.baseId, $schema); }} | |||||
| {{? it.opts.missingRefs == 'fail' }} | |||||
| {{ it.logger.error($message); }} | |||||
| {{# def.error:'$ref' }} | |||||
| {{? $breakOnError }} if (false) { {{?}} | |||||
| {{?? it.opts.missingRefs == 'ignore' }} | |||||
| {{ it.logger.warn($message); }} | |||||
| {{? $breakOnError }} if (true) { {{?}} | |||||
| {{??}} | |||||
| {{ throw new it.MissingRefError(it.baseId, $schema, $message); }} | |||||
| {{?}} | |||||
| {{?? $refVal.inline }} | |||||
| {{# def.setupNextLevel }} | |||||
| {{ | |||||
| $it.schema = $refVal.schema; | |||||
| $it.schemaPath = ''; | |||||
| $it.errSchemaPath = $schema; | |||||
| }} | |||||
| {{ var $code = it.validate($it).replace(/validate\.schema/g, $refVal.code); }} | |||||
| {{= $code }} | |||||
| {{? $breakOnError}} | |||||
| if ({{=$nextValid}}) { | |||||
| {{?}} | |||||
| {{??}} | |||||
| {{ | |||||
| $async = $refVal.$async === true || (it.async && $refVal.$async !== false); | |||||
| $refCode = $refVal.code; | |||||
| }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $refCode }} | |||||
| {{# def.beginDefOut}} | |||||
| {{# def._validateRef:$refCode }} | |||||
| {{# def.storeDefOut:__callValidate }} | |||||
| {{? $async }} | |||||
| {{ if (!it.async) throw new Error('async schema referenced by sync schema'); }} | |||||
| {{? $breakOnError }} var {{=$valid}}; {{?}} | |||||
| try { | |||||
| await {{=__callValidate}}; | |||||
| {{? $breakOnError }} {{=$valid}} = true; {{?}} | |||||
| } catch (e) { | |||||
| if (!(e instanceof ValidationError)) throw e; | |||||
| if (vErrors === null) vErrors = e.errors; | |||||
| else vErrors = vErrors.concat(e.errors); | |||||
| errors = vErrors.length; | |||||
| {{? $breakOnError }} {{=$valid}} = false; {{?}} | |||||
| } | |||||
| {{? $breakOnError }} if ({{=$valid}}) { {{?}} | |||||
| {{??}} | |||||
| if (!{{=__callValidate}}) { | |||||
| if (vErrors === null) vErrors = {{=$refCode}}.errors; | |||||
| else vErrors = vErrors.concat({{=$refCode}}.errors); | |||||
| errors = vErrors.length; | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,108 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.missing }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{ var $vSchema = 'schema' + $lvl; }} | |||||
| {{## def.setupLoop: | |||||
| {{? !$isData }} | |||||
| var {{=$vSchema}} = validate.schema{{=$schemaPath}}; | |||||
| {{?}} | |||||
| {{ | |||||
| var $i = 'i' + $lvl | |||||
| , $propertyPath = 'schema' + $lvl + '[' + $i + ']' | |||||
| , $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPathExpr($currentErrorPath, $propertyPath, it.opts.jsonPointers); | |||||
| } | |||||
| }} | |||||
| #}} | |||||
| {{## def.isRequiredOwnProperty: | |||||
| Object.prototype.hasOwnProperty.call({{=$data}}, {{=$vSchema}}[{{=$i}}]) | |||||
| #}} | |||||
| {{? !$isData }} | |||||
| {{? $schema.length < it.opts.loopRequired && | |||||
| it.schema.properties && Object.keys(it.schema.properties).length }} | |||||
| {{ var $required = []; }} | |||||
| {{~ $schema:$property }} | |||||
| {{ var $propertySch = it.schema.properties[$property]; }} | |||||
| {{? !($propertySch && {{# def.nonEmptySchema:$propertySch}}) }} | |||||
| {{ $required[$required.length] = $property; }} | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{??}} | |||||
| {{ var $required = $schema; }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $isData || $required.length }} | |||||
| {{ | |||||
| var $currentErrorPath = it.errorPath | |||||
| , $loopRequired = $isData || $required.length >= it.opts.loopRequired | |||||
| , $ownProperties = it.opts.ownProperties; | |||||
| }} | |||||
| {{? $breakOnError }} | |||||
| var missing{{=$lvl}}; | |||||
| {{? $loopRequired }} | |||||
| {{# def.setupLoop }} | |||||
| var {{=$valid}} = true; | |||||
| {{?$isData}}{{# def.check$dataIsArray }}{{?}} | |||||
| for (var {{=$i}} = 0; {{=$i}} < {{=$vSchema}}.length; {{=$i}}++) { | |||||
| {{=$valid}} = {{=$data}}[{{=$vSchema}}[{{=$i}}]] !== undefined | |||||
| {{? $ownProperties }} | |||||
| && {{# def.isRequiredOwnProperty }} | |||||
| {{?}}; | |||||
| if (!{{=$valid}}) break; | |||||
| } | |||||
| {{? $isData }} } {{?}} | |||||
| {{# def.checkError:'required' }} | |||||
| else { | |||||
| {{??}} | |||||
| if ({{# def.checkMissingProperty:$required }}) { | |||||
| {{# def.errorMissingProperty:'required' }} | |||||
| } else { | |||||
| {{?}} | |||||
| {{??}} | |||||
| {{? $loopRequired }} | |||||
| {{# def.setupLoop }} | |||||
| {{? $isData }} | |||||
| if ({{=$vSchema}} && !Array.isArray({{=$vSchema}})) { | |||||
| {{# def.addError:'required' }} | |||||
| } else if ({{=$vSchema}} !== undefined) { | |||||
| {{?}} | |||||
| for (var {{=$i}} = 0; {{=$i}} < {{=$vSchema}}.length; {{=$i}}++) { | |||||
| if ({{=$data}}[{{=$vSchema}}[{{=$i}}]] === undefined | |||||
| {{? $ownProperties }} | |||||
| || !{{# def.isRequiredOwnProperty }} | |||||
| {{?}}) { | |||||
| {{# def.addError:'required' }} | |||||
| } | |||||
| } | |||||
| {{? $isData }} } {{?}} | |||||
| {{??}} | |||||
| {{~ $required:$propertyKey }} | |||||
| {{# def.allErrorsMissingProperty:'required' }} | |||||
| {{~}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{ it.errorPath = $currentErrorPath; }} | |||||
| {{?? $breakOnError }} | |||||
| if (true) { | |||||
| {{?}} | |||||
| @ -0,0 +1,62 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.setupKeyword }} | |||||
| {{# def.$data }} | |||||
| {{? ($schema || $isData) && it.opts.uniqueItems !== false }} | |||||
| {{? $isData }} | |||||
| var {{=$valid}}; | |||||
| if ({{=$schemaValue}} === false || {{=$schemaValue}} === undefined) | |||||
| {{=$valid}} = true; | |||||
| else if (typeof {{=$schemaValue}} != 'boolean') | |||||
| {{=$valid}} = false; | |||||
| else { | |||||
| {{?}} | |||||
| var i = {{=$data}}.length | |||||
| , {{=$valid}} = true | |||||
| , j; | |||||
| if (i > 1) { | |||||
| {{ | |||||
| var $itemType = it.schema.items && it.schema.items.type | |||||
| , $typeIsArray = Array.isArray($itemType); | |||||
| }} | |||||
| {{? !$itemType || $itemType == 'object' || $itemType == 'array' || | |||||
| ($typeIsArray && ($itemType.indexOf('object') >= 0 || $itemType.indexOf('array') >= 0)) }} | |||||
| outer: | |||||
| for (;i--;) { | |||||
| for (j = i; j--;) { | |||||
| if (equal({{=$data}}[i], {{=$data}}[j])) { | |||||
| {{=$valid}} = false; | |||||
| break outer; | |||||
| } | |||||
| } | |||||
| } | |||||
| {{??}} | |||||
| var itemIndices = {}, item; | |||||
| for (;i--;) { | |||||
| var item = {{=$data}}[i]; | |||||
| {{ var $method = 'checkDataType' + ($typeIsArray ? 's' : ''); }} | |||||
| if ({{= it.util[$method]($itemType, 'item', true) }}) continue; | |||||
| {{? $typeIsArray}} | |||||
| if (typeof item == 'string') item = '"' + item; | |||||
| {{?}} | |||||
| if (typeof itemIndices[item] == 'number') { | |||||
| {{=$valid}} = false; | |||||
| j = itemIndices[item]; | |||||
| break; | |||||
| } | |||||
| itemIndices[item] = i; | |||||
| } | |||||
| {{?}} | |||||
| } | |||||
| {{? $isData }} } {{?}} | |||||
| if (!{{=$valid}}) { | |||||
| {{# def.error:'uniqueItems' }} | |||||
| } {{? $breakOnError }} else { {{?}} | |||||
| {{??}} | |||||
| {{? $breakOnError }} if (true) { {{?}} | |||||
| {{?}} | |||||
| @ -0,0 +1,282 @@ | |||||
| {{# def.definitions }} | |||||
| {{# def.errors }} | |||||
| {{# def.defaults }} | |||||
| {{# def.coerce }} | |||||
| {{ /** | |||||
| * schema compilation (render) time: | |||||
| * it = { schema, RULES, _validate, opts } | |||||
| * it.validate - this template function, | |||||
| * it is used recursively to generate code for subschemas | |||||
| * | |||||
| * runtime: | |||||
| * "validate" is a variable name to which this function will be assigned | |||||
| * validateRef etc. are defined in the parent scope in index.js | |||||
| */ }} | |||||
| {{ | |||||
| var $async = it.schema.$async === true | |||||
| , $refKeywords = it.util.schemaHasRulesExcept(it.schema, it.RULES.all, '$ref') | |||||
| , $id = it.self._getId(it.schema); | |||||
| }} | |||||
| {{ | |||||
| if (it.opts.strictKeywords) { | |||||
| var $unknownKwd = it.util.schemaUnknownRules(it.schema, it.RULES.keywords); | |||||
| if ($unknownKwd) { | |||||
| var $keywordsMsg = 'unknown keyword: ' + $unknownKwd; | |||||
| if (it.opts.strictKeywords === 'log') it.logger.warn($keywordsMsg); | |||||
| else throw new Error($keywordsMsg); | |||||
| } | |||||
| } | |||||
| }} | |||||
| {{? it.isTop }} | |||||
| var validate = {{?$async}}{{it.async = true;}}async {{?}}function(data, dataPath, parentData, parentDataProperty, rootData) { | |||||
| 'use strict'; | |||||
| {{? $id && (it.opts.sourceCode || it.opts.processCode) }} | |||||
| {{= '/\*# sourceURL=' + $id + ' */' }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? typeof it.schema == 'boolean' || !($refKeywords || it.schema.$ref) }} | |||||
| {{ var $keyword = 'false schema'; }} | |||||
| {{# def.setupKeyword }} | |||||
| {{? it.schema === false}} | |||||
| {{? it.isTop}} | |||||
| {{ $breakOnError = true; }} | |||||
| {{??}} | |||||
| var {{=$valid}} = false; | |||||
| {{?}} | |||||
| {{# def.error:'false schema' }} | |||||
| {{??}} | |||||
| {{? it.isTop}} | |||||
| {{? $async }} | |||||
| return data; | |||||
| {{??}} | |||||
| validate.errors = null; | |||||
| return true; | |||||
| {{?}} | |||||
| {{??}} | |||||
| var {{=$valid}} = true; | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? it.isTop}} | |||||
| }; | |||||
| return validate; | |||||
| {{?}} | |||||
| {{ return out; }} | |||||
| {{?}} | |||||
| {{? it.isTop }} | |||||
| {{ | |||||
| var $top = it.isTop | |||||
| , $lvl = it.level = 0 | |||||
| , $dataLvl = it.dataLevel = 0 | |||||
| , $data = 'data'; | |||||
| it.rootId = it.resolve.fullPath(it.self._getId(it.root.schema)); | |||||
| it.baseId = it.baseId || it.rootId; | |||||
| delete it.isTop; | |||||
| it.dataPathArr = [undefined]; | |||||
| if (it.schema.default !== undefined && it.opts.useDefaults && it.opts.strictDefaults) { | |||||
| var $defaultMsg = 'default is ignored in the schema root'; | |||||
| if (it.opts.strictDefaults === 'log') it.logger.warn($defaultMsg); | |||||
| else throw new Error($defaultMsg); | |||||
| } | |||||
| }} | |||||
| var vErrors = null; {{ /* don't edit, used in replace */ }} | |||||
| var errors = 0; {{ /* don't edit, used in replace */ }} | |||||
| if (rootData === undefined) rootData = data; {{ /* don't edit, used in replace */ }} | |||||
| {{??}} | |||||
| {{ | |||||
| var $lvl = it.level | |||||
| , $dataLvl = it.dataLevel | |||||
| , $data = 'data' + ($dataLvl || ''); | |||||
| if ($id) it.baseId = it.resolve.url(it.baseId, $id); | |||||
| if ($async && !it.async) throw new Error('async schema in sync schema'); | |||||
| }} | |||||
| var errs_{{=$lvl}} = errors; | |||||
| {{?}} | |||||
| {{ | |||||
| var $valid = 'valid' + $lvl | |||||
| , $breakOnError = !it.opts.allErrors | |||||
| , $closingBraces1 = '' | |||||
| , $closingBraces2 = ''; | |||||
| var $errorKeyword; | |||||
| var $typeSchema = it.schema.type | |||||
| , $typeIsArray = Array.isArray($typeSchema); | |||||
| if ($typeSchema && it.opts.nullable && it.schema.nullable === true) { | |||||
| if ($typeIsArray) { | |||||
| if ($typeSchema.indexOf('null') == -1) | |||||
| $typeSchema = $typeSchema.concat('null'); | |||||
| } else if ($typeSchema != 'null') { | |||||
| $typeSchema = [$typeSchema, 'null']; | |||||
| $typeIsArray = true; | |||||
| } | |||||
| } | |||||
| if ($typeIsArray && $typeSchema.length == 1) { | |||||
| $typeSchema = $typeSchema[0]; | |||||
| $typeIsArray = false; | |||||
| } | |||||
| }} | |||||
| {{## def.checkType: | |||||
| {{ | |||||
| var $schemaPath = it.schemaPath + '.type' | |||||
| , $errSchemaPath = it.errSchemaPath + '/type' | |||||
| , $method = $typeIsArray ? 'checkDataTypes' : 'checkDataType'; | |||||
| }} | |||||
| if ({{= it.util[$method]($typeSchema, $data, true) }}) { | |||||
| #}} | |||||
| {{? it.schema.$ref && $refKeywords }} | |||||
| {{? it.opts.extendRefs == 'fail' }} | |||||
| {{ throw new Error('$ref: validation keywords used in schema at path "' + it.errSchemaPath + '" (see option extendRefs)'); }} | |||||
| {{?? it.opts.extendRefs !== true }} | |||||
| {{ | |||||
| $refKeywords = false; | |||||
| it.logger.warn('$ref: keywords ignored in schema at path "' + it.errSchemaPath + '"'); | |||||
| }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? it.schema.$comment && it.opts.$comment }} | |||||
| {{= it.RULES.all.$comment.code(it, '$comment') }} | |||||
| {{?}} | |||||
| {{? $typeSchema }} | |||||
| {{? it.opts.coerceTypes }} | |||||
| {{ var $coerceToTypes = it.util.coerceToTypes(it.opts.coerceTypes, $typeSchema); }} | |||||
| {{?}} | |||||
| {{ var $rulesGroup = it.RULES.types[$typeSchema]; }} | |||||
| {{? $coerceToTypes || $typeIsArray || $rulesGroup === true || | |||||
| ($rulesGroup && !$shouldUseGroup($rulesGroup)) }} | |||||
| {{ | |||||
| var $schemaPath = it.schemaPath + '.type' | |||||
| , $errSchemaPath = it.errSchemaPath + '/type'; | |||||
| }} | |||||
| {{# def.checkType }} | |||||
| {{? $coerceToTypes }} | |||||
| {{# def.coerceType }} | |||||
| {{??}} | |||||
| {{# def.error:'type' }} | |||||
| {{?}} | |||||
| } | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? it.schema.$ref && !$refKeywords }} | |||||
| {{= it.RULES.all.$ref.code(it, '$ref') }} | |||||
| {{? $breakOnError }} | |||||
| } | |||||
| if (errors === {{?$top}}0{{??}}errs_{{=$lvl}}{{?}}) { | |||||
| {{ $closingBraces2 += '}'; }} | |||||
| {{?}} | |||||
| {{??}} | |||||
| {{~ it.RULES:$rulesGroup }} | |||||
| {{? $shouldUseGroup($rulesGroup) }} | |||||
| {{? $rulesGroup.type }} | |||||
| if ({{= it.util.checkDataType($rulesGroup.type, $data) }}) { | |||||
| {{?}} | |||||
| {{? it.opts.useDefaults }} | |||||
| {{? $rulesGroup.type == 'object' && it.schema.properties }} | |||||
| {{# def.defaultProperties }} | |||||
| {{?? $rulesGroup.type == 'array' && Array.isArray(it.schema.items) }} | |||||
| {{# def.defaultItems }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{~ $rulesGroup.rules:$rule }} | |||||
| {{? $shouldUseRule($rule) }} | |||||
| {{ var $code = $rule.code(it, $rule.keyword, $rulesGroup.type); }} | |||||
| {{? $code }} | |||||
| {{= $code }} | |||||
| {{? $breakOnError }} | |||||
| {{ $closingBraces1 += '}'; }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{? $breakOnError }} | |||||
| {{= $closingBraces1 }} | |||||
| {{ $closingBraces1 = ''; }} | |||||
| {{?}} | |||||
| {{? $rulesGroup.type }} | |||||
| } | |||||
| {{? $typeSchema && $typeSchema === $rulesGroup.type && !$coerceToTypes }} | |||||
| else { | |||||
| {{ | |||||
| var $schemaPath = it.schemaPath + '.type' | |||||
| , $errSchemaPath = it.errSchemaPath + '/type'; | |||||
| }} | |||||
| {{# def.error:'type' }} | |||||
| } | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{? $breakOnError }} | |||||
| if (errors === {{?$top}}0{{??}}errs_{{=$lvl}}{{?}}) { | |||||
| {{ $closingBraces2 += '}'; }} | |||||
| {{?}} | |||||
| {{?}} | |||||
| {{~}} | |||||
| {{?}} | |||||
| {{? $breakOnError }} {{= $closingBraces2 }} {{?}} | |||||
| {{? $top }} | |||||
| {{? $async }} | |||||
| if (errors === 0) return data; {{ /* don't edit, used in replace */ }} | |||||
| else throw new ValidationError(vErrors); {{ /* don't edit, used in replace */ }} | |||||
| {{??}} | |||||
| validate.errors = vErrors; {{ /* don't edit, used in replace */ }} | |||||
| return errors === 0; {{ /* don't edit, used in replace */ }} | |||||
| {{?}} | |||||
| }; | |||||
| return validate; | |||||
| {{??}} | |||||
| var {{=$valid}} = errors === errs_{{=$lvl}}; | |||||
| {{?}} | |||||
| {{# def.cleanUp }} | |||||
| {{? $top }} | |||||
| {{# def.finalCleanUp }} | |||||
| {{?}} | |||||
| {{ | |||||
| function $shouldUseGroup($rulesGroup) { | |||||
| var rules = $rulesGroup.rules; | |||||
| for (var i=0; i < rules.length; i++) | |||||
| if ($shouldUseRule(rules[i])) | |||||
| return true; | |||||
| } | |||||
| function $shouldUseRule($rule) { | |||||
| return it.schema[$rule.keyword] !== undefined || | |||||
| ($rule.implements && $ruleImplementsSomeKeyword($rule)); | |||||
| } | |||||
| function $ruleImplementsSomeKeyword($rule) { | |||||
| var impl = $rule.implements; | |||||
| for (var i=0; i < impl.length; i++) | |||||
| if (it.schema[impl[i]] !== undefined) | |||||
| return true; | |||||
| } | |||||
| }} | |||||
| @ -0,0 +1,3 @@ | |||||
| These files are compiled dot templates from dot folder. | |||||
| Do NOT edit them directly, edit the templates and run `npm run build` from main ajv folder. | |||||
| @ -0,0 +1,157 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate__limit(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $isMax = $keyword == 'maximum', | |||||
| $exclusiveKeyword = $isMax ? 'exclusiveMaximum' : 'exclusiveMinimum', | |||||
| $schemaExcl = it.schema[$exclusiveKeyword], | |||||
| $isDataExcl = it.opts.$data && $schemaExcl && $schemaExcl.$data, | |||||
| $op = $isMax ? '<' : '>', | |||||
| $notOp = $isMax ? '>' : '<', | |||||
| $errorKeyword = undefined; | |||||
| if ($isDataExcl) { | |||||
| var $schemaValueExcl = it.util.getData($schemaExcl.$data, $dataLvl, it.dataPathArr), | |||||
| $exclusive = 'exclusive' + $lvl, | |||||
| $exclType = 'exclType' + $lvl, | |||||
| $exclIsNumber = 'exclIsNumber' + $lvl, | |||||
| $opExpr = 'op' + $lvl, | |||||
| $opStr = '\' + ' + $opExpr + ' + \''; | |||||
| out += ' var schemaExcl' + ($lvl) + ' = ' + ($schemaValueExcl) + '; '; | |||||
| $schemaValueExcl = 'schemaExcl' + $lvl; | |||||
| out += ' var ' + ($exclusive) + '; var ' + ($exclType) + ' = typeof ' + ($schemaValueExcl) + '; if (' + ($exclType) + ' != \'boolean\' && ' + ($exclType) + ' != \'undefined\' && ' + ($exclType) + ' != \'number\') { '; | |||||
| var $errorKeyword = $exclusiveKeyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || '_exclusiveLimit') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'' + ($exclusiveKeyword) + ' should be boolean\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| out += ' ' + ($exclType) + ' == \'number\' ? ( (' + ($exclusive) + ' = ' + ($schemaValue) + ' === undefined || ' + ($schemaValueExcl) + ' ' + ($op) + '= ' + ($schemaValue) + ') ? ' + ($data) + ' ' + ($notOp) + '= ' + ($schemaValueExcl) + ' : ' + ($data) + ' ' + ($notOp) + ' ' + ($schemaValue) + ' ) : ( (' + ($exclusive) + ' = ' + ($schemaValueExcl) + ' === true) ? ' + ($data) + ' ' + ($notOp) + '= ' + ($schemaValue) + ' : ' + ($data) + ' ' + ($notOp) + ' ' + ($schemaValue) + ' ) || ' + ($data) + ' !== ' + ($data) + ') { var op' + ($lvl) + ' = ' + ($exclusive) + ' ? \'' + ($op) + '\' : \'' + ($op) + '=\'; '; | |||||
| if ($schema === undefined) { | |||||
| $errorKeyword = $exclusiveKeyword; | |||||
| $errSchemaPath = it.errSchemaPath + '/' + $exclusiveKeyword; | |||||
| $schemaValue = $schemaValueExcl; | |||||
| $isData = $isDataExcl; | |||||
| } | |||||
| } else { | |||||
| var $exclIsNumber = typeof $schemaExcl == 'number', | |||||
| $opStr = $op; | |||||
| if ($exclIsNumber && $isData) { | |||||
| var $opExpr = '\'' + $opStr + '\''; | |||||
| out += ' if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| out += ' ( ' + ($schemaValue) + ' === undefined || ' + ($schemaExcl) + ' ' + ($op) + '= ' + ($schemaValue) + ' ? ' + ($data) + ' ' + ($notOp) + '= ' + ($schemaExcl) + ' : ' + ($data) + ' ' + ($notOp) + ' ' + ($schemaValue) + ' ) || ' + ($data) + ' !== ' + ($data) + ') { '; | |||||
| } else { | |||||
| if ($exclIsNumber && $schema === undefined) { | |||||
| $exclusive = true; | |||||
| $errorKeyword = $exclusiveKeyword; | |||||
| $errSchemaPath = it.errSchemaPath + '/' + $exclusiveKeyword; | |||||
| $schemaValue = $schemaExcl; | |||||
| $notOp += '='; | |||||
| } else { | |||||
| if ($exclIsNumber) $schemaValue = Math[$isMax ? 'min' : 'max']($schemaExcl, $schema); | |||||
| if ($schemaExcl === ($exclIsNumber ? $schemaValue : true)) { | |||||
| $exclusive = true; | |||||
| $errorKeyword = $exclusiveKeyword; | |||||
| $errSchemaPath = it.errSchemaPath + '/' + $exclusiveKeyword; | |||||
| $notOp += '='; | |||||
| } else { | |||||
| $exclusive = false; | |||||
| $opStr += '='; | |||||
| } | |||||
| } | |||||
| var $opExpr = '\'' + $opStr + '\''; | |||||
| out += ' if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| out += ' ' + ($data) + ' ' + ($notOp) + ' ' + ($schemaValue) + ' || ' + ($data) + ' !== ' + ($data) + ') { '; | |||||
| } | |||||
| } | |||||
| $errorKeyword = $errorKeyword || $keyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || '_limit') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { comparison: ' + ($opExpr) + ', limit: ' + ($schemaValue) + ', exclusive: ' + ($exclusive) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be ' + ($opStr) + ' '; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue); | |||||
| } else { | |||||
| out += '' + ($schemaValue) + '\''; | |||||
| } | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,77 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate__limitItems(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $op = $keyword == 'maxItems' ? '>' : '<'; | |||||
| out += 'if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| out += ' ' + ($data) + '.length ' + ($op) + ' ' + ($schemaValue) + ') { '; | |||||
| var $errorKeyword = $keyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || '_limitItems') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { limit: ' + ($schemaValue) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT have '; | |||||
| if ($keyword == 'maxItems') { | |||||
| out += 'more'; | |||||
| } else { | |||||
| out += 'fewer'; | |||||
| } | |||||
| out += ' than '; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue) + ' + \''; | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' items\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += '} '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,82 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate__limitLength(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $op = $keyword == 'maxLength' ? '>' : '<'; | |||||
| out += 'if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| if (it.opts.unicode === false) { | |||||
| out += ' ' + ($data) + '.length '; | |||||
| } else { | |||||
| out += ' ucs2length(' + ($data) + ') '; | |||||
| } | |||||
| out += ' ' + ($op) + ' ' + ($schemaValue) + ') { '; | |||||
| var $errorKeyword = $keyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || '_limitLength') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { limit: ' + ($schemaValue) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT be '; | |||||
| if ($keyword == 'maxLength') { | |||||
| out += 'longer'; | |||||
| } else { | |||||
| out += 'shorter'; | |||||
| } | |||||
| out += ' than '; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue) + ' + \''; | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' characters\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += '} '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,77 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate__limitProperties(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $op = $keyword == 'maxProperties' ? '>' : '<'; | |||||
| out += 'if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'number\') || '; | |||||
| } | |||||
| out += ' Object.keys(' + ($data) + ').length ' + ($op) + ' ' + ($schemaValue) + ') { '; | |||||
| var $errorKeyword = $keyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || '_limitProperties') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { limit: ' + ($schemaValue) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT have '; | |||||
| if ($keyword == 'maxProperties') { | |||||
| out += 'more'; | |||||
| } else { | |||||
| out += 'fewer'; | |||||
| } | |||||
| out += ' than '; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue) + ' + \''; | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' properties\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += '} '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,43 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_allOf(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $currentBaseId = $it.baseId, | |||||
| $allSchemasEmpty = true; | |||||
| var arr1 = $schema; | |||||
| if (arr1) { | |||||
| var $sch, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $sch = arr1[$i += 1]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| $allSchemasEmpty = false; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| if ($allSchemasEmpty) { | |||||
| out += ' if (true) { '; | |||||
| } else { | |||||
| out += ' ' + ($closingBraces.slice(0, -1)) + ' '; | |||||
| } | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,74 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_anyOf(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $noEmptySchema = $schema.every(function($sch) { | |||||
| return (it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all)); | |||||
| }); | |||||
| if ($noEmptySchema) { | |||||
| var $currentBaseId = $it.baseId; | |||||
| out += ' var ' + ($errs) + ' = errors; var ' + ($valid) + ' = false; '; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| var arr1 = $schema; | |||||
| if (arr1) { | |||||
| var $sch, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $sch = arr1[$i += 1]; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| out += ' ' + ($valid) + ' = ' + ($valid) + ' || ' + ($nextValid) + '; if (!' + ($valid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += ' ' + ($closingBraces) + ' if (!' + ($valid) + ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('anyOf') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should match some schema in anyOf\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; return false; '; | |||||
| } | |||||
| } | |||||
| out += ' } else { errors = ' + ($errs) + '; if (vErrors !== null) { if (' + ($errs) + ') vErrors.length = ' + ($errs) + '; else vErrors = null; } '; | |||||
| if (it.opts.allErrors) { | |||||
| out += ' } '; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| } else { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,14 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_comment(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $comment = it.util.toQuotedString($schema); | |||||
| if (it.opts.$comment === true) { | |||||
| out += ' console.log(' + ($comment) + ');'; | |||||
| } else if (typeof it.opts.$comment == 'function') { | |||||
| out += ' self._opts.$comment(' + ($comment) + ', ' + (it.util.toQuotedString($errSchemaPath)) + ', validate.root.schema);'; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,56 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_const(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| if (!$isData) { | |||||
| out += ' var schema' + ($lvl) + ' = validate.schema' + ($schemaPath) + ';'; | |||||
| } | |||||
| out += 'var ' + ($valid) + ' = equal(' + ($data) + ', schema' + ($lvl) + '); if (!' + ($valid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('const') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { allowedValue: schema' + ($lvl) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be equal to constant\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' }'; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,82 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_contains(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $idx = 'i' + $lvl, | |||||
| $dataNxt = $it.dataLevel = it.dataLevel + 1, | |||||
| $nextData = 'data' + $dataNxt, | |||||
| $currentBaseId = it.baseId, | |||||
| $nonEmptySchema = (it.opts.strictKeywords ? typeof $schema == 'object' && Object.keys($schema).length > 0 : it.util.schemaHasRules($schema, it.RULES.all)); | |||||
| out += 'var ' + ($errs) + ' = errors;var ' + ($valid) + ';'; | |||||
| if ($nonEmptySchema) { | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| out += ' var ' + ($nextValid) + ' = false; for (var ' + ($idx) + ' = 0; ' + ($idx) + ' < ' + ($data) + '.length; ' + ($idx) + '++) { '; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $idx, it.opts.jsonPointers, true); | |||||
| var $passData = $data + '[' + $idx + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $idx; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| out += ' if (' + ($nextValid) + ') break; } '; | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += ' ' + ($closingBraces) + ' if (!' + ($nextValid) + ') {'; | |||||
| } else { | |||||
| out += ' if (' + ($data) + '.length == 0) {'; | |||||
| } | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('contains') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should contain a valid item\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else { '; | |||||
| if ($nonEmptySchema) { | |||||
| out += ' errors = ' + ($errs) + '; if (vErrors !== null) { if (' + ($errs) + ') vErrors.length = ' + ($errs) + '; else vErrors = null; } '; | |||||
| } | |||||
| if (it.opts.allErrors) { | |||||
| out += ' } '; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,228 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_custom(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $rule = this, | |||||
| $definition = 'definition' + $lvl, | |||||
| $rDef = $rule.definition, | |||||
| $closingBraces = ''; | |||||
| var $compile, $inline, $macro, $ruleValidate, $validateCode; | |||||
| if ($isData && $rDef.$data) { | |||||
| $validateCode = 'keywordValidate' + $lvl; | |||||
| var $validateSchema = $rDef.validateSchema; | |||||
| out += ' var ' + ($definition) + ' = RULES.custom[\'' + ($keyword) + '\'].definition; var ' + ($validateCode) + ' = ' + ($definition) + '.validate;'; | |||||
| } else { | |||||
| $ruleValidate = it.useCustomRule($rule, $schema, it.schema, it); | |||||
| if (!$ruleValidate) return; | |||||
| $schemaValue = 'validate.schema' + $schemaPath; | |||||
| $validateCode = $ruleValidate.code; | |||||
| $compile = $rDef.compile; | |||||
| $inline = $rDef.inline; | |||||
| $macro = $rDef.macro; | |||||
| } | |||||
| var $ruleErrs = $validateCode + '.errors', | |||||
| $i = 'i' + $lvl, | |||||
| $ruleErr = 'ruleErr' + $lvl, | |||||
| $asyncKeyword = $rDef.async; | |||||
| if ($asyncKeyword && !it.async) throw new Error('async keyword in sync schema'); | |||||
| if (!($inline || $macro)) { | |||||
| out += '' + ($ruleErrs) + ' = null;'; | |||||
| } | |||||
| out += 'var ' + ($errs) + ' = errors;var ' + ($valid) + ';'; | |||||
| if ($isData && $rDef.$data) { | |||||
| $closingBraces += '}'; | |||||
| out += ' if (' + ($schemaValue) + ' === undefined) { ' + ($valid) + ' = true; } else { '; | |||||
| if ($validateSchema) { | |||||
| $closingBraces += '}'; | |||||
| out += ' ' + ($valid) + ' = ' + ($definition) + '.validateSchema(' + ($schemaValue) + '); if (' + ($valid) + ') { '; | |||||
| } | |||||
| } | |||||
| if ($inline) { | |||||
| if ($rDef.statements) { | |||||
| out += ' ' + ($ruleValidate.validate) + ' '; | |||||
| } else { | |||||
| out += ' ' + ($valid) + ' = ' + ($ruleValidate.validate) + '; '; | |||||
| } | |||||
| } else if ($macro) { | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| $it.schema = $ruleValidate.validate; | |||||
| $it.schemaPath = ''; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| var $code = it.validate($it).replace(/validate\.schema/g, $validateCode); | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += ' ' + ($code); | |||||
| } else { | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; | |||||
| out += ' ' + ($validateCode) + '.call( '; | |||||
| if (it.opts.passContext) { | |||||
| out += 'this'; | |||||
| } else { | |||||
| out += 'self'; | |||||
| } | |||||
| if ($compile || $rDef.schema === false) { | |||||
| out += ' , ' + ($data) + ' '; | |||||
| } else { | |||||
| out += ' , ' + ($schemaValue) + ' , ' + ($data) + ' , validate.schema' + (it.schemaPath) + ' '; | |||||
| } | |||||
| out += ' , (dataPath || \'\')'; | |||||
| if (it.errorPath != '""') { | |||||
| out += ' + ' + (it.errorPath); | |||||
| } | |||||
| var $parentData = $dataLvl ? 'data' + (($dataLvl - 1) || '') : 'parentData', | |||||
| $parentDataProperty = $dataLvl ? it.dataPathArr[$dataLvl] : 'parentDataProperty'; | |||||
| out += ' , ' + ($parentData) + ' , ' + ($parentDataProperty) + ' , rootData ) '; | |||||
| var def_callRuleValidate = out; | |||||
| out = $$outStack.pop(); | |||||
| if ($rDef.errors === false) { | |||||
| out += ' ' + ($valid) + ' = '; | |||||
| if ($asyncKeyword) { | |||||
| out += 'await '; | |||||
| } | |||||
| out += '' + (def_callRuleValidate) + '; '; | |||||
| } else { | |||||
| if ($asyncKeyword) { | |||||
| $ruleErrs = 'customErrors' + $lvl; | |||||
| out += ' var ' + ($ruleErrs) + ' = null; try { ' + ($valid) + ' = await ' + (def_callRuleValidate) + '; } catch (e) { ' + ($valid) + ' = false; if (e instanceof ValidationError) ' + ($ruleErrs) + ' = e.errors; else throw e; } '; | |||||
| } else { | |||||
| out += ' ' + ($ruleErrs) + ' = null; ' + ($valid) + ' = ' + (def_callRuleValidate) + '; '; | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($rDef.modifying) { | |||||
| out += ' if (' + ($parentData) + ') ' + ($data) + ' = ' + ($parentData) + '[' + ($parentDataProperty) + '];'; | |||||
| } | |||||
| out += '' + ($closingBraces); | |||||
| if ($rDef.valid) { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| } else { | |||||
| out += ' if ( '; | |||||
| if ($rDef.valid === undefined) { | |||||
| out += ' !'; | |||||
| if ($macro) { | |||||
| out += '' + ($nextValid); | |||||
| } else { | |||||
| out += '' + ($valid); | |||||
| } | |||||
| } else { | |||||
| out += ' ' + (!$rDef.valid) + ' '; | |||||
| } | |||||
| out += ') { '; | |||||
| $errorKeyword = $rule.keyword; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'custom') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { keyword: \'' + ($rule.keyword) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should pass "' + ($rule.keyword) + '" keyword validation\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| var def_customError = out; | |||||
| out = $$outStack.pop(); | |||||
| if ($inline) { | |||||
| if ($rDef.errors) { | |||||
| if ($rDef.errors != 'full') { | |||||
| out += ' for (var ' + ($i) + '=' + ($errs) + '; ' + ($i) + '<errors; ' + ($i) + '++) { var ' + ($ruleErr) + ' = vErrors[' + ($i) + ']; if (' + ($ruleErr) + '.dataPath === undefined) ' + ($ruleErr) + '.dataPath = (dataPath || \'\') + ' + (it.errorPath) + '; if (' + ($ruleErr) + '.schemaPath === undefined) { ' + ($ruleErr) + '.schemaPath = "' + ($errSchemaPath) + '"; } '; | |||||
| if (it.opts.verbose) { | |||||
| out += ' ' + ($ruleErr) + '.schema = ' + ($schemaValue) + '; ' + ($ruleErr) + '.data = ' + ($data) + '; '; | |||||
| } | |||||
| out += ' } '; | |||||
| } | |||||
| } else { | |||||
| if ($rDef.errors === false) { | |||||
| out += ' ' + (def_customError) + ' '; | |||||
| } else { | |||||
| out += ' if (' + ($errs) + ' == errors) { ' + (def_customError) + ' } else { for (var ' + ($i) + '=' + ($errs) + '; ' + ($i) + '<errors; ' + ($i) + '++) { var ' + ($ruleErr) + ' = vErrors[' + ($i) + ']; if (' + ($ruleErr) + '.dataPath === undefined) ' + ($ruleErr) + '.dataPath = (dataPath || \'\') + ' + (it.errorPath) + '; if (' + ($ruleErr) + '.schemaPath === undefined) { ' + ($ruleErr) + '.schemaPath = "' + ($errSchemaPath) + '"; } '; | |||||
| if (it.opts.verbose) { | |||||
| out += ' ' + ($ruleErr) + '.schema = ' + ($schemaValue) + '; ' + ($ruleErr) + '.data = ' + ($data) + '; '; | |||||
| } | |||||
| out += ' } } '; | |||||
| } | |||||
| } | |||||
| } else if ($macro) { | |||||
| out += ' var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'custom') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { keyword: \'' + ($rule.keyword) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should pass "' + ($rule.keyword) + '" keyword validation\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; return false; '; | |||||
| } | |||||
| } | |||||
| } else { | |||||
| if ($rDef.errors === false) { | |||||
| out += ' ' + (def_customError) + ' '; | |||||
| } else { | |||||
| out += ' if (Array.isArray(' + ($ruleErrs) + ')) { if (vErrors === null) vErrors = ' + ($ruleErrs) + '; else vErrors = vErrors.concat(' + ($ruleErrs) + '); errors = vErrors.length; for (var ' + ($i) + '=' + ($errs) + '; ' + ($i) + '<errors; ' + ($i) + '++) { var ' + ($ruleErr) + ' = vErrors[' + ($i) + ']; if (' + ($ruleErr) + '.dataPath === undefined) ' + ($ruleErr) + '.dataPath = (dataPath || \'\') + ' + (it.errorPath) + '; ' + ($ruleErr) + '.schemaPath = "' + ($errSchemaPath) + '"; '; | |||||
| if (it.opts.verbose) { | |||||
| out += ' ' + ($ruleErr) + '.schema = ' + ($schemaValue) + '; ' + ($ruleErr) + '.data = ' + ($data) + '; '; | |||||
| } | |||||
| out += ' } } else { ' + (def_customError) + ' } '; | |||||
| } | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,168 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_dependencies(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $schemaDeps = {}, | |||||
| $propertyDeps = {}, | |||||
| $ownProperties = it.opts.ownProperties; | |||||
| for ($property in $schema) { | |||||
| var $sch = $schema[$property]; | |||||
| var $deps = Array.isArray($sch) ? $propertyDeps : $schemaDeps; | |||||
| $deps[$property] = $sch; | |||||
| } | |||||
| out += 'var ' + ($errs) + ' = errors;'; | |||||
| var $currentErrorPath = it.errorPath; | |||||
| out += 'var missing' + ($lvl) + ';'; | |||||
| for (var $property in $propertyDeps) { | |||||
| $deps = $propertyDeps[$property]; | |||||
| if ($deps.length) { | |||||
| out += ' if ( ' + ($data) + (it.util.getProperty($property)) + ' !== undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' && Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($property)) + '\') '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' && ( '; | |||||
| var arr1 = $deps; | |||||
| if (arr1) { | |||||
| var $propertyKey, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $propertyKey = arr1[$i += 1]; | |||||
| if ($i) { | |||||
| out += ' || '; | |||||
| } | |||||
| var $prop = it.util.getProperty($propertyKey), | |||||
| $useData = $data + $prop; | |||||
| out += ' ( ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') && (missing' + ($lvl) + ' = ' + (it.util.toQuotedString(it.opts.jsonPointers ? $propertyKey : $prop)) + ') ) '; | |||||
| } | |||||
| } | |||||
| out += ')) { '; | |||||
| var $propertyPath = 'missing' + $lvl, | |||||
| $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.opts.jsonPointers ? it.util.getPathExpr($currentErrorPath, $propertyPath, true) : $currentErrorPath + ' + ' + $propertyPath; | |||||
| } | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('dependencies') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { property: \'' + (it.util.escapeQuotes($property)) + '\', missingProperty: \'' + ($missingProperty) + '\', depsCount: ' + ($deps.length) + ', deps: \'' + (it.util.escapeQuotes($deps.length == 1 ? $deps[0] : $deps.join(", "))) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should have '; | |||||
| if ($deps.length == 1) { | |||||
| out += 'property ' + (it.util.escapeQuotes($deps[0])); | |||||
| } else { | |||||
| out += 'properties ' + (it.util.escapeQuotes($deps.join(", "))); | |||||
| } | |||||
| out += ' when property ' + (it.util.escapeQuotes($property)) + ' is present\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| } else { | |||||
| out += ' ) { '; | |||||
| var arr2 = $deps; | |||||
| if (arr2) { | |||||
| var $propertyKey, i2 = -1, | |||||
| l2 = arr2.length - 1; | |||||
| while (i2 < l2) { | |||||
| $propertyKey = arr2[i2 += 1]; | |||||
| var $prop = it.util.getProperty($propertyKey), | |||||
| $missingProperty = it.util.escapeQuotes($propertyKey), | |||||
| $useData = $data + $prop; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPath($currentErrorPath, $propertyKey, it.opts.jsonPointers); | |||||
| } | |||||
| out += ' if ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('dependencies') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { property: \'' + (it.util.escapeQuotes($property)) + '\', missingProperty: \'' + ($missingProperty) + '\', depsCount: ' + ($deps.length) + ', deps: \'' + (it.util.escapeQuotes($deps.length == 1 ? $deps[0] : $deps.join(", "))) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should have '; | |||||
| if ($deps.length == 1) { | |||||
| out += 'property ' + (it.util.escapeQuotes($deps[0])); | |||||
| } else { | |||||
| out += 'properties ' + (it.util.escapeQuotes($deps.join(", "))); | |||||
| } | |||||
| out += ' when property ' + (it.util.escapeQuotes($property)) + ' is present\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; } '; | |||||
| } | |||||
| } | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| $closingBraces += '}'; | |||||
| out += ' else { '; | |||||
| } | |||||
| } | |||||
| } | |||||
| it.errorPath = $currentErrorPath; | |||||
| var $currentBaseId = $it.baseId; | |||||
| for (var $property in $schemaDeps) { | |||||
| var $sch = $schemaDeps[$property]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| out += ' ' + ($nextValid) + ' = true; if ( ' + ($data) + (it.util.getProperty($property)) + ' !== undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' && Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($property)) + '\') '; | |||||
| } | |||||
| out += ') { '; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + it.util.getProperty($property); | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + it.util.escapeFragment($property); | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces) + ' if (' + ($errs) + ' == errors) {'; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,66 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_enum(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $i = 'i' + $lvl, | |||||
| $vSchema = 'schema' + $lvl; | |||||
| if (!$isData) { | |||||
| out += ' var ' + ($vSchema) + ' = validate.schema' + ($schemaPath) + ';'; | |||||
| } | |||||
| out += 'var ' + ($valid) + ';'; | |||||
| if ($isData) { | |||||
| out += ' if (schema' + ($lvl) + ' === undefined) ' + ($valid) + ' = true; else if (!Array.isArray(schema' + ($lvl) + ')) ' + ($valid) + ' = false; else {'; | |||||
| } | |||||
| out += '' + ($valid) + ' = false;for (var ' + ($i) + '=0; ' + ($i) + '<' + ($vSchema) + '.length; ' + ($i) + '++) if (equal(' + ($data) + ', ' + ($vSchema) + '[' + ($i) + '])) { ' + ($valid) + ' = true; break; }'; | |||||
| if ($isData) { | |||||
| out += ' } '; | |||||
| } | |||||
| out += ' if (!' + ($valid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('enum') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { allowedValues: schema' + ($lvl) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be equal to one of the allowed values\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' }'; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,150 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_format(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| if (it.opts.format === false) { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $unknownFormats = it.opts.unknownFormats, | |||||
| $allowUnknown = Array.isArray($unknownFormats); | |||||
| if ($isData) { | |||||
| var $format = 'format' + $lvl, | |||||
| $isObject = 'isObject' + $lvl, | |||||
| $formatType = 'formatType' + $lvl; | |||||
| out += ' var ' + ($format) + ' = formats[' + ($schemaValue) + ']; var ' + ($isObject) + ' = typeof ' + ($format) + ' == \'object\' && !(' + ($format) + ' instanceof RegExp) && ' + ($format) + '.validate; var ' + ($formatType) + ' = ' + ($isObject) + ' && ' + ($format) + '.type || \'string\'; if (' + ($isObject) + ') { '; | |||||
| if (it.async) { | |||||
| out += ' var async' + ($lvl) + ' = ' + ($format) + '.async; '; | |||||
| } | |||||
| out += ' ' + ($format) + ' = ' + ($format) + '.validate; } if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'string\') || '; | |||||
| } | |||||
| out += ' ('; | |||||
| if ($unknownFormats != 'ignore') { | |||||
| out += ' (' + ($schemaValue) + ' && !' + ($format) + ' '; | |||||
| if ($allowUnknown) { | |||||
| out += ' && self._opts.unknownFormats.indexOf(' + ($schemaValue) + ') == -1 '; | |||||
| } | |||||
| out += ') || '; | |||||
| } | |||||
| out += ' (' + ($format) + ' && ' + ($formatType) + ' == \'' + ($ruleType) + '\' && !(typeof ' + ($format) + ' == \'function\' ? '; | |||||
| if (it.async) { | |||||
| out += ' (async' + ($lvl) + ' ? await ' + ($format) + '(' + ($data) + ') : ' + ($format) + '(' + ($data) + ')) '; | |||||
| } else { | |||||
| out += ' ' + ($format) + '(' + ($data) + ') '; | |||||
| } | |||||
| out += ' : ' + ($format) + '.test(' + ($data) + '))))) {'; | |||||
| } else { | |||||
| var $format = it.formats[$schema]; | |||||
| if (!$format) { | |||||
| if ($unknownFormats == 'ignore') { | |||||
| it.logger.warn('unknown format "' + $schema + '" ignored in schema at path "' + it.errSchemaPath + '"'); | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| return out; | |||||
| } else if ($allowUnknown && $unknownFormats.indexOf($schema) >= 0) { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| return out; | |||||
| } else { | |||||
| throw new Error('unknown format "' + $schema + '" is used in schema at path "' + it.errSchemaPath + '"'); | |||||
| } | |||||
| } | |||||
| var $isObject = typeof $format == 'object' && !($format instanceof RegExp) && $format.validate; | |||||
| var $formatType = $isObject && $format.type || 'string'; | |||||
| if ($isObject) { | |||||
| var $async = $format.async === true; | |||||
| $format = $format.validate; | |||||
| } | |||||
| if ($formatType != $ruleType) { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| if ($async) { | |||||
| if (!it.async) throw new Error('async format in sync schema'); | |||||
| var $formatRef = 'formats' + it.util.getProperty($schema) + '.validate'; | |||||
| out += ' if (!(await ' + ($formatRef) + '(' + ($data) + '))) { '; | |||||
| } else { | |||||
| out += ' if (! '; | |||||
| var $formatRef = 'formats' + it.util.getProperty($schema); | |||||
| if ($isObject) $formatRef += '.validate'; | |||||
| if (typeof $format == 'function') { | |||||
| out += ' ' + ($formatRef) + '(' + ($data) + ') '; | |||||
| } else { | |||||
| out += ' ' + ($formatRef) + '.test(' + ($data) + ') '; | |||||
| } | |||||
| out += ') { '; | |||||
| } | |||||
| } | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('format') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { format: '; | |||||
| if ($isData) { | |||||
| out += '' + ($schemaValue); | |||||
| } else { | |||||
| out += '' + (it.util.toQuotedString($schema)); | |||||
| } | |||||
| out += ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should match format "'; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue) + ' + \''; | |||||
| } else { | |||||
| out += '' + (it.util.escapeQuotes($schema)); | |||||
| } | |||||
| out += '"\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + (it.util.toQuotedString($schema)); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,104 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_if(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $thenSch = it.schema['then'], | |||||
| $elseSch = it.schema['else'], | |||||
| $thenPresent = $thenSch !== undefined && (it.opts.strictKeywords ? typeof $thenSch == 'object' && Object.keys($thenSch).length > 0 : it.util.schemaHasRules($thenSch, it.RULES.all)), | |||||
| $elsePresent = $elseSch !== undefined && (it.opts.strictKeywords ? typeof $elseSch == 'object' && Object.keys($elseSch).length > 0 : it.util.schemaHasRules($elseSch, it.RULES.all)), | |||||
| $currentBaseId = $it.baseId; | |||||
| if ($thenPresent || $elsePresent) { | |||||
| var $ifClause; | |||||
| $it.createErrors = false; | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| out += ' var ' + ($errs) + ' = errors; var ' + ($valid) + ' = true; '; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| $it.createErrors = true; | |||||
| out += ' errors = ' + ($errs) + '; if (vErrors !== null) { if (' + ($errs) + ') vErrors.length = ' + ($errs) + '; else vErrors = null; } '; | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| if ($thenPresent) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $it.schema = it.schema['then']; | |||||
| $it.schemaPath = it.schemaPath + '.then'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/then'; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| out += ' ' + ($valid) + ' = ' + ($nextValid) + '; '; | |||||
| if ($thenPresent && $elsePresent) { | |||||
| $ifClause = 'ifClause' + $lvl; | |||||
| out += ' var ' + ($ifClause) + ' = \'then\'; '; | |||||
| } else { | |||||
| $ifClause = '\'then\''; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($elsePresent) { | |||||
| out += ' else { '; | |||||
| } | |||||
| } else { | |||||
| out += ' if (!' + ($nextValid) + ') { '; | |||||
| } | |||||
| if ($elsePresent) { | |||||
| $it.schema = it.schema['else']; | |||||
| $it.schemaPath = it.schemaPath + '.else'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/else'; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| out += ' ' + ($valid) + ' = ' + ($nextValid) + '; '; | |||||
| if ($thenPresent && $elsePresent) { | |||||
| $ifClause = 'ifClause' + $lvl; | |||||
| out += ' var ' + ($ifClause) + ' = \'else\'; '; | |||||
| } else { | |||||
| $ifClause = '\'else\''; | |||||
| } | |||||
| out += ' } '; | |||||
| } | |||||
| out += ' if (!' + ($valid) + ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('if') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { failingKeyword: ' + ($ifClause) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should match "\' + ' + ($ifClause) + ' + \'" schema\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; return false; '; | |||||
| } | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| } else { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,33 @@ | |||||
| 'use strict'; | |||||
| //all requires must be explicit because browserify won't work with dynamic requires | |||||
| module.exports = { | |||||
| '$ref': require('./ref'), | |||||
| allOf: require('./allOf'), | |||||
| anyOf: require('./anyOf'), | |||||
| '$comment': require('./comment'), | |||||
| const: require('./const'), | |||||
| contains: require('./contains'), | |||||
| dependencies: require('./dependencies'), | |||||
| 'enum': require('./enum'), | |||||
| format: require('./format'), | |||||
| 'if': require('./if'), | |||||
| items: require('./items'), | |||||
| maximum: require('./_limit'), | |||||
| minimum: require('./_limit'), | |||||
| maxItems: require('./_limitItems'), | |||||
| minItems: require('./_limitItems'), | |||||
| maxLength: require('./_limitLength'), | |||||
| minLength: require('./_limitLength'), | |||||
| maxProperties: require('./_limitProperties'), | |||||
| minProperties: require('./_limitProperties'), | |||||
| multipleOf: require('./multipleOf'), | |||||
| not: require('./not'), | |||||
| oneOf: require('./oneOf'), | |||||
| pattern: require('./pattern'), | |||||
| properties: require('./properties'), | |||||
| propertyNames: require('./propertyNames'), | |||||
| required: require('./required'), | |||||
| uniqueItems: require('./uniqueItems'), | |||||
| validate: require('./validate') | |||||
| }; | |||||
| @ -0,0 +1,141 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_items(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $idx = 'i' + $lvl, | |||||
| $dataNxt = $it.dataLevel = it.dataLevel + 1, | |||||
| $nextData = 'data' + $dataNxt, | |||||
| $currentBaseId = it.baseId; | |||||
| out += 'var ' + ($errs) + ' = errors;var ' + ($valid) + ';'; | |||||
| if (Array.isArray($schema)) { | |||||
| var $additionalItems = it.schema.additionalItems; | |||||
| if ($additionalItems === false) { | |||||
| out += ' ' + ($valid) + ' = ' + ($data) + '.length <= ' + ($schema.length) + '; '; | |||||
| var $currErrSchemaPath = $errSchemaPath; | |||||
| $errSchemaPath = it.errSchemaPath + '/additionalItems'; | |||||
| out += ' if (!' + ($valid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('additionalItems') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { limit: ' + ($schema.length) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT have more than ' + ($schema.length) + ' items\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: false , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } '; | |||||
| $errSchemaPath = $currErrSchemaPath; | |||||
| if ($breakOnError) { | |||||
| $closingBraces += '}'; | |||||
| out += ' else { '; | |||||
| } | |||||
| } | |||||
| var arr1 = $schema; | |||||
| if (arr1) { | |||||
| var $sch, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $sch = arr1[$i += 1]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| out += ' ' + ($nextValid) + ' = true; if (' + ($data) + '.length > ' + ($i) + ') { '; | |||||
| var $passData = $data + '[' + $i + ']'; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $i, it.opts.jsonPointers, true); | |||||
| $it.dataPathArr[$dataNxt] = $i; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if (typeof $additionalItems == 'object' && (it.opts.strictKeywords ? typeof $additionalItems == 'object' && Object.keys($additionalItems).length > 0 : it.util.schemaHasRules($additionalItems, it.RULES.all))) { | |||||
| $it.schema = $additionalItems; | |||||
| $it.schemaPath = it.schemaPath + '.additionalItems'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/additionalItems'; | |||||
| out += ' ' + ($nextValid) + ' = true; if (' + ($data) + '.length > ' + ($schema.length) + ') { for (var ' + ($idx) + ' = ' + ($schema.length) + '; ' + ($idx) + ' < ' + ($data) + '.length; ' + ($idx) + '++) { '; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $idx, it.opts.jsonPointers, true); | |||||
| var $passData = $data + '[' + $idx + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $idx; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (!' + ($nextValid) + ') break; '; | |||||
| } | |||||
| out += ' } } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } else if ((it.opts.strictKeywords ? typeof $schema == 'object' && Object.keys($schema).length > 0 : it.util.schemaHasRules($schema, it.RULES.all))) { | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| out += ' for (var ' + ($idx) + ' = ' + (0) + '; ' + ($idx) + ' < ' + ($data) + '.length; ' + ($idx) + '++) { '; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $idx, it.opts.jsonPointers, true); | |||||
| var $passData = $data + '[' + $idx + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $idx; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (!' + ($nextValid) + ') break; '; | |||||
| } | |||||
| out += ' }'; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces) + ' if (' + ($errs) + ' == errors) {'; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,77 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_multipleOf(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| out += 'var division' + ($lvl) + ';if ('; | |||||
| if ($isData) { | |||||
| out += ' ' + ($schemaValue) + ' !== undefined && ( typeof ' + ($schemaValue) + ' != \'number\' || '; | |||||
| } | |||||
| out += ' (division' + ($lvl) + ' = ' + ($data) + ' / ' + ($schemaValue) + ', '; | |||||
| if (it.opts.multipleOfPrecision) { | |||||
| out += ' Math.abs(Math.round(division' + ($lvl) + ') - division' + ($lvl) + ') > 1e-' + (it.opts.multipleOfPrecision) + ' '; | |||||
| } else { | |||||
| out += ' division' + ($lvl) + ' !== parseInt(division' + ($lvl) + ') '; | |||||
| } | |||||
| out += ' ) '; | |||||
| if ($isData) { | |||||
| out += ' ) '; | |||||
| } | |||||
| out += ' ) { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('multipleOf') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { multipleOf: ' + ($schemaValue) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be multiple of '; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue); | |||||
| } else { | |||||
| out += '' + ($schemaValue) + '\''; | |||||
| } | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += '} '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,84 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_not(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| if ((it.opts.strictKeywords ? typeof $schema == 'object' && Object.keys($schema).length > 0 : it.util.schemaHasRules($schema, it.RULES.all))) { | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| out += ' var ' + ($errs) + ' = errors; '; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| $it.createErrors = false; | |||||
| var $allErrorsOption; | |||||
| if ($it.opts.allErrors) { | |||||
| $allErrorsOption = $it.opts.allErrors; | |||||
| $it.opts.allErrors = false; | |||||
| } | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.createErrors = true; | |||||
| if ($allErrorsOption) $it.opts.allErrors = $allErrorsOption; | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('not') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT be valid\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else { errors = ' + ($errs) + '; if (vErrors !== null) { if (' + ($errs) + ') vErrors.length = ' + ($errs) + '; else vErrors = null; } '; | |||||
| if (it.opts.allErrors) { | |||||
| out += ' } '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('not') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT be valid\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (false) { '; | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,73 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_oneOf(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $currentBaseId = $it.baseId, | |||||
| $prevValid = 'prevValid' + $lvl, | |||||
| $passingSchemas = 'passingSchemas' + $lvl; | |||||
| out += 'var ' + ($errs) + ' = errors , ' + ($prevValid) + ' = false , ' + ($valid) + ' = false , ' + ($passingSchemas) + ' = null; '; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| var arr1 = $schema; | |||||
| if (arr1) { | |||||
| var $sch, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $sch = arr1[$i += 1]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + '[' + $i + ']'; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + $i; | |||||
| out += ' ' + (it.validate($it)) + ' '; | |||||
| $it.baseId = $currentBaseId; | |||||
| } else { | |||||
| out += ' var ' + ($nextValid) + ' = true; '; | |||||
| } | |||||
| if ($i) { | |||||
| out += ' if (' + ($nextValid) + ' && ' + ($prevValid) + ') { ' + ($valid) + ' = false; ' + ($passingSchemas) + ' = [' + ($passingSchemas) + ', ' + ($i) + ']; } else { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| out += ' if (' + ($nextValid) + ') { ' + ($valid) + ' = ' + ($prevValid) + ' = true; ' + ($passingSchemas) + ' = ' + ($i) + '; }'; | |||||
| } | |||||
| } | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += '' + ($closingBraces) + 'if (!' + ($valid) + ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('oneOf') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { passingSchemas: ' + ($passingSchemas) + ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should match exactly one schema in oneOf\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; return false; '; | |||||
| } | |||||
| } | |||||
| out += '} else { errors = ' + ($errs) + '; if (vErrors !== null) { if (' + ($errs) + ') vErrors.length = ' + ($errs) + '; else vErrors = null; }'; | |||||
| if (it.opts.allErrors) { | |||||
| out += ' } '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,75 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_pattern(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $regexp = $isData ? '(new RegExp(' + $schemaValue + '))' : it.usePattern($schema); | |||||
| out += 'if ( '; | |||||
| if ($isData) { | |||||
| out += ' (' + ($schemaValue) + ' !== undefined && typeof ' + ($schemaValue) + ' != \'string\') || '; | |||||
| } | |||||
| out += ' !' + ($regexp) + '.test(' + ($data) + ') ) { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('pattern') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { pattern: '; | |||||
| if ($isData) { | |||||
| out += '' + ($schemaValue); | |||||
| } else { | |||||
| out += '' + (it.util.toQuotedString($schema)); | |||||
| } | |||||
| out += ' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should match pattern "'; | |||||
| if ($isData) { | |||||
| out += '\' + ' + ($schemaValue) + ' + \''; | |||||
| } else { | |||||
| out += '' + (it.util.escapeQuotes($schema)); | |||||
| } | |||||
| out += '"\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + (it.util.toQuotedString($schema)); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += '} '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,330 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_properties(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| var $key = 'key' + $lvl, | |||||
| $idx = 'idx' + $lvl, | |||||
| $dataNxt = $it.dataLevel = it.dataLevel + 1, | |||||
| $nextData = 'data' + $dataNxt, | |||||
| $dataProperties = 'dataProperties' + $lvl; | |||||
| var $schemaKeys = Object.keys($schema || {}), | |||||
| $pProperties = it.schema.patternProperties || {}, | |||||
| $pPropertyKeys = Object.keys($pProperties), | |||||
| $aProperties = it.schema.additionalProperties, | |||||
| $someProperties = $schemaKeys.length || $pPropertyKeys.length, | |||||
| $noAdditional = $aProperties === false, | |||||
| $additionalIsSchema = typeof $aProperties == 'object' && Object.keys($aProperties).length, | |||||
| $removeAdditional = it.opts.removeAdditional, | |||||
| $checkAdditional = $noAdditional || $additionalIsSchema || $removeAdditional, | |||||
| $ownProperties = it.opts.ownProperties, | |||||
| $currentBaseId = it.baseId; | |||||
| var $required = it.schema.required; | |||||
| if ($required && !(it.opts.$data && $required.$data) && $required.length < it.opts.loopRequired) var $requiredHash = it.util.toHash($required); | |||||
| out += 'var ' + ($errs) + ' = errors;var ' + ($nextValid) + ' = true;'; | |||||
| if ($ownProperties) { | |||||
| out += ' var ' + ($dataProperties) + ' = undefined;'; | |||||
| } | |||||
| if ($checkAdditional) { | |||||
| if ($ownProperties) { | |||||
| out += ' ' + ($dataProperties) + ' = ' + ($dataProperties) + ' || Object.keys(' + ($data) + '); for (var ' + ($idx) + '=0; ' + ($idx) + '<' + ($dataProperties) + '.length; ' + ($idx) + '++) { var ' + ($key) + ' = ' + ($dataProperties) + '[' + ($idx) + ']; '; | |||||
| } else { | |||||
| out += ' for (var ' + ($key) + ' in ' + ($data) + ') { '; | |||||
| } | |||||
| if ($someProperties) { | |||||
| out += ' var isAdditional' + ($lvl) + ' = !(false '; | |||||
| if ($schemaKeys.length) { | |||||
| if ($schemaKeys.length > 8) { | |||||
| out += ' || validate.schema' + ($schemaPath) + '.hasOwnProperty(' + ($key) + ') '; | |||||
| } else { | |||||
| var arr1 = $schemaKeys; | |||||
| if (arr1) { | |||||
| var $propertyKey, i1 = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while (i1 < l1) { | |||||
| $propertyKey = arr1[i1 += 1]; | |||||
| out += ' || ' + ($key) + ' == ' + (it.util.toQuotedString($propertyKey)) + ' '; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($pPropertyKeys.length) { | |||||
| var arr2 = $pPropertyKeys; | |||||
| if (arr2) { | |||||
| var $pProperty, $i = -1, | |||||
| l2 = arr2.length - 1; | |||||
| while ($i < l2) { | |||||
| $pProperty = arr2[$i += 1]; | |||||
| out += ' || ' + (it.usePattern($pProperty)) + '.test(' + ($key) + ') '; | |||||
| } | |||||
| } | |||||
| } | |||||
| out += ' ); if (isAdditional' + ($lvl) + ') { '; | |||||
| } | |||||
| if ($removeAdditional == 'all') { | |||||
| out += ' delete ' + ($data) + '[' + ($key) + ']; '; | |||||
| } else { | |||||
| var $currentErrorPath = it.errorPath; | |||||
| var $additionalProperty = '\' + ' + $key + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| } | |||||
| if ($noAdditional) { | |||||
| if ($removeAdditional) { | |||||
| out += ' delete ' + ($data) + '[' + ($key) + ']; '; | |||||
| } else { | |||||
| out += ' ' + ($nextValid) + ' = false; '; | |||||
| var $currErrSchemaPath = $errSchemaPath; | |||||
| $errSchemaPath = it.errSchemaPath + '/additionalProperties'; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('additionalProperties') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { additionalProperty: \'' + ($additionalProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is an invalid additional property'; | |||||
| } else { | |||||
| out += 'should NOT have additional properties'; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: false , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| $errSchemaPath = $currErrSchemaPath; | |||||
| if ($breakOnError) { | |||||
| out += ' break; '; | |||||
| } | |||||
| } | |||||
| } else if ($additionalIsSchema) { | |||||
| if ($removeAdditional == 'failing') { | |||||
| out += ' var ' + ($errs) + ' = errors; '; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| $it.schema = $aProperties; | |||||
| $it.schemaPath = it.schemaPath + '.additionalProperties'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/additionalProperties'; | |||||
| $it.errorPath = it.opts._errorDataPathProperty ? it.errorPath : it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| var $passData = $data + '[' + $key + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $key; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| out += ' if (!' + ($nextValid) + ') { errors = ' + ($errs) + '; if (validate.errors !== null) { if (errors) validate.errors.length = errors; else validate.errors = null; } delete ' + ($data) + '[' + ($key) + ']; } '; | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| } else { | |||||
| $it.schema = $aProperties; | |||||
| $it.schemaPath = it.schemaPath + '.additionalProperties'; | |||||
| $it.errSchemaPath = it.errSchemaPath + '/additionalProperties'; | |||||
| $it.errorPath = it.opts._errorDataPathProperty ? it.errorPath : it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| var $passData = $data + '[' + $key + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $key; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (!' + ($nextValid) + ') break; '; | |||||
| } | |||||
| } | |||||
| } | |||||
| it.errorPath = $currentErrorPath; | |||||
| } | |||||
| if ($someProperties) { | |||||
| out += ' } '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| var $useDefaults = it.opts.useDefaults && !it.compositeRule; | |||||
| if ($schemaKeys.length) { | |||||
| var arr3 = $schemaKeys; | |||||
| if (arr3) { | |||||
| var $propertyKey, i3 = -1, | |||||
| l3 = arr3.length - 1; | |||||
| while (i3 < l3) { | |||||
| $propertyKey = arr3[i3 += 1]; | |||||
| var $sch = $schema[$propertyKey]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| var $prop = it.util.getProperty($propertyKey), | |||||
| $passData = $data + $prop, | |||||
| $hasDefault = $useDefaults && $sch.default !== undefined; | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = $schemaPath + $prop; | |||||
| $it.errSchemaPath = $errSchemaPath + '/' + it.util.escapeFragment($propertyKey); | |||||
| $it.errorPath = it.util.getPath(it.errorPath, $propertyKey, it.opts.jsonPointers); | |||||
| $it.dataPathArr[$dataNxt] = it.util.toQuotedString($propertyKey); | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| $code = it.util.varReplace($code, $nextData, $passData); | |||||
| var $useData = $passData; | |||||
| } else { | |||||
| var $useData = $nextData; | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; '; | |||||
| } | |||||
| if ($hasDefault) { | |||||
| out += ' ' + ($code) + ' '; | |||||
| } else { | |||||
| if ($requiredHash && $requiredHash[$propertyKey]) { | |||||
| out += ' if ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') { ' + ($nextValid) + ' = false; '; | |||||
| var $currentErrorPath = it.errorPath, | |||||
| $currErrSchemaPath = $errSchemaPath, | |||||
| $missingProperty = it.util.escapeQuotes($propertyKey); | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPath($currentErrorPath, $propertyKey, it.opts.jsonPointers); | |||||
| } | |||||
| $errSchemaPath = it.errSchemaPath + '/required'; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| $errSchemaPath = $currErrSchemaPath; | |||||
| it.errorPath = $currentErrorPath; | |||||
| out += ' } else { '; | |||||
| } else { | |||||
| if ($breakOnError) { | |||||
| out += ' if ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') { ' + ($nextValid) + ' = true; } else { '; | |||||
| } else { | |||||
| out += ' if (' + ($useData) + ' !== undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' && Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ' ) { '; | |||||
| } | |||||
| } | |||||
| out += ' ' + ($code) + ' } '; | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($pPropertyKeys.length) { | |||||
| var arr4 = $pPropertyKeys; | |||||
| if (arr4) { | |||||
| var $pProperty, i4 = -1, | |||||
| l4 = arr4.length - 1; | |||||
| while (i4 < l4) { | |||||
| $pProperty = arr4[i4 += 1]; | |||||
| var $sch = $pProperties[$pProperty]; | |||||
| if ((it.opts.strictKeywords ? typeof $sch == 'object' && Object.keys($sch).length > 0 : it.util.schemaHasRules($sch, it.RULES.all))) { | |||||
| $it.schema = $sch; | |||||
| $it.schemaPath = it.schemaPath + '.patternProperties' + it.util.getProperty($pProperty); | |||||
| $it.errSchemaPath = it.errSchemaPath + '/patternProperties/' + it.util.escapeFragment($pProperty); | |||||
| if ($ownProperties) { | |||||
| out += ' ' + ($dataProperties) + ' = ' + ($dataProperties) + ' || Object.keys(' + ($data) + '); for (var ' + ($idx) + '=0; ' + ($idx) + '<' + ($dataProperties) + '.length; ' + ($idx) + '++) { var ' + ($key) + ' = ' + ($dataProperties) + '[' + ($idx) + ']; '; | |||||
| } else { | |||||
| out += ' for (var ' + ($key) + ' in ' + ($data) + ') { '; | |||||
| } | |||||
| out += ' if (' + (it.usePattern($pProperty)) + '.test(' + ($key) + ')) { '; | |||||
| $it.errorPath = it.util.getPathExpr(it.errorPath, $key, it.opts.jsonPointers); | |||||
| var $passData = $data + '[' + $key + ']'; | |||||
| $it.dataPathArr[$dataNxt] = $key; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (!' + ($nextValid) + ') break; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else ' + ($nextValid) + ' = true; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| $closingBraces += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces) + ' if (' + ($errs) + ' == errors) {'; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,82 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_propertyNames(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $errs = 'errs__' + $lvl; | |||||
| var $it = it.util.copy(it); | |||||
| var $closingBraces = ''; | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| out += 'var ' + ($errs) + ' = errors;'; | |||||
| if ((it.opts.strictKeywords ? typeof $schema == 'object' && Object.keys($schema).length > 0 : it.util.schemaHasRules($schema, it.RULES.all))) { | |||||
| $it.schema = $schema; | |||||
| $it.schemaPath = $schemaPath; | |||||
| $it.errSchemaPath = $errSchemaPath; | |||||
| var $key = 'key' + $lvl, | |||||
| $idx = 'idx' + $lvl, | |||||
| $i = 'i' + $lvl, | |||||
| $invalidName = '\' + ' + $key + ' + \'', | |||||
| $dataNxt = $it.dataLevel = it.dataLevel + 1, | |||||
| $nextData = 'data' + $dataNxt, | |||||
| $dataProperties = 'dataProperties' + $lvl, | |||||
| $ownProperties = it.opts.ownProperties, | |||||
| $currentBaseId = it.baseId; | |||||
| if ($ownProperties) { | |||||
| out += ' var ' + ($dataProperties) + ' = undefined; '; | |||||
| } | |||||
| if ($ownProperties) { | |||||
| out += ' ' + ($dataProperties) + ' = ' + ($dataProperties) + ' || Object.keys(' + ($data) + '); for (var ' + ($idx) + '=0; ' + ($idx) + '<' + ($dataProperties) + '.length; ' + ($idx) + '++) { var ' + ($key) + ' = ' + ($dataProperties) + '[' + ($idx) + ']; '; | |||||
| } else { | |||||
| out += ' for (var ' + ($key) + ' in ' + ($data) + ') { '; | |||||
| } | |||||
| out += ' var startErrs' + ($lvl) + ' = errors; '; | |||||
| var $passData = $key; | |||||
| var $wasComposite = it.compositeRule; | |||||
| it.compositeRule = $it.compositeRule = true; | |||||
| var $code = it.validate($it); | |||||
| $it.baseId = $currentBaseId; | |||||
| if (it.util.varOccurences($code, $nextData) < 2) { | |||||
| out += ' ' + (it.util.varReplace($code, $nextData, $passData)) + ' '; | |||||
| } else { | |||||
| out += ' var ' + ($nextData) + ' = ' + ($passData) + '; ' + ($code) + ' '; | |||||
| } | |||||
| it.compositeRule = $it.compositeRule = $wasComposite; | |||||
| out += ' if (!' + ($nextValid) + ') { for (var ' + ($i) + '=startErrs' + ($lvl) + '; ' + ($i) + '<errors; ' + ($i) + '++) { vErrors[' + ($i) + '].propertyName = ' + ($key) + '; } var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('propertyNames') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { propertyName: \'' + ($invalidName) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'property name \\\'' + ($invalidName) + '\\\' is invalid\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; return false; '; | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' break; '; | |||||
| } | |||||
| out += ' } }'; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces) + ' if (' + ($errs) + ' == errors) {'; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,124 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_ref(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $async, $refCode; | |||||
| if ($schema == '#' || $schema == '#/') { | |||||
| if (it.isRoot) { | |||||
| $async = it.async; | |||||
| $refCode = 'validate'; | |||||
| } else { | |||||
| $async = it.root.schema.$async === true; | |||||
| $refCode = 'root.refVal[0]'; | |||||
| } | |||||
| } else { | |||||
| var $refVal = it.resolveRef(it.baseId, $schema, it.isRoot); | |||||
| if ($refVal === undefined) { | |||||
| var $message = it.MissingRefError.message(it.baseId, $schema); | |||||
| if (it.opts.missingRefs == 'fail') { | |||||
| it.logger.error($message); | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('$ref') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { ref: \'' + (it.util.escapeQuotes($schema)) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'can\\\'t resolve reference ' + (it.util.escapeQuotes($schema)) + '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: ' + (it.util.toQuotedString($schema)) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (false) { '; | |||||
| } | |||||
| } else if (it.opts.missingRefs == 'ignore') { | |||||
| it.logger.warn($message); | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| } else { | |||||
| throw new it.MissingRefError(it.baseId, $schema, $message); | |||||
| } | |||||
| } else if ($refVal.inline) { | |||||
| var $it = it.util.copy(it); | |||||
| $it.level++; | |||||
| var $nextValid = 'valid' + $it.level; | |||||
| $it.schema = $refVal.schema; | |||||
| $it.schemaPath = ''; | |||||
| $it.errSchemaPath = $schema; | |||||
| var $code = it.validate($it).replace(/validate\.schema/g, $refVal.code); | |||||
| out += ' ' + ($code) + ' '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($nextValid) + ') { '; | |||||
| } | |||||
| } else { | |||||
| $async = $refVal.$async === true || (it.async && $refVal.$async !== false); | |||||
| $refCode = $refVal.code; | |||||
| } | |||||
| } | |||||
| if ($refCode) { | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; | |||||
| if (it.opts.passContext) { | |||||
| out += ' ' + ($refCode) + '.call(this, '; | |||||
| } else { | |||||
| out += ' ' + ($refCode) + '( '; | |||||
| } | |||||
| out += ' ' + ($data) + ', (dataPath || \'\')'; | |||||
| if (it.errorPath != '""') { | |||||
| out += ' + ' + (it.errorPath); | |||||
| } | |||||
| var $parentData = $dataLvl ? 'data' + (($dataLvl - 1) || '') : 'parentData', | |||||
| $parentDataProperty = $dataLvl ? it.dataPathArr[$dataLvl] : 'parentDataProperty'; | |||||
| out += ' , ' + ($parentData) + ' , ' + ($parentDataProperty) + ', rootData) '; | |||||
| var __callValidate = out; | |||||
| out = $$outStack.pop(); | |||||
| if ($async) { | |||||
| if (!it.async) throw new Error('async schema referenced by sync schema'); | |||||
| if ($breakOnError) { | |||||
| out += ' var ' + ($valid) + '; '; | |||||
| } | |||||
| out += ' try { await ' + (__callValidate) + '; '; | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($valid) + ' = true; '; | |||||
| } | |||||
| out += ' } catch (e) { if (!(e instanceof ValidationError)) throw e; if (vErrors === null) vErrors = e.errors; else vErrors = vErrors.concat(e.errors); errors = vErrors.length; '; | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($valid) + ' = false; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' if (' + ($valid) + ') { '; | |||||
| } | |||||
| } else { | |||||
| out += ' if (!' + (__callValidate) + ') { if (vErrors === null) vErrors = ' + ($refCode) + '.errors; else vErrors = vErrors.concat(' + ($refCode) + '.errors); errors = vErrors.length; } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,270 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_required(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| var $vSchema = 'schema' + $lvl; | |||||
| if (!$isData) { | |||||
| if ($schema.length < it.opts.loopRequired && it.schema.properties && Object.keys(it.schema.properties).length) { | |||||
| var $required = []; | |||||
| var arr1 = $schema; | |||||
| if (arr1) { | |||||
| var $property, i1 = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while (i1 < l1) { | |||||
| $property = arr1[i1 += 1]; | |||||
| var $propertySch = it.schema.properties[$property]; | |||||
| if (!($propertySch && (it.opts.strictKeywords ? typeof $propertySch == 'object' && Object.keys($propertySch).length > 0 : it.util.schemaHasRules($propertySch, it.RULES.all)))) { | |||||
| $required[$required.length] = $property; | |||||
| } | |||||
| } | |||||
| } | |||||
| } else { | |||||
| var $required = $schema; | |||||
| } | |||||
| } | |||||
| if ($isData || $required.length) { | |||||
| var $currentErrorPath = it.errorPath, | |||||
| $loopRequired = $isData || $required.length >= it.opts.loopRequired, | |||||
| $ownProperties = it.opts.ownProperties; | |||||
| if ($breakOnError) { | |||||
| out += ' var missing' + ($lvl) + '; '; | |||||
| if ($loopRequired) { | |||||
| if (!$isData) { | |||||
| out += ' var ' + ($vSchema) + ' = validate.schema' + ($schemaPath) + '; '; | |||||
| } | |||||
| var $i = 'i' + $lvl, | |||||
| $propertyPath = 'schema' + $lvl + '[' + $i + ']', | |||||
| $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPathExpr($currentErrorPath, $propertyPath, it.opts.jsonPointers); | |||||
| } | |||||
| out += ' var ' + ($valid) + ' = true; '; | |||||
| if ($isData) { | |||||
| out += ' if (schema' + ($lvl) + ' === undefined) ' + ($valid) + ' = true; else if (!Array.isArray(schema' + ($lvl) + ')) ' + ($valid) + ' = false; else {'; | |||||
| } | |||||
| out += ' for (var ' + ($i) + ' = 0; ' + ($i) + ' < ' + ($vSchema) + '.length; ' + ($i) + '++) { ' + ($valid) + ' = ' + ($data) + '[' + ($vSchema) + '[' + ($i) + ']] !== undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' && Object.prototype.hasOwnProperty.call(' + ($data) + ', ' + ($vSchema) + '[' + ($i) + ']) '; | |||||
| } | |||||
| out += '; if (!' + ($valid) + ') break; } '; | |||||
| if ($isData) { | |||||
| out += ' } '; | |||||
| } | |||||
| out += ' if (!' + ($valid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else { '; | |||||
| } else { | |||||
| out += ' if ( '; | |||||
| var arr2 = $required; | |||||
| if (arr2) { | |||||
| var $propertyKey, $i = -1, | |||||
| l2 = arr2.length - 1; | |||||
| while ($i < l2) { | |||||
| $propertyKey = arr2[$i += 1]; | |||||
| if ($i) { | |||||
| out += ' || '; | |||||
| } | |||||
| var $prop = it.util.getProperty($propertyKey), | |||||
| $useData = $data + $prop; | |||||
| out += ' ( ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') && (missing' + ($lvl) + ' = ' + (it.util.toQuotedString(it.opts.jsonPointers ? $propertyKey : $prop)) + ') ) '; | |||||
| } | |||||
| } | |||||
| out += ') { '; | |||||
| var $propertyPath = 'missing' + $lvl, | |||||
| $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.opts.jsonPointers ? it.util.getPathExpr($currentErrorPath, $propertyPath, true) : $currentErrorPath + ' + ' + $propertyPath; | |||||
| } | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else { '; | |||||
| } | |||||
| } else { | |||||
| if ($loopRequired) { | |||||
| if (!$isData) { | |||||
| out += ' var ' + ($vSchema) + ' = validate.schema' + ($schemaPath) + '; '; | |||||
| } | |||||
| var $i = 'i' + $lvl, | |||||
| $propertyPath = 'schema' + $lvl + '[' + $i + ']', | |||||
| $missingProperty = '\' + ' + $propertyPath + ' + \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPathExpr($currentErrorPath, $propertyPath, it.opts.jsonPointers); | |||||
| } | |||||
| if ($isData) { | |||||
| out += ' if (' + ($vSchema) + ' && !Array.isArray(' + ($vSchema) + ')) { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; } else if (' + ($vSchema) + ' !== undefined) { '; | |||||
| } | |||||
| out += ' for (var ' + ($i) + ' = 0; ' + ($i) + ' < ' + ($vSchema) + '.length; ' + ($i) + '++) { if (' + ($data) + '[' + ($vSchema) + '[' + ($i) + ']] === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', ' + ($vSchema) + '[' + ($i) + ']) '; | |||||
| } | |||||
| out += ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; } } '; | |||||
| if ($isData) { | |||||
| out += ' } '; | |||||
| } | |||||
| } else { | |||||
| var arr3 = $required; | |||||
| if (arr3) { | |||||
| var $propertyKey, i3 = -1, | |||||
| l3 = arr3.length - 1; | |||||
| while (i3 < l3) { | |||||
| $propertyKey = arr3[i3 += 1]; | |||||
| var $prop = it.util.getProperty($propertyKey), | |||||
| $missingProperty = it.util.escapeQuotes($propertyKey), | |||||
| $useData = $data + $prop; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| it.errorPath = it.util.getPath($currentErrorPath, $propertyKey, it.opts.jsonPointers); | |||||
| } | |||||
| out += ' if ( ' + ($useData) + ' === undefined '; | |||||
| if ($ownProperties) { | |||||
| out += ' || ! Object.prototype.hasOwnProperty.call(' + ($data) + ', \'' + (it.util.escapeQuotes($propertyKey)) + '\') '; | |||||
| } | |||||
| out += ') { var err = '; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('required') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { missingProperty: \'' + ($missingProperty) + '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \''; | |||||
| if (it.opts._errorDataPathProperty) { | |||||
| out += 'is a required property'; | |||||
| } else { | |||||
| out += 'should have required property \\\'' + ($missingProperty) + '\\\''; | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| out += '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; } '; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| it.errorPath = $currentErrorPath; | |||||
| } else if ($breakOnError) { | |||||
| out += ' if (true) {'; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,86 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_uniqueItems(it, $keyword, $ruleType) { | |||||
| var out = ' '; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| var $isData = it.opts.$data && $schema && $schema.$data, | |||||
| $schemaValue; | |||||
| if ($isData) { | |||||
| out += ' var schema' + ($lvl) + ' = ' + (it.util.getData($schema.$data, $dataLvl, it.dataPathArr)) + '; '; | |||||
| $schemaValue = 'schema' + $lvl; | |||||
| } else { | |||||
| $schemaValue = $schema; | |||||
| } | |||||
| if (($schema || $isData) && it.opts.uniqueItems !== false) { | |||||
| if ($isData) { | |||||
| out += ' var ' + ($valid) + '; if (' + ($schemaValue) + ' === false || ' + ($schemaValue) + ' === undefined) ' + ($valid) + ' = true; else if (typeof ' + ($schemaValue) + ' != \'boolean\') ' + ($valid) + ' = false; else { '; | |||||
| } | |||||
| out += ' var i = ' + ($data) + '.length , ' + ($valid) + ' = true , j; if (i > 1) { '; | |||||
| var $itemType = it.schema.items && it.schema.items.type, | |||||
| $typeIsArray = Array.isArray($itemType); | |||||
| if (!$itemType || $itemType == 'object' || $itemType == 'array' || ($typeIsArray && ($itemType.indexOf('object') >= 0 || $itemType.indexOf('array') >= 0))) { | |||||
| out += ' outer: for (;i--;) { for (j = i; j--;) { if (equal(' + ($data) + '[i], ' + ($data) + '[j])) { ' + ($valid) + ' = false; break outer; } } } '; | |||||
| } else { | |||||
| out += ' var itemIndices = {}, item; for (;i--;) { var item = ' + ($data) + '[i]; '; | |||||
| var $method = 'checkDataType' + ($typeIsArray ? 's' : ''); | |||||
| out += ' if (' + (it.util[$method]($itemType, 'item', true)) + ') continue; '; | |||||
| if ($typeIsArray) { | |||||
| out += ' if (typeof item == \'string\') item = \'"\' + item; '; | |||||
| } | |||||
| out += ' if (typeof itemIndices[item] == \'number\') { ' + ($valid) + ' = false; j = itemIndices[item]; break; } itemIndices[item] = i; } '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($isData) { | |||||
| out += ' } '; | |||||
| } | |||||
| out += ' if (!' + ($valid) + ') { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ('uniqueItems') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { i: i, j: j } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should NOT have duplicate items (items ## \' + j + \' and \' + i + \' are identical)\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: '; | |||||
| if ($isData) { | |||||
| out += 'validate.schema' + ($schemaPath); | |||||
| } else { | |||||
| out += '' + ($schema); | |||||
| } | |||||
| out += ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } '; | |||||
| if ($breakOnError) { | |||||
| out += ' else { '; | |||||
| } | |||||
| } else { | |||||
| if ($breakOnError) { | |||||
| out += ' if (true) { '; | |||||
| } | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,494 @@ | |||||
| 'use strict'; | |||||
| module.exports = function generate_validate(it, $keyword, $ruleType) { | |||||
| var out = ''; | |||||
| var $async = it.schema.$async === true, | |||||
| $refKeywords = it.util.schemaHasRulesExcept(it.schema, it.RULES.all, '$ref'), | |||||
| $id = it.self._getId(it.schema); | |||||
| if (it.opts.strictKeywords) { | |||||
| var $unknownKwd = it.util.schemaUnknownRules(it.schema, it.RULES.keywords); | |||||
| if ($unknownKwd) { | |||||
| var $keywordsMsg = 'unknown keyword: ' + $unknownKwd; | |||||
| if (it.opts.strictKeywords === 'log') it.logger.warn($keywordsMsg); | |||||
| else throw new Error($keywordsMsg); | |||||
| } | |||||
| } | |||||
| if (it.isTop) { | |||||
| out += ' var validate = '; | |||||
| if ($async) { | |||||
| it.async = true; | |||||
| out += 'async '; | |||||
| } | |||||
| out += 'function(data, dataPath, parentData, parentDataProperty, rootData) { \'use strict\'; '; | |||||
| if ($id && (it.opts.sourceCode || it.opts.processCode)) { | |||||
| out += ' ' + ('/\*# sourceURL=' + $id + ' */') + ' '; | |||||
| } | |||||
| } | |||||
| if (typeof it.schema == 'boolean' || !($refKeywords || it.schema.$ref)) { | |||||
| var $keyword = 'false schema'; | |||||
| var $lvl = it.level; | |||||
| var $dataLvl = it.dataLevel; | |||||
| var $schema = it.schema[$keyword]; | |||||
| var $schemaPath = it.schemaPath + it.util.getProperty($keyword); | |||||
| var $errSchemaPath = it.errSchemaPath + '/' + $keyword; | |||||
| var $breakOnError = !it.opts.allErrors; | |||||
| var $errorKeyword; | |||||
| var $data = 'data' + ($dataLvl || ''); | |||||
| var $valid = 'valid' + $lvl; | |||||
| if (it.schema === false) { | |||||
| if (it.isTop) { | |||||
| $breakOnError = true; | |||||
| } else { | |||||
| out += ' var ' + ($valid) + ' = false; '; | |||||
| } | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'false schema') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: {} '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'boolean schema is false\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: false , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| } else { | |||||
| if (it.isTop) { | |||||
| if ($async) { | |||||
| out += ' return data; '; | |||||
| } else { | |||||
| out += ' validate.errors = null; return true; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var ' + ($valid) + ' = true; '; | |||||
| } | |||||
| } | |||||
| if (it.isTop) { | |||||
| out += ' }; return validate; '; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| if (it.isTop) { | |||||
| var $top = it.isTop, | |||||
| $lvl = it.level = 0, | |||||
| $dataLvl = it.dataLevel = 0, | |||||
| $data = 'data'; | |||||
| it.rootId = it.resolve.fullPath(it.self._getId(it.root.schema)); | |||||
| it.baseId = it.baseId || it.rootId; | |||||
| delete it.isTop; | |||||
| it.dataPathArr = [undefined]; | |||||
| if (it.schema.default !== undefined && it.opts.useDefaults && it.opts.strictDefaults) { | |||||
| var $defaultMsg = 'default is ignored in the schema root'; | |||||
| if (it.opts.strictDefaults === 'log') it.logger.warn($defaultMsg); | |||||
| else throw new Error($defaultMsg); | |||||
| } | |||||
| out += ' var vErrors = null; '; | |||||
| out += ' var errors = 0; '; | |||||
| out += ' if (rootData === undefined) rootData = data; '; | |||||
| } else { | |||||
| var $lvl = it.level, | |||||
| $dataLvl = it.dataLevel, | |||||
| $data = 'data' + ($dataLvl || ''); | |||||
| if ($id) it.baseId = it.resolve.url(it.baseId, $id); | |||||
| if ($async && !it.async) throw new Error('async schema in sync schema'); | |||||
| out += ' var errs_' + ($lvl) + ' = errors;'; | |||||
| } | |||||
| var $valid = 'valid' + $lvl, | |||||
| $breakOnError = !it.opts.allErrors, | |||||
| $closingBraces1 = '', | |||||
| $closingBraces2 = ''; | |||||
| var $errorKeyword; | |||||
| var $typeSchema = it.schema.type, | |||||
| $typeIsArray = Array.isArray($typeSchema); | |||||
| if ($typeSchema && it.opts.nullable && it.schema.nullable === true) { | |||||
| if ($typeIsArray) { | |||||
| if ($typeSchema.indexOf('null') == -1) $typeSchema = $typeSchema.concat('null'); | |||||
| } else if ($typeSchema != 'null') { | |||||
| $typeSchema = [$typeSchema, 'null']; | |||||
| $typeIsArray = true; | |||||
| } | |||||
| } | |||||
| if ($typeIsArray && $typeSchema.length == 1) { | |||||
| $typeSchema = $typeSchema[0]; | |||||
| $typeIsArray = false; | |||||
| } | |||||
| if (it.schema.$ref && $refKeywords) { | |||||
| if (it.opts.extendRefs == 'fail') { | |||||
| throw new Error('$ref: validation keywords used in schema at path "' + it.errSchemaPath + '" (see option extendRefs)'); | |||||
| } else if (it.opts.extendRefs !== true) { | |||||
| $refKeywords = false; | |||||
| it.logger.warn('$ref: keywords ignored in schema at path "' + it.errSchemaPath + '"'); | |||||
| } | |||||
| } | |||||
| if (it.schema.$comment && it.opts.$comment) { | |||||
| out += ' ' + (it.RULES.all.$comment.code(it, '$comment')); | |||||
| } | |||||
| if ($typeSchema) { | |||||
| if (it.opts.coerceTypes) { | |||||
| var $coerceToTypes = it.util.coerceToTypes(it.opts.coerceTypes, $typeSchema); | |||||
| } | |||||
| var $rulesGroup = it.RULES.types[$typeSchema]; | |||||
| if ($coerceToTypes || $typeIsArray || $rulesGroup === true || ($rulesGroup && !$shouldUseGroup($rulesGroup))) { | |||||
| var $schemaPath = it.schemaPath + '.type', | |||||
| $errSchemaPath = it.errSchemaPath + '/type'; | |||||
| var $schemaPath = it.schemaPath + '.type', | |||||
| $errSchemaPath = it.errSchemaPath + '/type', | |||||
| $method = $typeIsArray ? 'checkDataTypes' : 'checkDataType'; | |||||
| out += ' if (' + (it.util[$method]($typeSchema, $data, true)) + ') { '; | |||||
| if ($coerceToTypes) { | |||||
| var $dataType = 'dataType' + $lvl, | |||||
| $coerced = 'coerced' + $lvl; | |||||
| out += ' var ' + ($dataType) + ' = typeof ' + ($data) + '; '; | |||||
| if (it.opts.coerceTypes == 'array') { | |||||
| out += ' if (' + ($dataType) + ' == \'object\' && Array.isArray(' + ($data) + ')) ' + ($dataType) + ' = \'array\'; '; | |||||
| } | |||||
| out += ' var ' + ($coerced) + ' = undefined; '; | |||||
| var $bracesCoercion = ''; | |||||
| var arr1 = $coerceToTypes; | |||||
| if (arr1) { | |||||
| var $type, $i = -1, | |||||
| l1 = arr1.length - 1; | |||||
| while ($i < l1) { | |||||
| $type = arr1[$i += 1]; | |||||
| if ($i) { | |||||
| out += ' if (' + ($coerced) + ' === undefined) { '; | |||||
| $bracesCoercion += '}'; | |||||
| } | |||||
| if (it.opts.coerceTypes == 'array' && $type != 'array') { | |||||
| out += ' if (' + ($dataType) + ' == \'array\' && ' + ($data) + '.length == 1) { ' + ($coerced) + ' = ' + ($data) + ' = ' + ($data) + '[0]; ' + ($dataType) + ' = typeof ' + ($data) + '; } '; | |||||
| } | |||||
| if ($type == 'string') { | |||||
| out += ' if (' + ($dataType) + ' == \'number\' || ' + ($dataType) + ' == \'boolean\') ' + ($coerced) + ' = \'\' + ' + ($data) + '; else if (' + ($data) + ' === null) ' + ($coerced) + ' = \'\'; '; | |||||
| } else if ($type == 'number' || $type == 'integer') { | |||||
| out += ' if (' + ($dataType) + ' == \'boolean\' || ' + ($data) + ' === null || (' + ($dataType) + ' == \'string\' && ' + ($data) + ' && ' + ($data) + ' == +' + ($data) + ' '; | |||||
| if ($type == 'integer') { | |||||
| out += ' && !(' + ($data) + ' % 1)'; | |||||
| } | |||||
| out += ')) ' + ($coerced) + ' = +' + ($data) + '; '; | |||||
| } else if ($type == 'boolean') { | |||||
| out += ' if (' + ($data) + ' === \'false\' || ' + ($data) + ' === 0 || ' + ($data) + ' === null) ' + ($coerced) + ' = false; else if (' + ($data) + ' === \'true\' || ' + ($data) + ' === 1) ' + ($coerced) + ' = true; '; | |||||
| } else if ($type == 'null') { | |||||
| out += ' if (' + ($data) + ' === \'\' || ' + ($data) + ' === 0 || ' + ($data) + ' === false) ' + ($coerced) + ' = null; '; | |||||
| } else if (it.opts.coerceTypes == 'array' && $type == 'array') { | |||||
| out += ' if (' + ($dataType) + ' == \'string\' || ' + ($dataType) + ' == \'number\' || ' + ($dataType) + ' == \'boolean\' || ' + ($data) + ' == null) ' + ($coerced) + ' = [' + ($data) + ']; '; | |||||
| } | |||||
| } | |||||
| } | |||||
| out += ' ' + ($bracesCoercion) + ' if (' + ($coerced) + ' === undefined) { '; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'type') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { type: \''; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be '; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } else { '; | |||||
| var $parentData = $dataLvl ? 'data' + (($dataLvl - 1) || '') : 'parentData', | |||||
| $parentDataProperty = $dataLvl ? it.dataPathArr[$dataLvl] : 'parentDataProperty'; | |||||
| out += ' ' + ($data) + ' = ' + ($coerced) + '; '; | |||||
| if (!$dataLvl) { | |||||
| out += 'if (' + ($parentData) + ' !== undefined)'; | |||||
| } | |||||
| out += ' ' + ($parentData) + '[' + ($parentDataProperty) + '] = ' + ($coerced) + '; } '; | |||||
| } else { | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'type') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { type: \''; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be '; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| } | |||||
| out += ' } '; | |||||
| } | |||||
| } | |||||
| if (it.schema.$ref && !$refKeywords) { | |||||
| out += ' ' + (it.RULES.all.$ref.code(it, '$ref')) + ' '; | |||||
| if ($breakOnError) { | |||||
| out += ' } if (errors === '; | |||||
| if ($top) { | |||||
| out += '0'; | |||||
| } else { | |||||
| out += 'errs_' + ($lvl); | |||||
| } | |||||
| out += ') { '; | |||||
| $closingBraces2 += '}'; | |||||
| } | |||||
| } else { | |||||
| var arr2 = it.RULES; | |||||
| if (arr2) { | |||||
| var $rulesGroup, i2 = -1, | |||||
| l2 = arr2.length - 1; | |||||
| while (i2 < l2) { | |||||
| $rulesGroup = arr2[i2 += 1]; | |||||
| if ($shouldUseGroup($rulesGroup)) { | |||||
| if ($rulesGroup.type) { | |||||
| out += ' if (' + (it.util.checkDataType($rulesGroup.type, $data)) + ') { '; | |||||
| } | |||||
| if (it.opts.useDefaults) { | |||||
| if ($rulesGroup.type == 'object' && it.schema.properties) { | |||||
| var $schema = it.schema.properties, | |||||
| $schemaKeys = Object.keys($schema); | |||||
| var arr3 = $schemaKeys; | |||||
| if (arr3) { | |||||
| var $propertyKey, i3 = -1, | |||||
| l3 = arr3.length - 1; | |||||
| while (i3 < l3) { | |||||
| $propertyKey = arr3[i3 += 1]; | |||||
| var $sch = $schema[$propertyKey]; | |||||
| if ($sch.default !== undefined) { | |||||
| var $passData = $data + it.util.getProperty($propertyKey); | |||||
| if (it.compositeRule) { | |||||
| if (it.opts.strictDefaults) { | |||||
| var $defaultMsg = 'default is ignored for: ' + $passData; | |||||
| if (it.opts.strictDefaults === 'log') it.logger.warn($defaultMsg); | |||||
| else throw new Error($defaultMsg); | |||||
| } | |||||
| } else { | |||||
| out += ' if (' + ($passData) + ' === undefined '; | |||||
| if (it.opts.useDefaults == 'empty') { | |||||
| out += ' || ' + ($passData) + ' === null || ' + ($passData) + ' === \'\' '; | |||||
| } | |||||
| out += ' ) ' + ($passData) + ' = '; | |||||
| if (it.opts.useDefaults == 'shared') { | |||||
| out += ' ' + (it.useDefault($sch.default)) + ' '; | |||||
| } else { | |||||
| out += ' ' + (JSON.stringify($sch.default)) + ' '; | |||||
| } | |||||
| out += '; '; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } else if ($rulesGroup.type == 'array' && Array.isArray(it.schema.items)) { | |||||
| var arr4 = it.schema.items; | |||||
| if (arr4) { | |||||
| var $sch, $i = -1, | |||||
| l4 = arr4.length - 1; | |||||
| while ($i < l4) { | |||||
| $sch = arr4[$i += 1]; | |||||
| if ($sch.default !== undefined) { | |||||
| var $passData = $data + '[' + $i + ']'; | |||||
| if (it.compositeRule) { | |||||
| if (it.opts.strictDefaults) { | |||||
| var $defaultMsg = 'default is ignored for: ' + $passData; | |||||
| if (it.opts.strictDefaults === 'log') it.logger.warn($defaultMsg); | |||||
| else throw new Error($defaultMsg); | |||||
| } | |||||
| } else { | |||||
| out += ' if (' + ($passData) + ' === undefined '; | |||||
| if (it.opts.useDefaults == 'empty') { | |||||
| out += ' || ' + ($passData) + ' === null || ' + ($passData) + ' === \'\' '; | |||||
| } | |||||
| out += ' ) ' + ($passData) + ' = '; | |||||
| if (it.opts.useDefaults == 'shared') { | |||||
| out += ' ' + (it.useDefault($sch.default)) + ' '; | |||||
| } else { | |||||
| out += ' ' + (JSON.stringify($sch.default)) + ' '; | |||||
| } | |||||
| out += '; '; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| var arr5 = $rulesGroup.rules; | |||||
| if (arr5) { | |||||
| var $rule, i5 = -1, | |||||
| l5 = arr5.length - 1; | |||||
| while (i5 < l5) { | |||||
| $rule = arr5[i5 += 1]; | |||||
| if ($shouldUseRule($rule)) { | |||||
| var $code = $rule.code(it, $rule.keyword, $rulesGroup.type); | |||||
| if ($code) { | |||||
| out += ' ' + ($code) + ' '; | |||||
| if ($breakOnError) { | |||||
| $closingBraces1 += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces1) + ' '; | |||||
| $closingBraces1 = ''; | |||||
| } | |||||
| if ($rulesGroup.type) { | |||||
| out += ' } '; | |||||
| if ($typeSchema && $typeSchema === $rulesGroup.type && !$coerceToTypes) { | |||||
| out += ' else { '; | |||||
| var $schemaPath = it.schemaPath + '.type', | |||||
| $errSchemaPath = it.errSchemaPath + '/type'; | |||||
| var $$outStack = $$outStack || []; | |||||
| $$outStack.push(out); | |||||
| out = ''; /* istanbul ignore else */ | |||||
| if (it.createErrors !== false) { | |||||
| out += ' { keyword: \'' + ($errorKeyword || 'type') + '\' , dataPath: (dataPath || \'\') + ' + (it.errorPath) + ' , schemaPath: ' + (it.util.toQuotedString($errSchemaPath)) + ' , params: { type: \''; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' } '; | |||||
| if (it.opts.messages !== false) { | |||||
| out += ' , message: \'should be '; | |||||
| if ($typeIsArray) { | |||||
| out += '' + ($typeSchema.join(",")); | |||||
| } else { | |||||
| out += '' + ($typeSchema); | |||||
| } | |||||
| out += '\' '; | |||||
| } | |||||
| if (it.opts.verbose) { | |||||
| out += ' , schema: validate.schema' + ($schemaPath) + ' , parentSchema: validate.schema' + (it.schemaPath) + ' , data: ' + ($data) + ' '; | |||||
| } | |||||
| out += ' } '; | |||||
| } else { | |||||
| out += ' {} '; | |||||
| } | |||||
| var __err = out; | |||||
| out = $$outStack.pop(); | |||||
| if (!it.compositeRule && $breakOnError) { | |||||
| /* istanbul ignore if */ | |||||
| if (it.async) { | |||||
| out += ' throw new ValidationError([' + (__err) + ']); '; | |||||
| } else { | |||||
| out += ' validate.errors = [' + (__err) + ']; return false; '; | |||||
| } | |||||
| } else { | |||||
| out += ' var err = ' + (__err) + '; if (vErrors === null) vErrors = [err]; else vErrors.push(err); errors++; '; | |||||
| } | |||||
| out += ' } '; | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' if (errors === '; | |||||
| if ($top) { | |||||
| out += '0'; | |||||
| } else { | |||||
| out += 'errs_' + ($lvl); | |||||
| } | |||||
| out += ') { '; | |||||
| $closingBraces2 += '}'; | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| if ($breakOnError) { | |||||
| out += ' ' + ($closingBraces2) + ' '; | |||||
| } | |||||
| if ($top) { | |||||
| if ($async) { | |||||
| out += ' if (errors === 0) return data; '; | |||||
| out += ' else throw new ValidationError(vErrors); '; | |||||
| } else { | |||||
| out += ' validate.errors = vErrors; '; | |||||
| out += ' return errors === 0; '; | |||||
| } | |||||
| out += ' }; return validate;'; | |||||
| } else { | |||||
| out += ' var ' + ($valid) + ' = errors === errs_' + ($lvl) + ';'; | |||||
| } | |||||
| out = it.util.cleanUpCode(out); | |||||
| if ($top) { | |||||
| out = it.util.finalCleanUpCode(out, $async); | |||||
| } | |||||
| function $shouldUseGroup($rulesGroup) { | |||||
| var rules = $rulesGroup.rules; | |||||
| for (var i = 0; i < rules.length; i++) | |||||
| if ($shouldUseRule(rules[i])) return true; | |||||
| } | |||||
| function $shouldUseRule($rule) { | |||||
| return it.schema[$rule.keyword] !== undefined || ($rule.implements && $ruleImplementsSomeKeyword($rule)); | |||||
| } | |||||
| function $ruleImplementsSomeKeyword($rule) { | |||||
| var impl = $rule.implements; | |||||
| for (var i = 0; i < impl.length; i++) | |||||
| if (it.schema[impl[i]] !== undefined) return true; | |||||
| } | |||||
| return out; | |||||
| } | |||||
| @ -0,0 +1,146 @@ | |||||
| 'use strict'; | |||||
| var IDENTIFIER = /^[a-z_$][a-z0-9_$-]*$/i; | |||||
| var customRuleCode = require('./dotjs/custom'); | |||||
| var definitionSchema = require('./definition_schema'); | |||||
| module.exports = { | |||||
| add: addKeyword, | |||||
| get: getKeyword, | |||||
| remove: removeKeyword, | |||||
| validate: validateKeyword | |||||
| }; | |||||
| /** | |||||
| * Define custom keyword | |||||
| * @this Ajv | |||||
| * @param {String} keyword custom keyword, should be unique (including different from all standard, custom and macro keywords). | |||||
| * @param {Object} definition keyword definition object with properties `type` (type(s) which the keyword applies to), `validate` or `compile`. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function addKeyword(keyword, definition) { | |||||
| /* jshint validthis: true */ | |||||
| /* eslint no-shadow: 0 */ | |||||
| var RULES = this.RULES; | |||||
| if (RULES.keywords[keyword]) | |||||
| throw new Error('Keyword ' + keyword + ' is already defined'); | |||||
| if (!IDENTIFIER.test(keyword)) | |||||
| throw new Error('Keyword ' + keyword + ' is not a valid identifier'); | |||||
| if (definition) { | |||||
| this.validateKeyword(definition, true); | |||||
| var dataType = definition.type; | |||||
| if (Array.isArray(dataType)) { | |||||
| for (var i=0; i<dataType.length; i++) | |||||
| _addRule(keyword, dataType[i], definition); | |||||
| } else { | |||||
| _addRule(keyword, dataType, definition); | |||||
| } | |||||
| var metaSchema = definition.metaSchema; | |||||
| if (metaSchema) { | |||||
| if (definition.$data && this._opts.$data) { | |||||
| metaSchema = { | |||||
| anyOf: [ | |||||
| metaSchema, | |||||
| { '$ref': 'https://raw.githubusercontent.com/epoberezkin/ajv/master/lib/refs/data.json#' } | |||||
| ] | |||||
| }; | |||||
| } | |||||
| definition.validateSchema = this.compile(metaSchema, true); | |||||
| } | |||||
| } | |||||
| RULES.keywords[keyword] = RULES.all[keyword] = true; | |||||
| function _addRule(keyword, dataType, definition) { | |||||
| var ruleGroup; | |||||
| for (var i=0; i<RULES.length; i++) { | |||||
| var rg = RULES[i]; | |||||
| if (rg.type == dataType) { | |||||
| ruleGroup = rg; | |||||
| break; | |||||
| } | |||||
| } | |||||
| if (!ruleGroup) { | |||||
| ruleGroup = { type: dataType, rules: [] }; | |||||
| RULES.push(ruleGroup); | |||||
| } | |||||
| var rule = { | |||||
| keyword: keyword, | |||||
| definition: definition, | |||||
| custom: true, | |||||
| code: customRuleCode, | |||||
| implements: definition.implements | |||||
| }; | |||||
| ruleGroup.rules.push(rule); | |||||
| RULES.custom[keyword] = rule; | |||||
| } | |||||
| return this; | |||||
| } | |||||
| /** | |||||
| * Get keyword | |||||
| * @this Ajv | |||||
| * @param {String} keyword pre-defined or custom keyword. | |||||
| * @return {Object|Boolean} custom keyword definition, `true` if it is a predefined keyword, `false` otherwise. | |||||
| */ | |||||
| function getKeyword(keyword) { | |||||
| /* jshint validthis: true */ | |||||
| var rule = this.RULES.custom[keyword]; | |||||
| return rule ? rule.definition : this.RULES.keywords[keyword] || false; | |||||
| } | |||||
| /** | |||||
| * Remove keyword | |||||
| * @this Ajv | |||||
| * @param {String} keyword pre-defined or custom keyword. | |||||
| * @return {Ajv} this for method chaining | |||||
| */ | |||||
| function removeKeyword(keyword) { | |||||
| /* jshint validthis: true */ | |||||
| var RULES = this.RULES; | |||||
| delete RULES.keywords[keyword]; | |||||
| delete RULES.all[keyword]; | |||||
| delete RULES.custom[keyword]; | |||||
| for (var i=0; i<RULES.length; i++) { | |||||
| var rules = RULES[i].rules; | |||||
| for (var j=0; j<rules.length; j++) { | |||||
| if (rules[j].keyword == keyword) { | |||||
| rules.splice(j, 1); | |||||
| break; | |||||
| } | |||||
| } | |||||
| } | |||||
| return this; | |||||
| } | |||||
| /** | |||||
| * Validate keyword definition | |||||
| * @this Ajv | |||||
| * @param {Object} definition keyword definition object. | |||||
| * @param {Boolean} throwError true to throw exception if definition is invalid | |||||
| * @return {boolean} validation result | |||||
| */ | |||||
| function validateKeyword(definition, throwError) { | |||||
| validateKeyword.errors = null; | |||||
| var v = this._validateKeyword = this._validateKeyword | |||||
| || this.compile(definitionSchema, true); | |||||
| if (v(definition)) return true; | |||||
| validateKeyword.errors = v.errors; | |||||
| if (throwError) | |||||
| throw new Error('custom keyword definition is invalid: ' + this.errorsText(v.errors)); | |||||
| else | |||||
| return false; | |||||
| } | |||||
| @ -0,0 +1,17 @@ | |||||
| { | |||||
| "$schema": "http://json-schema.org/draft-07/schema#", | |||||
| "$id": "https://raw.githubusercontent.com/epoberezkin/ajv/master/lib/refs/data.json#", | |||||
| "description": "Meta-schema for $data reference (JSON Schema extension proposal)", | |||||
| "type": "object", | |||||
| "required": [ "$data" ], | |||||
| "properties": { | |||||
| "$data": { | |||||
| "type": "string", | |||||
| "anyOf": [ | |||||
| { "format": "relative-json-pointer" }, | |||||
| { "format": "json-pointer" } | |||||
| ] | |||||
| } | |||||
| }, | |||||
| "additionalProperties": false | |||||
| } | |||||
| @ -0,0 +1,149 @@ | |||||
| { | |||||
| "id": "http://json-schema.org/draft-04/schema#", | |||||
| "$schema": "http://json-schema.org/draft-04/schema#", | |||||
| "description": "Core schema meta-schema", | |||||
| "definitions": { | |||||
| "schemaArray": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "items": { "$ref": "#" } | |||||
| }, | |||||
| "positiveInteger": { | |||||
| "type": "integer", | |||||
| "minimum": 0 | |||||
| }, | |||||
| "positiveIntegerDefault0": { | |||||
| "allOf": [ { "$ref": "#/definitions/positiveInteger" }, { "default": 0 } ] | |||||
| }, | |||||
| "simpleTypes": { | |||||
| "enum": [ "array", "boolean", "integer", "null", "number", "object", "string" ] | |||||
| }, | |||||
| "stringArray": { | |||||
| "type": "array", | |||||
| "items": { "type": "string" }, | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| } | |||||
| }, | |||||
| "type": "object", | |||||
| "properties": { | |||||
| "id": { | |||||
| "type": "string" | |||||
| }, | |||||
| "$schema": { | |||||
| "type": "string" | |||||
| }, | |||||
| "title": { | |||||
| "type": "string" | |||||
| }, | |||||
| "description": { | |||||
| "type": "string" | |||||
| }, | |||||
| "default": {}, | |||||
| "multipleOf": { | |||||
| "type": "number", | |||||
| "minimum": 0, | |||||
| "exclusiveMinimum": true | |||||
| }, | |||||
| "maximum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMaximum": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "minimum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMinimum": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "maxLength": { "$ref": "#/definitions/positiveInteger" }, | |||||
| "minLength": { "$ref": "#/definitions/positiveIntegerDefault0" }, | |||||
| "pattern": { | |||||
| "type": "string", | |||||
| "format": "regex" | |||||
| }, | |||||
| "additionalItems": { | |||||
| "anyOf": [ | |||||
| { "type": "boolean" }, | |||||
| { "$ref": "#" } | |||||
| ], | |||||
| "default": {} | |||||
| }, | |||||
| "items": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/schemaArray" } | |||||
| ], | |||||
| "default": {} | |||||
| }, | |||||
| "maxItems": { "$ref": "#/definitions/positiveInteger" }, | |||||
| "minItems": { "$ref": "#/definitions/positiveIntegerDefault0" }, | |||||
| "uniqueItems": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "maxProperties": { "$ref": "#/definitions/positiveInteger" }, | |||||
| "minProperties": { "$ref": "#/definitions/positiveIntegerDefault0" }, | |||||
| "required": { "$ref": "#/definitions/stringArray" }, | |||||
| "additionalProperties": { | |||||
| "anyOf": [ | |||||
| { "type": "boolean" }, | |||||
| { "$ref": "#" } | |||||
| ], | |||||
| "default": {} | |||||
| }, | |||||
| "definitions": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "properties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "patternProperties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "dependencies": { | |||||
| "type": "object", | |||||
| "additionalProperties": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/stringArray" } | |||||
| ] | |||||
| } | |||||
| }, | |||||
| "enum": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| }, | |||||
| "type": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#/definitions/simpleTypes" }, | |||||
| { | |||||
| "type": "array", | |||||
| "items": { "$ref": "#/definitions/simpleTypes" }, | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| } | |||||
| ] | |||||
| }, | |||||
| "format": { "type": "string" }, | |||||
| "allOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "anyOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "oneOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "not": { "$ref": "#" } | |||||
| }, | |||||
| "dependencies": { | |||||
| "exclusiveMaximum": [ "maximum" ], | |||||
| "exclusiveMinimum": [ "minimum" ] | |||||
| }, | |||||
| "default": {} | |||||
| } | |||||
| @ -0,0 +1,154 @@ | |||||
| { | |||||
| "$schema": "http://json-schema.org/draft-06/schema#", | |||||
| "$id": "http://json-schema.org/draft-06/schema#", | |||||
| "title": "Core schema meta-schema", | |||||
| "definitions": { | |||||
| "schemaArray": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "items": { "$ref": "#" } | |||||
| }, | |||||
| "nonNegativeInteger": { | |||||
| "type": "integer", | |||||
| "minimum": 0 | |||||
| }, | |||||
| "nonNegativeIntegerDefault0": { | |||||
| "allOf": [ | |||||
| { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| { "default": 0 } | |||||
| ] | |||||
| }, | |||||
| "simpleTypes": { | |||||
| "enum": [ | |||||
| "array", | |||||
| "boolean", | |||||
| "integer", | |||||
| "null", | |||||
| "number", | |||||
| "object", | |||||
| "string" | |||||
| ] | |||||
| }, | |||||
| "stringArray": { | |||||
| "type": "array", | |||||
| "items": { "type": "string" }, | |||||
| "uniqueItems": true, | |||||
| "default": [] | |||||
| } | |||||
| }, | |||||
| "type": ["object", "boolean"], | |||||
| "properties": { | |||||
| "$id": { | |||||
| "type": "string", | |||||
| "format": "uri-reference" | |||||
| }, | |||||
| "$schema": { | |||||
| "type": "string", | |||||
| "format": "uri" | |||||
| }, | |||||
| "$ref": { | |||||
| "type": "string", | |||||
| "format": "uri-reference" | |||||
| }, | |||||
| "title": { | |||||
| "type": "string" | |||||
| }, | |||||
| "description": { | |||||
| "type": "string" | |||||
| }, | |||||
| "default": {}, | |||||
| "examples": { | |||||
| "type": "array", | |||||
| "items": {} | |||||
| }, | |||||
| "multipleOf": { | |||||
| "type": "number", | |||||
| "exclusiveMinimum": 0 | |||||
| }, | |||||
| "maximum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMaximum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "minimum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMinimum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "maxLength": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minLength": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "pattern": { | |||||
| "type": "string", | |||||
| "format": "regex" | |||||
| }, | |||||
| "additionalItems": { "$ref": "#" }, | |||||
| "items": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/schemaArray" } | |||||
| ], | |||||
| "default": {} | |||||
| }, | |||||
| "maxItems": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minItems": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "uniqueItems": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "contains": { "$ref": "#" }, | |||||
| "maxProperties": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minProperties": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "required": { "$ref": "#/definitions/stringArray" }, | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "definitions": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "properties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "patternProperties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "dependencies": { | |||||
| "type": "object", | |||||
| "additionalProperties": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/stringArray" } | |||||
| ] | |||||
| } | |||||
| }, | |||||
| "propertyNames": { "$ref": "#" }, | |||||
| "const": {}, | |||||
| "enum": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| }, | |||||
| "type": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#/definitions/simpleTypes" }, | |||||
| { | |||||
| "type": "array", | |||||
| "items": { "$ref": "#/definitions/simpleTypes" }, | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| } | |||||
| ] | |||||
| }, | |||||
| "format": { "type": "string" }, | |||||
| "allOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "anyOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "oneOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "not": { "$ref": "#" } | |||||
| }, | |||||
| "default": {} | |||||
| } | |||||
| @ -0,0 +1,168 @@ | |||||
| { | |||||
| "$schema": "http://json-schema.org/draft-07/schema#", | |||||
| "$id": "http://json-schema.org/draft-07/schema#", | |||||
| "title": "Core schema meta-schema", | |||||
| "definitions": { | |||||
| "schemaArray": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "items": { "$ref": "#" } | |||||
| }, | |||||
| "nonNegativeInteger": { | |||||
| "type": "integer", | |||||
| "minimum": 0 | |||||
| }, | |||||
| "nonNegativeIntegerDefault0": { | |||||
| "allOf": [ | |||||
| { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| { "default": 0 } | |||||
| ] | |||||
| }, | |||||
| "simpleTypes": { | |||||
| "enum": [ | |||||
| "array", | |||||
| "boolean", | |||||
| "integer", | |||||
| "null", | |||||
| "number", | |||||
| "object", | |||||
| "string" | |||||
| ] | |||||
| }, | |||||
| "stringArray": { | |||||
| "type": "array", | |||||
| "items": { "type": "string" }, | |||||
| "uniqueItems": true, | |||||
| "default": [] | |||||
| } | |||||
| }, | |||||
| "type": ["object", "boolean"], | |||||
| "properties": { | |||||
| "$id": { | |||||
| "type": "string", | |||||
| "format": "uri-reference" | |||||
| }, | |||||
| "$schema": { | |||||
| "type": "string", | |||||
| "format": "uri" | |||||
| }, | |||||
| "$ref": { | |||||
| "type": "string", | |||||
| "format": "uri-reference" | |||||
| }, | |||||
| "$comment": { | |||||
| "type": "string" | |||||
| }, | |||||
| "title": { | |||||
| "type": "string" | |||||
| }, | |||||
| "description": { | |||||
| "type": "string" | |||||
| }, | |||||
| "default": true, | |||||
| "readOnly": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "examples": { | |||||
| "type": "array", | |||||
| "items": true | |||||
| }, | |||||
| "multipleOf": { | |||||
| "type": "number", | |||||
| "exclusiveMinimum": 0 | |||||
| }, | |||||
| "maximum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMaximum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "minimum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "exclusiveMinimum": { | |||||
| "type": "number" | |||||
| }, | |||||
| "maxLength": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minLength": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "pattern": { | |||||
| "type": "string", | |||||
| "format": "regex" | |||||
| }, | |||||
| "additionalItems": { "$ref": "#" }, | |||||
| "items": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/schemaArray" } | |||||
| ], | |||||
| "default": true | |||||
| }, | |||||
| "maxItems": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minItems": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "uniqueItems": { | |||||
| "type": "boolean", | |||||
| "default": false | |||||
| }, | |||||
| "contains": { "$ref": "#" }, | |||||
| "maxProperties": { "$ref": "#/definitions/nonNegativeInteger" }, | |||||
| "minProperties": { "$ref": "#/definitions/nonNegativeIntegerDefault0" }, | |||||
| "required": { "$ref": "#/definitions/stringArray" }, | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "definitions": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "properties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "default": {} | |||||
| }, | |||||
| "patternProperties": { | |||||
| "type": "object", | |||||
| "additionalProperties": { "$ref": "#" }, | |||||
| "propertyNames": { "format": "regex" }, | |||||
| "default": {} | |||||
| }, | |||||
| "dependencies": { | |||||
| "type": "object", | |||||
| "additionalProperties": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#" }, | |||||
| { "$ref": "#/definitions/stringArray" } | |||||
| ] | |||||
| } | |||||
| }, | |||||
| "propertyNames": { "$ref": "#" }, | |||||
| "const": true, | |||||
| "enum": { | |||||
| "type": "array", | |||||
| "items": true, | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| }, | |||||
| "type": { | |||||
| "anyOf": [ | |||||
| { "$ref": "#/definitions/simpleTypes" }, | |||||
| { | |||||
| "type": "array", | |||||
| "items": { "$ref": "#/definitions/simpleTypes" }, | |||||
| "minItems": 1, | |||||
| "uniqueItems": true | |||||
| } | |||||
| ] | |||||
| }, | |||||
| "format": { "type": "string" }, | |||||
| "contentMediaType": { "type": "string" }, | |||||
| "contentEncoding": { "type": "string" }, | |||||
| "if": {"$ref": "#"}, | |||||
| "then": {"$ref": "#"}, | |||||
| "else": {"$ref": "#"}, | |||||
| "allOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "anyOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "oneOf": { "$ref": "#/definitions/schemaArray" }, | |||||
| "not": { "$ref": "#" } | |||||
| }, | |||||
| "default": true | |||||
| } | |||||
| @ -0,0 +1,94 @@ | |||||
| { | |||||
| "$schema": "http://json-schema.org/draft-07/schema#", | |||||
| "$id": "https://raw.githubusercontent.com/epoberezkin/ajv/master/lib/refs/json-schema-secure.json#", | |||||
| "title": "Meta-schema for the security assessment of JSON Schemas", | |||||
| "description": "If a JSON Schema fails validation against this meta-schema, it may be unsafe to validate untrusted data", | |||||
| "definitions": { | |||||
| "schemaArray": { | |||||
| "type": "array", | |||||
| "minItems": 1, | |||||
| "items": {"$ref": "#"} | |||||
| } | |||||
| }, | |||||
| "dependencies": { | |||||
| "patternProperties": { | |||||
| "description": "prevent slow validation of large property names", | |||||
| "required": ["propertyNames"], | |||||
| "properties": { | |||||
| "propertyNames": { | |||||
| "required": ["maxLength"] | |||||
| } | |||||
| } | |||||
| }, | |||||
| "uniqueItems": { | |||||
| "description": "prevent slow validation of large non-scalar arrays", | |||||
| "if": { | |||||
| "properties": { | |||||
| "uniqueItems": {"const": true}, | |||||
| "items": { | |||||
| "properties": { | |||||
| "type": { | |||||
| "anyOf": [ | |||||
| { | |||||
| "enum": ["object", "array"] | |||||
| }, | |||||
| { | |||||
| "type": "array", | |||||
| "contains": {"enum": ["object", "array"]} | |||||
| } | |||||
| ] | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| }, | |||||
| "then": { | |||||
| "required": ["maxItems"] | |||||
| } | |||||
| }, | |||||
| "pattern": { | |||||
| "description": "prevent slow pattern matching of large strings", | |||||
| "required": ["maxLength"] | |||||
| }, | |||||
| "format": { | |||||
| "description": "prevent slow format validation of large strings", | |||||
| "required": ["maxLength"] | |||||
| } | |||||
| }, | |||||
| "properties": { | |||||
| "additionalItems": {"$ref": "#"}, | |||||
| "additionalProperties": {"$ref": "#"}, | |||||
| "dependencies": { | |||||
| "additionalProperties": { | |||||
| "anyOf": [ | |||||
| {"type": "array"}, | |||||
| {"$ref": "#"} | |||||
| ] | |||||
| } | |||||
| }, | |||||
| "items": { | |||||
| "anyOf": [ | |||||
| {"$ref": "#"}, | |||||
| {"$ref": "#/definitions/schemaArray"} | |||||
| ] | |||||
| }, | |||||
| "definitions": { | |||||
| "additionalProperties": {"$ref": "#"} | |||||
| }, | |||||
| "patternProperties": { | |||||
| "additionalProperties": {"$ref": "#"} | |||||
| }, | |||||
| "properties": { | |||||
| "additionalProperties": {"$ref": "#"} | |||||
| }, | |||||
| "if": {"$ref": "#"}, | |||||
| "then": {"$ref": "#"}, | |||||
| "else": {"$ref": "#"}, | |||||
| "allOf": {"$ref": "#/definitions/schemaArray"}, | |||||
| "anyOf": {"$ref": "#/definitions/schemaArray"}, | |||||
| "oneOf": {"$ref": "#/definitions/schemaArray"}, | |||||
| "not": {"$ref": "#"}, | |||||
| "contains": {"$ref": "#"}, | |||||
| "propertyNames": {"$ref": "#"} | |||||
| } | |||||
| } | |||||
| @ -0,0 +1,125 @@ | |||||
| { | |||||
| "_from": "ajv@^6.5.5", | |||||
| "_id": "ajv@6.12.0", | |||||
| "_inBundle": false, | |||||
| "_integrity": "sha512-D6gFiFA0RRLyUbvijN74DWAjXSFxWKaWP7mldxkVhyhAV3+SWA9HEJPHQ2c9soIeTFJqcSdFDGFgdqs1iUU2Hw==", | |||||
| "_location": "/ajv", | |||||
| "_phantomChildren": {}, | |||||
| "_requested": { | |||||
| "type": "range", | |||||
| "registry": true, | |||||
| "raw": "ajv@^6.5.5", | |||||
| "name": "ajv", | |||||
| "escapedName": "ajv", | |||||
| "rawSpec": "^6.5.5", | |||||
| "saveSpec": null, | |||||
| "fetchSpec": "^6.5.5" | |||||
| }, | |||||
| "_requiredBy": [ | |||||
| "/har-validator" | |||||
| ], | |||||
| "_resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.0.tgz", | |||||
| "_shasum": "06d60b96d87b8454a5adaba86e7854da629db4b7", | |||||
| "_spec": "ajv@^6.5.5", | |||||
| "_where": "/home/g/Workspace/hatthieves/familyARK/familyark/app/node_modules/har-validator", | |||||
| "author": { | |||||
| "name": "Evgeny Poberezkin" | |||||
| }, | |||||
| "bugs": { | |||||
| "url": "https://github.com/epoberezkin/ajv/issues" | |||||
| }, | |||||
| "bundleDependencies": false, | |||||
| "dependencies": { | |||||
| "fast-deep-equal": "^3.1.1", | |||||
| "fast-json-stable-stringify": "^2.0.0", | |||||
| "json-schema-traverse": "^0.4.1", | |||||
| "uri-js": "^4.2.2" | |||||
| }, | |||||
| "deprecated": false, | |||||
| "description": "Another JSON Schema Validator", | |||||
| "devDependencies": { | |||||
| "ajv-async": "^1.0.0", | |||||
| "bluebird": "^3.5.3", | |||||
| "brfs": "^2.0.0", | |||||
| "browserify": "^16.2.0", | |||||
| "chai": "^4.0.1", | |||||
| "coveralls": "^3.0.1", | |||||
| "del-cli": "^3.0.0", | |||||
| "dot": "^1.0.3", | |||||
| "eslint": "^6.0.0", | |||||
| "gh-pages-generator": "^0.2.3", | |||||
| "glob": "^7.0.0", | |||||
| "if-node-version": "^1.0.0", | |||||
| "js-beautify": "^1.7.3", | |||||
| "jshint": "^2.10.2", | |||||
| "json-schema-test": "^2.0.0", | |||||
| "karma": "^4.0.1", | |||||
| "karma-chrome-launcher": "^3.0.0", | |||||
| "karma-mocha": "^1.1.1", | |||||
| "karma-sauce-launcher": "^2.0.0", | |||||
| "mocha": "^7.0.1", | |||||
| "nyc": "^15.0.0", | |||||
| "pre-commit": "^1.1.1", | |||||
| "require-globify": "^1.3.0", | |||||
| "typescript": "^2.8.3", | |||||
| "uglify-js": "^3.6.9", | |||||
| "watch": "^1.0.0" | |||||
| }, | |||||
| "files": [ | |||||
| "lib/", | |||||
| "dist/", | |||||
| "scripts/", | |||||
| "LICENSE", | |||||
| ".tonic_example.js" | |||||
| ], | |||||
| "homepage": "https://github.com/epoberezkin/ajv", | |||||
| "keywords": [ | |||||
| "JSON", | |||||
| "schema", | |||||
| "validator", | |||||
| "validation", | |||||
| "jsonschema", | |||||
| "json-schema", | |||||
| "json-schema-validator", | |||||
| "json-schema-validation" | |||||
| ], | |||||
| "license": "MIT", | |||||
| "main": "lib/ajv.js", | |||||
| "name": "ajv", | |||||
| "nyc": { | |||||
| "exclude": [ | |||||
| "**/spec/**", | |||||
| "node_modules" | |||||
| ], | |||||
| "reporter": [ | |||||
| "lcov", | |||||
| "text-summary" | |||||
| ] | |||||
| }, | |||||
| "repository": { | |||||
| "type": "git", | |||||
| "url": "git+https://github.com/epoberezkin/ajv.git" | |||||
| }, | |||||
| "scripts": { | |||||
| "build": "del-cli lib/dotjs/*.js \"!lib/dotjs/index.js\" && node scripts/compile-dots.js", | |||||
| "bundle": "del-cli dist && node ./scripts/bundle.js . Ajv pure_getters", | |||||
| "bundle-beautify": "node ./scripts/bundle.js js-beautify", | |||||
| "eslint": "eslint lib/{compile/,}*.js spec/{**/,}*.js scripts --ignore-pattern spec/JSON-Schema-Test-Suite", | |||||
| "jshint": "jshint lib/{compile/,}*.js", | |||||
| "lint": "npm run jshint && npm run eslint", | |||||
| "prepublish": "npm run build && npm run bundle", | |||||
| "test": "npm run lint && npm run build && npm run test-all", | |||||
| "test-all": "npm run test-cov && if-node-version 10 npm run test-browser", | |||||
| "test-browser": "del-cli .browser && npm run bundle && scripts/prepare-tests && npm run test-karma", | |||||
| "test-cov": "nyc npm run test-spec", | |||||
| "test-debug": "npm run test-spec -- --inspect-brk", | |||||
| "test-fast": "AJV_FAST_TEST=true npm run test-spec", | |||||
| "test-karma": "karma start", | |||||
| "test-spec": "mocha spec/{**/,}*.spec.js -R spec", | |||||
| "test-ts": "tsc --target ES5 --noImplicitAny --noEmit spec/typescript/index.ts", | |||||
| "watch": "watch \"npm run build\" ./lib/dot" | |||||
| }, | |||||
| "tonicExampleFilename": ".tonic_example.js", | |||||
| "typings": "lib/ajv.d.ts", | |||||
| "version": "6.12.0" | |||||
| } | |||||
| @ -0,0 +1,3 @@ | |||||
| rules: | |||||
| no-console: 0 | |||||
| no-empty: [2, allowEmptyCatch: true] | |||||
| @ -0,0 +1,61 @@ | |||||
| 'use strict'; | |||||
| var fs = require('fs') | |||||
| , path = require('path') | |||||
| , browserify = require('browserify') | |||||
| , uglify = require('uglify-js'); | |||||
| var pkg = process.argv[2] | |||||
| , standalone = process.argv[3] | |||||
| , compress = process.argv[4]; | |||||
| var packageDir = path.join(__dirname, '..'); | |||||
| if (pkg != '.') packageDir = path.join(packageDir, 'node_modules', pkg); | |||||
| var json = require(path.join(packageDir, 'package.json')); | |||||
| var distDir = path.join(__dirname, '..', 'dist'); | |||||
| if (!fs.existsSync(distDir)) fs.mkdirSync(distDir); | |||||
| var bOpts = {}; | |||||
| if (standalone) bOpts.standalone = standalone; | |||||
| browserify(bOpts) | |||||
| .require(path.join(packageDir, json.main), {expose: json.name}) | |||||
| .bundle(function (err, buf) { | |||||
| if (err) { | |||||
| console.error('browserify error:', err); | |||||
| process.exit(1); | |||||
| } | |||||
| var outputFile = path.join(distDir, json.name); | |||||
| var uglifyOpts = { | |||||
| warnings: true, | |||||
| compress: {}, | |||||
| output: { | |||||
| preamble: '/* ' + json.name + ' ' + json.version + ': ' + json.description + ' */' | |||||
| } | |||||
| }; | |||||
| if (compress) { | |||||
| var compressOpts = compress.split(','); | |||||
| for (var i=0, il = compressOpts.length; i<il; ++i) { | |||||
| var pair = compressOpts[i].split('='); | |||||
| uglifyOpts.compress[pair[0]] = pair.length < 1 || pair[1] != 'false'; | |||||
| } | |||||
| } | |||||
| if (standalone) { | |||||
| uglifyOpts.sourceMap = { | |||||
| filename: json.name + '.min.js', | |||||
| url: json.name + '.min.js.map' | |||||
| }; | |||||
| } | |||||
| var result = uglify.minify(buf.toString(), uglifyOpts); | |||||
| fs.writeFileSync(outputFile + '.min.js', result.code); | |||||
| if (result.map) fs.writeFileSync(outputFile + '.min.js.map', result.map); | |||||
| if (standalone) fs.writeFileSync(outputFile + '.bundle.js', buf); | |||||
| if (result.warnings) { | |||||
| for (var j=0, jl = result.warnings.length; j<jl; ++j) | |||||
| console.warn('UglifyJS warning:', result.warnings[j]); | |||||
| } | |||||
| }); | |||||
| @ -0,0 +1,73 @@ | |||||
| //compile doT templates to js functions | |||||
| 'use strict'; | |||||
| var glob = require('glob') | |||||
| , fs = require('fs') | |||||
| , path = require('path') | |||||
| , doT = require('dot') | |||||
| , beautify = require('js-beautify').js_beautify; | |||||
| var defsRootPath = process.argv[2] || path.join(__dirname, '../lib'); | |||||
| var defs = {}; | |||||
| var defFiles = glob.sync('./dot/**/*.def', { cwd: defsRootPath }); | |||||
| defFiles.forEach(function (f) { | |||||
| var name = path.basename(f, '.def'); | |||||
| defs[name] = fs.readFileSync(path.join(defsRootPath, f)); | |||||
| }); | |||||
| var filesRootPath = process.argv[3] || path.join(__dirname, '../lib'); | |||||
| var files = glob.sync('./dot/**/*.jst', { cwd: filesRootPath }); | |||||
| var dotjsPath = path.join(filesRootPath, './dotjs'); | |||||
| try { fs.mkdirSync(dotjsPath); } catch(e) {} | |||||
| console.log('\n\nCompiling:'); | |||||
| var FUNCTION_NAME = /function\s+anonymous\s*\(it[^)]*\)\s*{/; | |||||
| var OUT_EMPTY_STRING = /out\s*\+=\s*'\s*';/g; | |||||
| var ISTANBUL = /'(istanbul[^']+)';/g; | |||||
| var ERROR_KEYWORD = /\$errorKeyword/g; | |||||
| var ERROR_KEYWORD_OR = /\$errorKeyword\s+\|\|/g; | |||||
| var VARS = [ | |||||
| '$errs', '$valid', '$lvl', '$data', '$dataLvl', | |||||
| '$errorKeyword', '$closingBraces', '$schemaPath', | |||||
| '$validate' | |||||
| ]; | |||||
| files.forEach(function (f) { | |||||
| var keyword = path.basename(f, '.jst'); | |||||
| var targetPath = path.join(dotjsPath, keyword + '.js'); | |||||
| var template = fs.readFileSync(path.join(filesRootPath, f)); | |||||
| var code = doT.compile(template, defs); | |||||
| code = code.toString() | |||||
| .replace(OUT_EMPTY_STRING, '') | |||||
| .replace(FUNCTION_NAME, 'function generate_' + keyword + '(it, $keyword, $ruleType) {') | |||||
| .replace(ISTANBUL, '/* $1 */'); | |||||
| removeAlwaysFalsyInOr(); | |||||
| VARS.forEach(removeUnusedVar); | |||||
| code = "'use strict';\nmodule.exports = " + code; | |||||
| code = beautify(code, { indent_size: 2 }) + '\n'; | |||||
| fs.writeFileSync(targetPath, code); | |||||
| console.log('compiled', keyword); | |||||
| function removeUnusedVar(v) { | |||||
| v = v.replace(/\$/g, '\\$$'); | |||||
| var regexp = new RegExp(v + '[^A-Za-z0-9_$]', 'g'); | |||||
| var count = occurrences(regexp); | |||||
| if (count == 1) { | |||||
| regexp = new RegExp('var\\s+' + v + '\\s*=[^;]+;|var\\s+' + v + ';'); | |||||
| code = code.replace(regexp, ''); | |||||
| } | |||||
| } | |||||
| function removeAlwaysFalsyInOr() { | |||||
| var countUsed = occurrences(ERROR_KEYWORD); | |||||
| var countOr = occurrences(ERROR_KEYWORD_OR); | |||||
| if (countUsed == countOr + 1) code = code.replace(ERROR_KEYWORD_OR, ''); | |||||
| } | |||||
| function occurrences(regexp) { | |||||
| return (code.match(regexp) || []).length; | |||||
| } | |||||
| }); | |||||
| @ -0,0 +1,10 @@ | |||||
| #!/usr/bin/env node | |||||
| 'use strict'; | |||||
| var fs = require('fs'); | |||||
| var name = process.argv[2] || '.'; | |||||
| var property = process.argv[3] || 'version'; | |||||
| if (name != '.') name = 'node_modules/' + name; | |||||
| var json = JSON.parse(fs.readFileSync(name + '/package.json', 'utf8')); | |||||
| console.log(json[property]); | |||||
| @ -0,0 +1,12 @@ | |||||
| #!/usr/bin/env sh | |||||
| set -e | |||||
| mkdir -p .browser | |||||
| echo | |||||
| echo Preparing browser tests: | |||||
| find spec -type f -name '*.spec.js' | \ | |||||
| xargs -I {} sh -c \ | |||||
| 'export f="{}"; echo $f; browserify $f -t require-globify -t brfs -x ajv -u buffer -o $(echo $f | sed -e "s/spec/.browser/");' | |||||
| @ -0,0 +1,32 @@ | |||||
| #!/usr/bin/env bash | |||||
| set -e | |||||
| if [[ -n $TRAVIS_TAG && $TRAVIS_JOB_NUMBER =~ ".3" ]]; then | |||||
| echo "About to publish $TRAVIS_TAG to ajv-dist..." | |||||
| git config user.email "$GIT_USER_EMAIL" | |||||
| git config user.name "$GIT_USER_NAME" | |||||
| git clone https://${GITHUB_TOKEN}@github.com/epoberezkin/ajv-dist.git ../ajv-dist | |||||
| rm -rf ../ajv-dist/dist | |||||
| mkdir ../ajv-dist/dist | |||||
| cp ./dist/ajv.* ../ajv-dist/dist | |||||
| cat bower.json | sed 's/"name": "ajv"/"name": "ajv-dist"/' > ../ajv-dist/bower.json | |||||
| cd ../ajv-dist | |||||
| if [[ `git status --porcelain` ]]; then | |||||
| echo "Changes detected. Updating master branch..." | |||||
| git add -A | |||||
| git commit -m "updated by travis build #$TRAVIS_BUILD_NUMBER" | |||||
| git push --quiet origin master > /dev/null 2>&1 | |||||
| fi | |||||
| echo "Publishing tag..." | |||||
| git tag $TRAVIS_TAG | |||||
| git push --tags > /dev/null 2>&1 | |||||
| echo "Done" | |||||
| fi | |||||
| @ -0,0 +1,23 @@ | |||||
| #!/usr/bin/env bash | |||||
| set -e | |||||
| if [[ "$TRAVIS_BRANCH" == "master" && "$TRAVIS_PULL_REQUEST" == "false" && $TRAVIS_JOB_NUMBER =~ ".3" ]]; then | |||||
| git diff --name-only $TRAVIS_COMMIT_RANGE | grep -qE '\.md$|^LICENSE$|travis-gh-pages$' && { | |||||
| rm -rf ../gh-pages | |||||
| git clone -b gh-pages --single-branch https://${GITHUB_TOKEN}@github.com/epoberezkin/ajv.git ../gh-pages | |||||
| mkdir -p ../gh-pages/_source | |||||
| cp *.md ../gh-pages/_source | |||||
| cp LICENSE ../gh-pages/_source | |||||
| currentDir=$(pwd) | |||||
| cd ../gh-pages | |||||
| $currentDir/node_modules/.bin/gh-pages-generator | |||||
| # remove logo from README | |||||
| sed -i -E "s/<img[^>]+ajv_logo[^>]+>//" index.md | |||||
| git config user.email "$GIT_USER_EMAIL" | |||||
| git config user.name "$GIT_USER_NAME" | |||||
| git add . | |||||
| git commit -am "updated by travis build #$TRAVIS_BUILD_NUMBER" | |||||
| git push --quiet origin gh-pages > /dev/null 2>&1 | |||||
| } | |||||
| fi | |||||
| @ -0,0 +1,19 @@ | |||||
| Copyright (c) 2011 Mark Cavage, All rights reserved. | |||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | |||||
| of this software and associated documentation files (the "Software"), to deal | |||||
| in the Software without restriction, including without limitation the rights | |||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |||||
| copies of the Software, and to permit persons to whom the Software is | |||||
| furnished to do so, subject to the following conditions: | |||||
| The above copyright notice and this permission notice shall be included in | |||||
| all copies or substantial portions of the Software. | |||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |||||
| THE SOFTWARE | |||||
| @ -0,0 +1,50 @@ | |||||
| node-asn1 is a library for encoding and decoding ASN.1 datatypes in pure JS. | |||||
| Currently BER encoding is supported; at some point I'll likely have to do DER. | |||||
| ## Usage | |||||
| Mostly, if you're *actually* needing to read and write ASN.1, you probably don't | |||||
| need this readme to explain what and why. If you have no idea what ASN.1 is, | |||||
| see this: ftp://ftp.rsa.com/pub/pkcs/ascii/layman.asc | |||||
| The source is pretty much self-explanatory, and has read/write methods for the | |||||
| common types out there. | |||||
| ### Decoding | |||||
| The following reads an ASN.1 sequence with a boolean. | |||||
| var Ber = require('asn1').Ber; | |||||
| var reader = new Ber.Reader(Buffer.from([0x30, 0x03, 0x01, 0x01, 0xff])); | |||||
| reader.readSequence(); | |||||
| console.log('Sequence len: ' + reader.length); | |||||
| if (reader.peek() === Ber.Boolean) | |||||
| console.log(reader.readBoolean()); | |||||
| ### Encoding | |||||
| The following generates the same payload as above. | |||||
| var Ber = require('asn1').Ber; | |||||
| var writer = new Ber.Writer(); | |||||
| writer.startSequence(); | |||||
| writer.writeBoolean(true); | |||||
| writer.endSequence(); | |||||
| console.log(writer.buffer); | |||||
| ## Installation | |||||
| npm install asn1 | |||||
| ## License | |||||
| MIT. | |||||
| ## Bugs | |||||
| See <https://github.com/joyent/node-asn1/issues>. | |||||
| @ -0,0 +1,13 @@ | |||||
| // Copyright 2011 Mark Cavage <mcavage@gmail.com> All rights reserved. | |||||
| module.exports = { | |||||
| newInvalidAsn1Error: function (msg) { | |||||
| var e = new Error(); | |||||
| e.name = 'InvalidAsn1Error'; | |||||
| e.message = msg || ''; | |||||
| return e; | |||||
| } | |||||
| }; | |||||
| @ -0,0 +1,27 @@ | |||||
| // Copyright 2011 Mark Cavage <mcavage@gmail.com> All rights reserved. | |||||
| var errors = require('./errors'); | |||||
| var types = require('./types'); | |||||
| var Reader = require('./reader'); | |||||
| var Writer = require('./writer'); | |||||
| // --- Exports | |||||
| module.exports = { | |||||
| Reader: Reader, | |||||
| Writer: Writer | |||||
| }; | |||||
| for (var t in types) { | |||||
| if (types.hasOwnProperty(t)) | |||||
| module.exports[t] = types[t]; | |||||
| } | |||||
| for (var e in errors) { | |||||
| if (errors.hasOwnProperty(e)) | |||||
| module.exports[e] = errors[e]; | |||||
| } | |||||