| @ -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]; | |||
| } | |||