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