You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					1373 lines
				
				36 KiB
			
		
		
			
		
	
	
					1373 lines
				
				36 KiB
			| 
											3 years ago
										 | 'use strict' | ||
|  | 
 | ||
|  | /* eslint no-prototype-builtins: 0 */ | ||
|  | 
 | ||
|  | const Ajv = require('ajv') | ||
|  | const merge = require('deepmerge') | ||
|  | const clone = require('rfdc')({ proto: true }) | ||
|  | const fjsCloned = Symbol('fast-json-stringify.cloned') | ||
|  | 
 | ||
|  | const validate = require('./schema-validator') | ||
|  | let stringSimilarity = null | ||
|  | 
 | ||
|  | let isLong | ||
|  | try { | ||
|  |   isLong = require('long').isLong | ||
|  | } catch (e) { | ||
|  |   isLong = null | ||
|  | } | ||
|  | 
 | ||
|  | const addComma = `
 | ||
|  |   if (addComma) { | ||
|  |     json += ',' | ||
|  |   } else { | ||
|  |     addComma = true | ||
|  |   } | ||
|  | `
 | ||
|  | 
 | ||
|  | function isValidSchema (schema, name) { | ||
|  |   if (!validate(schema)) { | ||
|  |     if (name) { | ||
|  |       name = `"${name}" ` | ||
|  |     } else { | ||
|  |       name = '' | ||
|  |     } | ||
|  |     const first = validate.errors[0] | ||
|  |     const err = new Error(`${name}schema is invalid: data${first.dataPath} ${first.message}`) | ||
|  |     err.errors = isValidSchema.errors | ||
|  |     throw err | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function mergeLocation (source, dest) { | ||
|  |   return { | ||
|  |     schema: dest.schema || source.schema, | ||
|  |     root: dest.root || source.root, | ||
|  |     externalSchema: dest.externalSchema || source.externalSchema | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | const arrayItemsReferenceSerializersMap = new Map() | ||
|  | const objectReferenceSerializersMap = new Map() | ||
|  | 
 | ||
|  | function build (schema, options) { | ||
|  |   arrayItemsReferenceSerializersMap.clear() | ||
|  |   objectReferenceSerializersMap.clear() | ||
|  |   options = options || {} | ||
|  |   isValidSchema(schema) | ||
|  |   if (options.schema) { | ||
|  |     // eslint-disable-next-line
 | ||
|  |     for (var key of Object.keys(options.schema)) { | ||
|  |       isValidSchema(options.schema[key], key) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   let intParseFunctionName = 'trunc' | ||
|  |   if (options.rounding) { | ||
|  |     if (['floor', 'ceil', 'round'].includes(options.rounding)) { | ||
|  |       intParseFunctionName = options.rounding | ||
|  |     } else { | ||
|  |       throw new Error(`Unsupported integer rounding method ${options.rounding}`) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   /* eslint no-new-func: "off" */ | ||
|  |   let code = `
 | ||
|  |     'use strict' | ||
|  |   `
 | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     ${asFunctions} | ||
|  | 
 | ||
|  |     var isLong = ${isLong ? isLong.toString() : false} | ||
|  | 
 | ||
|  |     function parseInteger(int) { return Math.${intParseFunctionName}(int) } | ||
|  |     `
 | ||
|  | 
 | ||
|  |   let location = { | ||
|  |     schema, | ||
|  |     root: schema, | ||
|  |     externalSchema: options.schema | ||
|  |   } | ||
|  | 
 | ||
|  |   if (schema.$ref) { | ||
|  |     location = refFinder(schema.$ref, location) | ||
|  |     schema = location.schema | ||
|  |   } | ||
|  | 
 | ||
|  |   if (schema.type === undefined) { | ||
|  |     schema.type = inferTypeByKeyword(schema) | ||
|  |   } | ||
|  | 
 | ||
|  |   let main | ||
|  | 
 | ||
|  |   switch (schema.type) { | ||
|  |     case 'object': | ||
|  |       main = '$main' | ||
|  |       code = buildObject(location, code, main) | ||
|  |       break | ||
|  |     case 'string': | ||
|  |       main = schema.nullable ? '$asStringNullable' : getStringSerializer(schema.format) | ||
|  |       break | ||
|  |     case 'integer': | ||
|  |       main = schema.nullable ? '$asIntegerNullable' : '$asInteger' | ||
|  |       break | ||
|  |     case 'number': | ||
|  |       main = schema.nullable ? '$asNumberNullable' : '$asNumber' | ||
|  |       break | ||
|  |     case 'boolean': | ||
|  |       main = schema.nullable ? '$asBooleanNullable' : '$asBoolean' | ||
|  |       break | ||
|  |     case 'null': | ||
|  |       main = '$asNull' | ||
|  |       break | ||
|  |     case 'array': | ||
|  |       main = '$main' | ||
|  |       code = buildArray(location, code, main) | ||
|  |       schema = location.schema | ||
|  |       break | ||
|  |     case undefined: | ||
|  |       main = '$asAny' | ||
|  |       break | ||
|  |     default: | ||
|  |       throw new Error(`${schema.type} unsupported`) | ||
|  |   } | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     ; | ||
|  |      return ${main} | ||
|  |   `
 | ||
|  | 
 | ||
|  |   const dependencies = [new Ajv(options.ajv)] | ||
|  |   const dependenciesName = ['ajv'] | ||
|  |   dependenciesName.push(code) | ||
|  | 
 | ||
|  |   if (options.debugMode) { | ||
|  |     dependenciesName.toString = function () { | ||
|  |       return dependenciesName.join('\n') | ||
|  |     } | ||
|  |     return dependenciesName | ||
|  |   } | ||
|  | 
 | ||
|  |   arrayItemsReferenceSerializersMap.clear() | ||
|  |   objectReferenceSerializersMap.clear() | ||
|  | 
 | ||
|  |   return (Function.apply(null, dependenciesName).apply(null, dependencies)) | ||
|  | } | ||
|  | 
 | ||
|  | const objectKeywords = [ | ||
|  |   'maxProperties', | ||
|  |   'minProperties', | ||
|  |   'required', | ||
|  |   'properties', | ||
|  |   'patternProperties', | ||
|  |   'additionalProperties', | ||
|  |   'dependencies' | ||
|  | ] | ||
|  | 
 | ||
|  | const arrayKeywords = [ | ||
|  |   'items', | ||
|  |   'additionalItems', | ||
|  |   'maxItems', | ||
|  |   'minItems', | ||
|  |   'uniqueItems', | ||
|  |   'contains' | ||
|  | ] | ||
|  | 
 | ||
|  | const stringKeywords = [ | ||
|  |   'maxLength', | ||
|  |   'minLength', | ||
|  |   'pattern' | ||
|  | ] | ||
|  | 
 | ||
|  | const numberKeywords = [ | ||
|  |   'multipleOf', | ||
|  |   'maximum', | ||
|  |   'exclusiveMaximum', | ||
|  |   'minimum', | ||
|  |   'exclusiveMinimum' | ||
|  | ] | ||
|  | 
 | ||
|  | /** | ||
|  |  * Infer type based on keyword in order to generate optimized code | ||
|  |  * https://json-schema.org/latest/json-schema-validation.html#rfc.section.6
 | ||
|  |  */ | ||
|  | function inferTypeByKeyword (schema) { | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var keyword of objectKeywords) { | ||
|  |     if (keyword in schema) return 'object' | ||
|  |   } | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var keyword of arrayKeywords) { | ||
|  |     if (keyword in schema) return 'array' | ||
|  |   } | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var keyword of stringKeywords) { | ||
|  |     if (keyword in schema) return 'string' | ||
|  |   } | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var keyword of numberKeywords) { | ||
|  |     if (keyword in schema) return 'number' | ||
|  |   } | ||
|  |   return schema.type | ||
|  | } | ||
|  | 
 | ||
|  | const stringSerializerMap = { | ||
|  |   'date-time': '$asDatetime', | ||
|  |   date: '$asDate', | ||
|  |   time: '$asTime' | ||
|  | } | ||
|  | 
 | ||
|  | function getStringSerializer (format) { | ||
|  |   return stringSerializerMap[format] || | ||
|  |   '$asString' | ||
|  | } | ||
|  | 
 | ||
|  | function getTestSerializer (format) { | ||
|  |   return stringSerializerMap[format] | ||
|  | } | ||
|  | 
 | ||
|  | const asFunctions = `
 | ||
|  | function $pad2Zeros (num) { | ||
|  |   const s = '00' + num | ||
|  |   return s[s.length - 2] + s[s.length - 1] | ||
|  | } | ||
|  | 
 | ||
|  | function $asAny (i) { | ||
|  |   return JSON.stringify(i) | ||
|  | } | ||
|  | 
 | ||
|  | function $asNull () { | ||
|  |   return 'null' | ||
|  | } | ||
|  | 
 | ||
|  | function $asInteger (i) { | ||
|  |   if (isLong && isLong(i)) { | ||
|  |     return i.toString() | ||
|  |   } else if (typeof i === 'bigint') { | ||
|  |     return i.toString() | ||
|  |   } else if (Number.isInteger(i)) { | ||
|  |     return $asNumber(i) | ||
|  |   } else { | ||
|  |     /* eslint no-undef: "off" */ | ||
|  |     return $asNumber(parseInteger(i)) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asIntegerNullable (i) { | ||
|  |   return i === null ? null : $asInteger(i) | ||
|  | } | ||
|  | 
 | ||
|  | function $asNumber (i) { | ||
|  |   const num = Number(i) | ||
|  |   if (isNaN(num)) { | ||
|  |     return 'null' | ||
|  |   } else { | ||
|  |     return '' + num | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asNumberNullable (i) { | ||
|  |   return i === null ? null : $asNumber(i) | ||
|  | } | ||
|  | 
 | ||
|  | function $asBoolean (bool) { | ||
|  |   return bool && 'true' || 'false' // eslint-disable-line
 | ||
|  | } | ||
|  | 
 | ||
|  | function $asBooleanNullable (bool) { | ||
|  |   return bool === null ? null : $asBoolean(bool) | ||
|  | } | ||
|  | 
 | ||
|  | function $asDatetime (date, skipQuotes) { | ||
|  |   const quotes = skipQuotes === true ? '' : '"' | ||
|  |   if (date instanceof Date) { | ||
|  |     return quotes + date.toISOString() + quotes | ||
|  |   } else if (date && typeof date.toISOString === 'function') { | ||
|  |     return quotes + date.toISOString() + quotes | ||
|  |   } else { | ||
|  |     return $asString(date, skipQuotes) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asDate (date, skipQuotes) { | ||
|  |   const quotes = skipQuotes === true ? '' : '"' | ||
|  |   if (date instanceof Date) { | ||
|  |     return quotes + new Date(date.getTime() - (date.getTimezoneOffset() * 60000 )).toISOString().slice(0, 10) + quotes | ||
|  |   } else if (date && typeof date.format === 'function') { | ||
|  |     return quotes + date.format('YYYY-MM-DD') + quotes | ||
|  |   } else { | ||
|  |     return $asString(date, skipQuotes) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asTime (date, skipQuotes) { | ||
|  |   const quotes = skipQuotes === true ? '' : '"' | ||
|  |   if (date instanceof Date) { | ||
|  |     const hour = new Intl.DateTimeFormat('en', { hour: 'numeric', hour12: false }).format(date) | ||
|  |     const minute = new Intl.DateTimeFormat('en', { minute: 'numeric' }).format(date) | ||
|  |     const second = new Intl.DateTimeFormat('en', { second: 'numeric' }).format(date) | ||
|  |     return quotes + $pad2Zeros(hour) + ':' + $pad2Zeros(minute) + ':' + $pad2Zeros(second) + quotes | ||
|  |   } else if (date && typeof date.format === 'function') { | ||
|  |     return quotes + date.format('HH:mm:ss') + quotes | ||
|  |   } else { | ||
|  |     return $asString(date, skipQuotes) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asString (str, skipQuotes) { | ||
|  |   const quotes = skipQuotes === true ? '' : '"' | ||
|  |   if (str instanceof Date) { | ||
|  |     return quotes + str.toISOString() + quotes | ||
|  |   } else if (str === null) { | ||
|  |     return quotes + quotes | ||
|  |   } else if (str instanceof RegExp) { | ||
|  |     str = str.source | ||
|  |   } else if (typeof str !== 'string') { | ||
|  |     str = str.toString() | ||
|  |   } | ||
|  |   // If we skipQuotes it means that we are using it as test
 | ||
|  |   // no need to test the string length for the render
 | ||
|  |   if (skipQuotes) { | ||
|  |     return str | ||
|  |   } | ||
|  | 
 | ||
|  |   if (str.length < 42) { | ||
|  |     return $asStringSmall(str) | ||
|  |   } else { | ||
|  |     return JSON.stringify(str) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function $asStringNullable (str) { | ||
|  |   return str === null ? null : $asString(str) | ||
|  | } | ||
|  | 
 | ||
|  | // magically escape strings for json
 | ||
|  | // relying on their charCodeAt
 | ||
|  | // everything below 32 needs JSON.stringify()
 | ||
|  | // every string that contain surrogate needs JSON.stringify()
 | ||
|  | // 34 and 92 happens all the time, so we
 | ||
|  | // have a fast case for them
 | ||
|  | function $asStringSmall (str) { | ||
|  |   const l = str.length | ||
|  |   let result = '' | ||
|  |   let last = 0 | ||
|  |   let found = false | ||
|  |   let surrogateFound = false | ||
|  |   let point = 255 | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var i = 0; i < l && point >= 32; i++) { | ||
|  |     point = str.charCodeAt(i) | ||
|  |     if (point >= 0xD800 && point <= 0xDFFF) { | ||
|  |       // The current character is a surrogate.
 | ||
|  |       surrogateFound = true | ||
|  |     } | ||
|  |     if (point === 34 || point === 92) { | ||
|  |       result += str.slice(last, i) + '\\\\' | ||
|  |       last = i | ||
|  |       found = true | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!found) { | ||
|  |     result = str | ||
|  |   } else { | ||
|  |     result += str.slice(last) | ||
|  |   } | ||
|  |   return ((point < 32) || (surrogateFound === true)) ? JSON.stringify(str) : '"' + result + '"' | ||
|  | } | ||
|  | `
 | ||
|  | 
 | ||
|  | function addPatternProperties (location) { | ||
|  |   const schema = location.schema | ||
|  |   const pp = schema.patternProperties | ||
|  |   let code = `
 | ||
|  |       var properties = ${JSON.stringify(schema.properties)} || {} | ||
|  |       var keys = Object.keys(obj) | ||
|  |       for (var i = 0; i < keys.length; i++) { | ||
|  |         if (properties[keys[i]]) continue | ||
|  |   `
 | ||
|  |   Object.keys(pp).forEach((regex, index) => { | ||
|  |     let ppLocation = mergeLocation(location, { schema: pp[regex] }) | ||
|  |     if (pp[regex].$ref) { | ||
|  |       ppLocation = refFinder(pp[regex].$ref, location) | ||
|  |       pp[regex] = ppLocation.schema | ||
|  |     } | ||
|  |     const type = pp[regex].type | ||
|  |     const format = pp[regex].format | ||
|  |     const stringSerializer = getStringSerializer(format) | ||
|  |     try { | ||
|  |       RegExp(regex) | ||
|  |     } catch (err) { | ||
|  |       throw new Error(`${err.message}. Found at ${regex} matching ${JSON.stringify(pp[regex])}`) | ||
|  |     } | ||
|  | 
 | ||
|  |     const ifPpKeyExists = `if (/${regex.replace(/\\*\//g, '\\/')}/.test(keys[i])) {` | ||
|  | 
 | ||
|  |     if (type === 'object') { | ||
|  |       code += `${buildObject(ppLocation, '', 'buildObjectPP' + index)}
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + buildObjectPP${index}(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === 'array') { | ||
|  |       code += `${buildArray(ppLocation, '', 'buildArrayPP' + index)}
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + buildArrayPP${index}(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === 'null') { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) +':null' | ||
|  |       `
 | ||
|  |     } else if (type === 'string') { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + ${stringSerializer}(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === 'integer') { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + $asInteger(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === 'number') { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + $asNumber(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === 'boolean') { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + $asBoolean(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else if (type === undefined) { | ||
|  |       code += `
 | ||
|  |           ${ifPpKeyExists} | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + $asAny(obj[keys[i]]) | ||
|  |       `
 | ||
|  |     } else { | ||
|  |       code += `
 | ||
|  |         ${ifPpKeyExists} | ||
|  |         throw new Error('Cannot coerce ' + obj[keys[i]] + ' to ' + ${JSON.stringify(type)}) | ||
|  |       `
 | ||
|  |     } | ||
|  | 
 | ||
|  |     code += `
 | ||
|  |           continue | ||
|  |         } | ||
|  |     `
 | ||
|  |   }) | ||
|  |   if (schema.additionalProperties) { | ||
|  |     code += additionalProperty(location) | ||
|  |   } | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |       } | ||
|  |   `
 | ||
|  |   return code | ||
|  | } | ||
|  | 
 | ||
|  | function additionalProperty (location) { | ||
|  |   let ap = location.schema.additionalProperties | ||
|  |   let code = '' | ||
|  |   if (ap === true) { | ||
|  |     return `
 | ||
|  |         if (obj[keys[i]] !== undefined && typeof obj[keys[i]] !== 'function' && typeof obj[keys[i]] !== 'symbol') { | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + JSON.stringify(obj[keys[i]]) | ||
|  |         } | ||
|  |     `
 | ||
|  |   } | ||
|  |   let apLocation = mergeLocation(location, { schema: ap }) | ||
|  |   if (ap.$ref) { | ||
|  |     apLocation = refFinder(ap.$ref, location) | ||
|  |     ap = apLocation.schema | ||
|  |   } | ||
|  | 
 | ||
|  |   const type = ap.type | ||
|  |   const format = ap.format | ||
|  |   const stringSerializer = getStringSerializer(format) | ||
|  |   if (type === 'object') { | ||
|  |     code += `${buildObject(apLocation, '', 'buildObjectAP')}
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) + ':' + buildObjectAP(obj[keys[i]]) | ||
|  |     `
 | ||
|  |   } else if (type === 'array') { | ||
|  |     code += `${buildArray(apLocation, '', 'buildArrayAP')}
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) + ':' + buildArrayAP(obj[keys[i]]) | ||
|  |     `
 | ||
|  |   } else if (type === 'null') { | ||
|  |     code += `
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) +':null' | ||
|  |     `
 | ||
|  |   } else if (type === 'string') { | ||
|  |     code += `
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) + ':' + ${stringSerializer}(obj[keys[i]]) | ||
|  |     `
 | ||
|  |   } else if (type === 'integer') { | ||
|  |     code += `
 | ||
|  |         var t = Number(obj[keys[i]]) | ||
|  |     `
 | ||
|  |     if (isLong) { | ||
|  |       code += `
 | ||
|  |           if (isLong(obj[keys[i]]) || !isNaN(t)) { | ||
|  |             ${addComma} | ||
|  |             json += $asString(keys[i]) + ':' + $asInteger(obj[keys[i]]) | ||
|  |           } | ||
|  |       `
 | ||
|  |     } else { | ||
|  |       code += `
 | ||
|  |           if (!isNaN(t)) { | ||
|  |             ${addComma} | ||
|  |             json += $asString(keys[i]) + ':' + t | ||
|  |           } | ||
|  |       `
 | ||
|  |     } | ||
|  |   } else if (type === 'number') { | ||
|  |     code += `
 | ||
|  |         var t = Number(obj[keys[i]]) | ||
|  |         if (!isNaN(t)) { | ||
|  |           ${addComma} | ||
|  |           json += $asString(keys[i]) + ':' + t | ||
|  |         } | ||
|  |     `
 | ||
|  |   } else if (type === 'boolean') { | ||
|  |     code += `
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) + ':' + $asBoolean(obj[keys[i]]) | ||
|  |     `
 | ||
|  |   } else if (type === undefined) { | ||
|  |     code += `
 | ||
|  |         ${addComma} | ||
|  |         json += $asString(keys[i]) + ':' + $asAny(obj[keys[i]]) | ||
|  |     `
 | ||
|  |   } else { | ||
|  |     code += `
 | ||
|  |         throw new Error('Cannot coerce ' + obj[keys[i]] + ' to ' + ${JSON.stringify(type)}) | ||
|  |     `
 | ||
|  |   } | ||
|  |   return code | ||
|  | } | ||
|  | 
 | ||
|  | function addAdditionalProperties (location) { | ||
|  |   return `
 | ||
|  |       var properties = ${JSON.stringify(location.schema.properties)} || {} | ||
|  |       var keys = Object.keys(obj) | ||
|  |       for (var i = 0; i < keys.length; i++) { | ||
|  |         if (properties[keys[i]]) continue | ||
|  |         ${additionalProperty(location)} | ||
|  |       } | ||
|  |   `
 | ||
|  | } | ||
|  | 
 | ||
|  | function idFinder (schema, searchedId) { | ||
|  |   let objSchema | ||
|  |   const explore = (schema, searchedId) => { | ||
|  |     Object.keys(schema || {}).forEach((key, i, a) => { | ||
|  |       if (key === '$id' && schema[key] === searchedId) { | ||
|  |         objSchema = schema | ||
|  |       } else if (objSchema === undefined && typeof schema[key] === 'object') { | ||
|  |         explore(schema[key], searchedId) | ||
|  |       } | ||
|  |     }) | ||
|  |   } | ||
|  |   explore(schema, searchedId) | ||
|  |   return objSchema | ||
|  | } | ||
|  | 
 | ||
|  | function refFinder (ref, location) { | ||
|  |   const externalSchema = location.externalSchema | ||
|  |   let root = location.root | ||
|  |   let schema = location.schema | ||
|  | 
 | ||
|  |   if (externalSchema && externalSchema[ref]) { | ||
|  |     return { | ||
|  |       schema: externalSchema[ref], | ||
|  |       root: externalSchema[ref], | ||
|  |       externalSchema: externalSchema | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   // Split file from walk
 | ||
|  |   ref = ref.split('#') | ||
|  | 
 | ||
|  |   // If external file
 | ||
|  |   if (ref[0]) { | ||
|  |     schema = externalSchema[ref[0]] | ||
|  |     root = externalSchema[ref[0]] | ||
|  | 
 | ||
|  |     if (schema === undefined) { | ||
|  |       findBadKey(externalSchema, [ref[0]]) | ||
|  |     } | ||
|  | 
 | ||
|  |     if (schema.$ref) { | ||
|  |       return refFinder(schema.$ref, { | ||
|  |         schema: schema, | ||
|  |         root: root, | ||
|  |         externalSchema: externalSchema | ||
|  |       }) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   let code = 'return schema' | ||
|  |   // If it has a path
 | ||
|  |   if (ref[1]) { | ||
|  |     // ref[1] could contain a JSON pointer - ex: /definitions/num
 | ||
|  |     // or plain name fragment id without suffix # - ex: customId
 | ||
|  |     const walk = ref[1].split('/') | ||
|  |     if (walk.length === 1) { | ||
|  |       const targetId = `#${ref[1]}` | ||
|  |       let dereferenced = idFinder(schema, targetId) | ||
|  |       if (dereferenced === undefined && !ref[0]) { | ||
|  |         // eslint-disable-next-line
 | ||
|  |         for (var key of Object.keys(externalSchema)) { | ||
|  |           dereferenced = idFinder(externalSchema[key], targetId) | ||
|  |           if (dereferenced !== undefined) { | ||
|  |             root = externalSchema[key] | ||
|  |             break | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       return { | ||
|  |         schema: dereferenced, | ||
|  |         root: root, | ||
|  |         externalSchema: externalSchema | ||
|  |       } | ||
|  |     } else { | ||
|  |       // eslint-disable-next-line
 | ||
|  |       for (var i = 1; i < walk.length; i++) { | ||
|  |         code += `[${JSON.stringify(walk[i])}]` | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   let result | ||
|  |   try { | ||
|  |     result = (new Function('schema', code))(root) | ||
|  |   } catch (err) {} | ||
|  | 
 | ||
|  |   if (result === undefined && ref[1]) { | ||
|  |     const walk = ref[1].split('/') | ||
|  |     findBadKey(schema, walk.slice(1)) | ||
|  |   } | ||
|  | 
 | ||
|  |   if (result.$ref) { | ||
|  |     return refFinder(result.$ref, { | ||
|  |       schema: schema, | ||
|  |       root: root, | ||
|  |       externalSchema: externalSchema | ||
|  |     }) | ||
|  |   } | ||
|  | 
 | ||
|  |   return { | ||
|  |     schema: result, | ||
|  |     root: root, | ||
|  |     externalSchema: externalSchema | ||
|  |   } | ||
|  | 
 | ||
|  |   function findBadKey (obj, keys) { | ||
|  |     if (keys.length === 0) return null | ||
|  |     const key = keys.shift() | ||
|  |     if (obj[key] === undefined) { | ||
|  |       stringSimilarity = stringSimilarity || require('string-similarity') | ||
|  |       const { bestMatch } = stringSimilarity.findBestMatch(key, Object.keys(obj)) | ||
|  |       if (bestMatch.rating >= 0.5) { | ||
|  |         throw new Error(`Cannot find reference ${JSON.stringify(key)}, did you mean ${JSON.stringify(bestMatch.target)}?`) | ||
|  |       } else { | ||
|  |         throw new Error(`Cannot find reference ${JSON.stringify(key)}`) | ||
|  |       } | ||
|  |     } | ||
|  |     return findBadKey(obj[key], keys) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function buildCode (location, code, laterCode, name) { | ||
|  |   if (location.schema.$ref) { | ||
|  |     location = refFinder(location.schema.$ref, location) | ||
|  |   } | ||
|  | 
 | ||
|  |   const schema = location.schema | ||
|  |   let required = schema.required | ||
|  | 
 | ||
|  |   Object.keys(schema.properties || {}).forEach((key, i, a) => { | ||
|  |     let propertyLocation = mergeLocation(location, { schema: schema.properties[key] }) | ||
|  |     if (schema.properties[key].$ref) { | ||
|  |       propertyLocation = refFinder(schema.properties[key].$ref, location) | ||
|  |       schema.properties[key] = propertyLocation.schema | ||
|  |     } | ||
|  | 
 | ||
|  |     // Using obj['key'] !== undefined instead of obj.hasOwnProperty(prop) for perf reasons,
 | ||
|  |     // see https://github.com/mcollina/fast-json-stringify/pull/3 for discussion.
 | ||
|  | 
 | ||
|  |     const type = schema.properties[key].type | ||
|  |     const nullable = schema.properties[key].nullable | ||
|  |     const sanitized = JSON.stringify(key) | ||
|  |     const asString = JSON.stringify(sanitized) | ||
|  | 
 | ||
|  |     if (nullable) { | ||
|  |       code += `
 | ||
|  |         if (obj[${sanitized}] === null) { | ||
|  |           ${addComma} | ||
|  |           json += ${asString} + ':null' | ||
|  |           var rendered = true | ||
|  |         } else { | ||
|  |       `
 | ||
|  |     } | ||
|  | 
 | ||
|  |     if (type === 'number') { | ||
|  |       code += `
 | ||
|  |           var t = Number(obj[${sanitized}]) | ||
|  |           if (!isNaN(t)) { | ||
|  |             ${addComma} | ||
|  |             json += ${asString} + ':' + t | ||
|  |       `
 | ||
|  |     } else if (type === 'integer') { | ||
|  |       code += `
 | ||
|  |           var rendered = false | ||
|  |       `
 | ||
|  |       if (isLong) { | ||
|  |         code += `
 | ||
|  |             if (isLong(obj[${sanitized}])) { | ||
|  |               ${addComma} | ||
|  |               json += ${asString} + ':' + obj[${sanitized}].toString() | ||
|  |               rendered = true | ||
|  |             } else { | ||
|  |               var t = Number(obj[${sanitized}]) | ||
|  |               if (!isNaN(t)) { | ||
|  |                 ${addComma} | ||
|  |                 json += ${asString} + ':' + t | ||
|  |                 rendered = true | ||
|  |               } | ||
|  |             } | ||
|  |         `
 | ||
|  |       } else { | ||
|  |         code += `
 | ||
|  |             var t = $asInteger(obj[${sanitized}]) | ||
|  |             if (!isNaN(t)) { | ||
|  |               ${addComma} | ||
|  |               json += ${asString} + ':' + t | ||
|  |               rendered = true | ||
|  |             } | ||
|  |         `
 | ||
|  |       } | ||
|  |       code += `
 | ||
|  |           if (rendered) { | ||
|  |       `
 | ||
|  |     } else { | ||
|  |       code += `
 | ||
|  |         if (obj[${sanitized}] !== undefined) { | ||
|  |           ${addComma} | ||
|  |           json += ${asString} + ':' | ||
|  |         `
 | ||
|  | 
 | ||
|  |       const result = nested(laterCode, name, key, mergeLocation(propertyLocation, { schema: schema.properties[key] }), undefined, false) | ||
|  |       code += result.code | ||
|  |       laterCode = result.laterCode | ||
|  |     } | ||
|  | 
 | ||
|  |     const defaultValue = schema.properties[key].default | ||
|  |     if (defaultValue !== undefined) { | ||
|  |       required = filterRequired(required, key) | ||
|  |       code += `
 | ||
|  |       } else { | ||
|  |         ${addComma} | ||
|  |         json += ${asString} + ':' + ${JSON.stringify(JSON.stringify(defaultValue))} | ||
|  |       `
 | ||
|  |     } else if (required && required.indexOf(key) !== -1) { | ||
|  |       required = filterRequired(required, key) | ||
|  |       code += `
 | ||
|  |       } else { | ||
|  |         throw new Error('${sanitized} is required!') | ||
|  |       `
 | ||
|  |     } | ||
|  | 
 | ||
|  |     code += `
 | ||
|  |       } | ||
|  |     `
 | ||
|  | 
 | ||
|  |     if (nullable) { | ||
|  |       code += `
 | ||
|  |         } | ||
|  |       `
 | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   if (required && required.length > 0) { | ||
|  |     code += 'var required = [' | ||
|  |     // eslint-disable-next-line
 | ||
|  |     for (var i = 0; i < required.length; i++) { | ||
|  |       if (i > 0) { | ||
|  |         code += ',' | ||
|  |       } | ||
|  |       code += `${JSON.stringify(required[i])}` | ||
|  |     } | ||
|  |     code += `]
 | ||
|  |       for (var i = 0; i < required.length; i++) { | ||
|  |         if (obj[required[i]] === undefined) throw new Error('"' + required[i] + '" is required!') | ||
|  |       } | ||
|  |     `
 | ||
|  |   } | ||
|  | 
 | ||
|  |   if (schema.allOf) { | ||
|  |     const builtCode = buildCodeWithAllOfs(location, code, laterCode, name) | ||
|  |     code = builtCode.code | ||
|  |     laterCode = builtCode.laterCode | ||
|  |   } | ||
|  | 
 | ||
|  |   return { code: code, laterCode: laterCode } | ||
|  | } | ||
|  | 
 | ||
|  | function filterRequired (required, key) { | ||
|  |   if (!required) { | ||
|  |     return required | ||
|  |   } | ||
|  |   return required.filter(k => k !== key) | ||
|  | } | ||
|  | 
 | ||
|  | function buildCodeWithAllOfs (location, code, laterCode, name) { | ||
|  |   if (location.schema.allOf) { | ||
|  |     location.schema.allOf.forEach((ss) => { | ||
|  |       const builtCode = buildCodeWithAllOfs(mergeLocation(location, { schema: ss }), code, laterCode, name) | ||
|  |       code = builtCode.code | ||
|  |       laterCode = builtCode.laterCode | ||
|  |     }) | ||
|  |   } else { | ||
|  |     const builtCode = buildCode(location, code, laterCode, name) | ||
|  | 
 | ||
|  |     code = builtCode.code | ||
|  |     laterCode = builtCode.laterCode | ||
|  |   } | ||
|  | 
 | ||
|  |   return { code: code, laterCode: laterCode } | ||
|  | } | ||
|  | 
 | ||
|  | function buildInnerObject (location, name) { | ||
|  |   const schema = location.schema | ||
|  |   const result = buildCodeWithAllOfs(location, '', '', name) | ||
|  |   if (schema.patternProperties) { | ||
|  |     result.code += addPatternProperties(location) | ||
|  |   } else if (schema.additionalProperties && !schema.patternProperties) { | ||
|  |     result.code += addAdditionalProperties(location) | ||
|  |   } | ||
|  |   return result | ||
|  | } | ||
|  | 
 | ||
|  | function addIfThenElse (location, name) { | ||
|  |   let code = '' | ||
|  |   let r | ||
|  |   let laterCode = '' | ||
|  |   let innerR | ||
|  | 
 | ||
|  |   const schema = location.schema | ||
|  |   const copy = merge({}, schema) | ||
|  |   const i = copy.if | ||
|  |   const then = copy.then | ||
|  |   const e = copy.else ? copy.else : { additionalProperties: true } | ||
|  |   delete copy.if | ||
|  |   delete copy.then | ||
|  |   delete copy.else | ||
|  |   let merged = merge(copy, then) | ||
|  |   let mergedLocation = mergeLocation(location, { schema: merged }) | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     valid = ajv.validate(${JSON.stringify(i)}, obj) | ||
|  |     if (valid) { | ||
|  |   `
 | ||
|  |   if (merged.if && merged.then) { | ||
|  |     innerR = addIfThenElse(mergedLocation, name + 'Then') | ||
|  |     code += innerR.code | ||
|  |     laterCode = innerR.laterCode | ||
|  |   } | ||
|  | 
 | ||
|  |   r = buildInnerObject(mergedLocation, name + 'Then') | ||
|  |   code += r.code | ||
|  |   laterCode += r.laterCode | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     } | ||
|  |   `
 | ||
|  |   merged = merge(copy, e) | ||
|  |   mergedLocation = mergeLocation(mergedLocation, { schema: merged }) | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |       else { | ||
|  |     `
 | ||
|  | 
 | ||
|  |   if (merged.if && merged.then) { | ||
|  |     innerR = addIfThenElse(mergedLocation, name + 'Else') | ||
|  |     code += innerR.code | ||
|  |     laterCode += innerR.laterCode | ||
|  |   } | ||
|  | 
 | ||
|  |   r = buildInnerObject(mergedLocation, name + 'Else') | ||
|  |   code += r.code | ||
|  |   laterCode += r.laterCode | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |       } | ||
|  |     `
 | ||
|  |   return { code: code, laterCode: laterCode } | ||
|  | } | ||
|  | 
 | ||
|  | function toJSON (variableName) { | ||
|  |   return `(${variableName} && typeof ${variableName}.toJSON === 'function')
 | ||
|  |     ? ${variableName}.toJSON() | ||
|  |     : ${variableName} | ||
|  |   `
 | ||
|  | } | ||
|  | 
 | ||
|  | function buildObject (location, code, name) { | ||
|  |   const schema = location.schema | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     function ${name} (input) { | ||
|  |   `
 | ||
|  |   if (schema.nullable) { | ||
|  |     code += `
 | ||
|  |       if(input === null) { | ||
|  |         return 'null'; | ||
|  |       } | ||
|  |   `
 | ||
|  |   } | ||
|  | 
 | ||
|  |   if (objectReferenceSerializersMap.has(schema)) { | ||
|  |     code += `
 | ||
|  |       return ${objectReferenceSerializersMap.get(schema)}(input) | ||
|  |     } | ||
|  |     `
 | ||
|  |     return code | ||
|  |   } | ||
|  |   objectReferenceSerializersMap.set(schema, name) | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |       var obj = ${toJSON('input')} | ||
|  |       var json = '{' | ||
|  |       var addComma = false | ||
|  |   `
 | ||
|  | 
 | ||
|  |   let r | ||
|  |   if (schema.if && schema.then) { | ||
|  |     code += `
 | ||
|  |       var valid | ||
|  |     `
 | ||
|  |     r = addIfThenElse(location, name) | ||
|  |   } else { | ||
|  |     r = buildInnerObject(location, name) | ||
|  |   } | ||
|  | 
 | ||
|  |   // Removes the comma if is the last element of the string (in case there are not properties)
 | ||
|  |   code += `${r.code}
 | ||
|  |       json += '}' | ||
|  |       return json | ||
|  |     } | ||
|  |     ${r.laterCode} | ||
|  |   `
 | ||
|  | 
 | ||
|  |   return code | ||
|  | } | ||
|  | 
 | ||
|  | function buildArray (location, code, name, key = null) { | ||
|  |   let schema = location.schema | ||
|  |   code += `
 | ||
|  |     function ${name} (obj) { | ||
|  |   `
 | ||
|  |   if (schema.nullable) { | ||
|  |     code += `
 | ||
|  |       if(obj === null) { | ||
|  |         return 'null'; | ||
|  |       } | ||
|  |     `
 | ||
|  |   } | ||
|  |   const laterCode = '' | ||
|  | 
 | ||
|  |   // default to any items type
 | ||
|  |   if (!schema.items) { | ||
|  |     schema.items = {} | ||
|  |   } | ||
|  | 
 | ||
|  |   if (schema.items.$ref) { | ||
|  |     if (!schema[fjsCloned]) { | ||
|  |       location.schema = clone(location.schema) | ||
|  |       schema = location.schema | ||
|  |       schema[fjsCloned] = true | ||
|  |     } | ||
|  | 
 | ||
|  |     location = refFinder(schema.items.$ref, location) | ||
|  |     schema.items = location.schema | ||
|  | 
 | ||
|  |     if (arrayItemsReferenceSerializersMap.has(schema.items)) { | ||
|  |       code += `
 | ||
|  |       return ${arrayItemsReferenceSerializersMap.get(schema.items)}(obj) | ||
|  |       } | ||
|  |       `
 | ||
|  |       return code | ||
|  |     } | ||
|  |     arrayItemsReferenceSerializersMap.set(schema.items, name) | ||
|  |   } | ||
|  | 
 | ||
|  |   let result = { code: '', laterCode: '' } | ||
|  |   const accessor = '[i]' | ||
|  |   if (Array.isArray(schema.items)) { | ||
|  |     result = schema.items.reduce((res, item, i) => { | ||
|  |       const tmpRes = nested(laterCode, name, accessor, mergeLocation(location, { schema: item }), i, true) | ||
|  |       const condition = `i === ${i} && ${buildArrayTypeCondition(item.type, accessor)}` | ||
|  |       return { | ||
|  |         code: `${res.code}
 | ||
|  |         ${i > 0 ? 'else' : ''} if (${condition}) { | ||
|  |           ${tmpRes.code} | ||
|  |         }`,
 | ||
|  |         laterCode: `${res.laterCode}
 | ||
|  |         ${tmpRes.laterCode}`
 | ||
|  |       } | ||
|  |     }, result) | ||
|  | 
 | ||
|  |     if (schema.additionalItems) { | ||
|  |       const tmpRes = nested(laterCode, name, accessor, mergeLocation(location, { schema: schema.items }), undefined, true) | ||
|  |       result.code += `
 | ||
|  |       else if (i >= ${schema.items.length}) { | ||
|  |         ${tmpRes.code} | ||
|  |       } | ||
|  |       `
 | ||
|  |     } | ||
|  | 
 | ||
|  |     result.code += `
 | ||
|  |     else { | ||
|  |       throw new Error(\`Item at $\{i} does not match schema definition.\`)
 | ||
|  |     } | ||
|  |     `
 | ||
|  |   } else { | ||
|  |     result = nested(laterCode, name, accessor, mergeLocation(location, { schema: schema.items }), undefined, true) | ||
|  |   } | ||
|  | 
 | ||
|  |   if (key) { | ||
|  |     code += `
 | ||
|  |     if(!Array.isArray(obj)) { | ||
|  |       throw new TypeError(\`Property '${key}' should be of type array, received '$\{obj}' instead.\`)
 | ||
|  |     } | ||
|  |     `
 | ||
|  |   } | ||
|  | 
 | ||
|  |   code += `
 | ||
|  |     var l = obj.length | ||
|  |     var jsonOutput= '' | ||
|  |     for (var i = 0; i < l; i++) { | ||
|  |       var json = '' | ||
|  |       ${result.code} | ||
|  |       jsonOutput += json | ||
|  | 
 | ||
|  |       if (json.length > 0 && i < l - 1) { | ||
|  |         jsonOutput += ',' | ||
|  |       } | ||
|  |     } | ||
|  |     return \`[\${jsonOutput}]\`
 | ||
|  |   } | ||
|  |   ${result.laterCode} | ||
|  |   `
 | ||
|  | 
 | ||
|  |   return code | ||
|  | } | ||
|  | 
 | ||
|  | function buildArrayTypeCondition (type, accessor) { | ||
|  |   let condition | ||
|  |   switch (type) { | ||
|  |     case 'null': | ||
|  |       condition = `obj${accessor} === null` | ||
|  |       break | ||
|  |     case 'string': | ||
|  |       condition = `typeof obj${accessor} === 'string'` | ||
|  |       break | ||
|  |     case 'integer': | ||
|  |       condition = `Number.isInteger(obj${accessor})` | ||
|  |       break | ||
|  |     case 'number': | ||
|  |       condition = `Number.isFinite(obj${accessor})` | ||
|  |       break | ||
|  |     case 'boolean': | ||
|  |       condition = `typeof obj${accessor} === 'boolean'` | ||
|  |       break | ||
|  |     case 'object': | ||
|  |       condition = `obj${accessor} && typeof obj${accessor} === 'object' && obj${accessor}.constructor === Object` | ||
|  |       break | ||
|  |     case 'array': | ||
|  |       condition = `Array.isArray(obj${accessor})` | ||
|  |       break | ||
|  |     default: | ||
|  |       if (Array.isArray(type)) { | ||
|  |         const conditions = type.map((subType) => { | ||
|  |           return buildArrayTypeCondition(subType, accessor) | ||
|  |         }) | ||
|  |         condition = `(${conditions.join(' || ')})` | ||
|  |       } else { | ||
|  |         throw new Error(`${type} unsupported`) | ||
|  |       } | ||
|  |   } | ||
|  |   return condition | ||
|  | } | ||
|  | 
 | ||
|  | function dereferenceOfRefs (location, type) { | ||
|  |   if (!location.schema[fjsCloned]) { | ||
|  |     const schemaClone = clone(location.schema) | ||
|  |     schemaClone[fjsCloned] = true | ||
|  |     location.schema = schemaClone | ||
|  |   } | ||
|  | 
 | ||
|  |   const schema = location.schema | ||
|  |   const locations = [] | ||
|  | 
 | ||
|  |   schema[type].forEach((s, index) => { | ||
|  |     // follow the refs
 | ||
|  |     let sLocation = mergeLocation(location, { schema: s }) | ||
|  |     while (s.$ref) { | ||
|  |       sLocation = refFinder(s.$ref, sLocation) | ||
|  |       schema[type][index] = sLocation.schema | ||
|  |       s = schema[type][index] | ||
|  |     } | ||
|  |     locations[index] = sLocation | ||
|  |   }) | ||
|  | 
 | ||
|  |   return locations | ||
|  | } | ||
|  | 
 | ||
|  | let strNameCounter = 0 | ||
|  | function asFuncName (str) { | ||
|  |   // only allow chars that can work
 | ||
|  |   let rep = str.replace(/[^a-zA-Z0-9$_]/g, '') | ||
|  | 
 | ||
|  |   if (rep.length === 0) { | ||
|  |     return 'anan' + strNameCounter++ | ||
|  |   } else if (rep !== str) { | ||
|  |     rep += strNameCounter++ | ||
|  |   } | ||
|  | 
 | ||
|  |   return rep | ||
|  | } | ||
|  | 
 | ||
|  | function nested (laterCode, name, key, location, subKey, isArray) { | ||
|  |   let code = '' | ||
|  |   let funcName | ||
|  | 
 | ||
|  |   subKey = subKey || '' | ||
|  | 
 | ||
|  |   let schema = location.schema | ||
|  | 
 | ||
|  |   if (schema.$ref) { | ||
|  |     schema = refFinder(schema.$ref, location) | ||
|  |   } | ||
|  | 
 | ||
|  |   if (schema.type === undefined) { | ||
|  |     const inferredType = inferTypeByKeyword(schema) | ||
|  |     if (inferredType) { | ||
|  |       schema.type = inferredType | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   const type = schema.type | ||
|  |   const nullable = schema.nullable === true | ||
|  | 
 | ||
|  |   const accessor = isArray ? key : `[${JSON.stringify(key)}]` | ||
|  | 
 | ||
|  |   switch (type) { | ||
|  |     case 'null': | ||
|  |       code += `
 | ||
|  |         json += $asNull() | ||
|  |       `
 | ||
|  |       break | ||
|  |     case 'string': { | ||
|  |       const stringSerializer = getStringSerializer(schema.format) | ||
|  |       code += nullable ? `json += obj${accessor} === null ? null : ${stringSerializer}(obj${accessor})` : `json += ${stringSerializer}(obj${accessor})` | ||
|  |       break | ||
|  |     } | ||
|  |     case 'integer': | ||
|  |       code += nullable ? `json += obj${accessor} === null ? null : $asInteger(obj${accessor})` : `json += $asInteger(obj${accessor})` | ||
|  |       break | ||
|  |     case 'number': | ||
|  |       code += nullable ? `json += obj${accessor} === null ? null : $asNumber(obj${accessor})` : `json += $asNumber(obj${accessor})` | ||
|  |       break | ||
|  |     case 'boolean': | ||
|  |       code += nullable ? `json += obj${accessor} === null ? null : $asBoolean(obj${accessor})` : `json += $asBoolean(obj${accessor})` | ||
|  |       break | ||
|  |     case 'object': | ||
|  |       funcName = asFuncName(name + key + subKey) | ||
|  |       laterCode = buildObject(location, laterCode, funcName) | ||
|  |       code += `
 | ||
|  |         json += ${funcName}(obj${accessor}) | ||
|  |       `
 | ||
|  |       break | ||
|  |     case 'array': | ||
|  |       funcName = asFuncName('$arr' + name + key + subKey) // eslint-disable-line
 | ||
|  |       laterCode = buildArray(location, laterCode, funcName, key) | ||
|  |       code += `
 | ||
|  |         json += ${funcName}(obj${accessor}) | ||
|  |       `
 | ||
|  |       break | ||
|  |     case undefined: | ||
|  |       if ('anyOf' in schema) { | ||
|  |         // beware: dereferenceOfRefs has side effects and changes schema.anyOf
 | ||
|  |         const anyOfLocations = dereferenceOfRefs(location, 'anyOf') | ||
|  |         anyOfLocations.forEach((location, index) => { | ||
|  |           const nestedResult = nested(laterCode, name, key, location, subKey !== '' ? subKey : 'i' + index, isArray) | ||
|  |           // We need a test serializer as the String serializer will not work with
 | ||
|  |           // date/time ajv validations
 | ||
|  |           // see: https://github.com/fastify/fast-json-stringify/issues/325
 | ||
|  |           const testSerializer = getTestSerializer(location.schema.format) | ||
|  |           const testValue = testSerializer !== undefined ? `${testSerializer}(obj${accessor}, true)` : `obj${accessor}` | ||
|  | 
 | ||
|  |           // Since we are only passing the relevant schema to ajv.validate, it needs to be full dereferenced
 | ||
|  |           // otherwise any $ref pointing to an external schema would result in an error.
 | ||
|  |           // Full dereference of the schema happens as side effect of two functions:
 | ||
|  |           // 1. `dereferenceOfRefs` loops through the `schema.anyOf`` array and replaces any top level reference
 | ||
|  |           // with the actual schema
 | ||
|  |           // 2. `nested`, through `buildCode`, replaces any reference in object properties with the actual schema
 | ||
|  |           // (see https://github.com/fastify/fast-json-stringify/blob/6da3b3e8ac24b1ca5578223adedb4083b7adf8db/index.js#L631)
 | ||
|  |           code += `
 | ||
|  |             ${index === 0 ? 'if' : 'else if'}(ajv.validate(${JSON.stringify(location.schema)}, ${testValue})) | ||
|  |               ${nestedResult.code} | ||
|  |           `
 | ||
|  |           laterCode = nestedResult.laterCode | ||
|  |         }) | ||
|  |         code += `
 | ||
|  |           else json+= null | ||
|  |         `
 | ||
|  |       } else if ('oneOf' in schema) { | ||
|  |         // beware: dereferenceOfRefs has side effects and changes schema.oneOf
 | ||
|  |         const oneOfLocations = dereferenceOfRefs(location, 'oneOf') | ||
|  |         oneOfLocations.forEach((location, index) => { | ||
|  |           const nestedResult = nested(laterCode, name, key, location, subKey !== '' ? subKey : 'i' + index, isArray) | ||
|  |           const testSerializer = getTestSerializer(location.schema.format) | ||
|  |           const testValue = testSerializer !== undefined ? `${testSerializer}(obj${accessor}, true)` : `obj${accessor}` | ||
|  |           // see comment on anyOf about dereferencing the schema before calling ajv.validate
 | ||
|  |           code += `
 | ||
|  |             ${index === 0 ? 'if' : 'else if'}(ajv.validate(${JSON.stringify(location.schema)}, ${testValue})) | ||
|  |               ${nestedResult.code} | ||
|  |           `
 | ||
|  |           laterCode = nestedResult.laterCode | ||
|  |         }) | ||
|  | 
 | ||
|  |         if (!isArray) { | ||
|  |           code += `
 | ||
|  |             else json+= null | ||
|  |           `
 | ||
|  |         } | ||
|  |       } else if (isEmpty(schema)) { | ||
|  |         code += `
 | ||
|  |           json += JSON.stringify(obj${accessor}) | ||
|  |         `
 | ||
|  |       } else if ('const' in schema) { | ||
|  |         code += `
 | ||
|  |           if(ajv.validate(${JSON.stringify(schema)}, obj${accessor})) | ||
|  |             json += '${JSON.stringify(schema.const)}' | ||
|  |           else | ||
|  |             throw new Error(\`Item $\{JSON.stringify(obj${accessor})} does not match schema definition.\`)
 | ||
|  |         `
 | ||
|  |       } else if (schema.type === undefined) { | ||
|  |         code += `
 | ||
|  |           json += JSON.stringify(obj${accessor}) | ||
|  |         `
 | ||
|  |       } else { | ||
|  |         throw new Error(`${schema.type} unsupported`) | ||
|  |       } | ||
|  |       break | ||
|  |     default: | ||
|  |       if (Array.isArray(type)) { | ||
|  |         const nullIndex = type.indexOf('null') | ||
|  |         const sortedTypes = nullIndex !== -1 ? [type[nullIndex]].concat(type.slice(0, nullIndex)).concat(type.slice(nullIndex + 1)) : type | ||
|  |         sortedTypes.forEach((type, index) => { | ||
|  |           const statement = index === 0 ? 'if' : 'else if' | ||
|  |           const tempSchema = Object.assign({}, schema, { type }) | ||
|  |           const nestedResult = nested(laterCode, name, key, mergeLocation(location, { schema: tempSchema }), subKey, isArray) | ||
|  |           switch (type) { | ||
|  |             case 'string': { | ||
|  |               code += `
 | ||
|  |                 ${statement}(obj${accessor} === null || typeof obj${accessor} === "${type}" || obj${accessor} instanceof Date || typeof obj${accessor}.toISOString === "function" || obj${accessor} instanceof RegExp || (typeof obj${accessor} === "object" && Object.hasOwnProperty.call(obj${accessor}, "toString"))) | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |             case 'null': { | ||
|  |               code += `
 | ||
|  |                 ${statement}(obj${accessor} == null) | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |             case 'array': { | ||
|  |               code += `
 | ||
|  |                 ${statement}(Array.isArray(obj${accessor})) | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |             case 'integer': { | ||
|  |               code += `
 | ||
|  |                 ${statement}(Number.isInteger(obj${accessor}) || obj${accessor} === null) | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |             case 'number': { | ||
|  |               code += `
 | ||
|  |                 ${statement}(isNaN(obj${accessor}) === false) | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |             default: { | ||
|  |               code += `
 | ||
|  |                 ${statement}(typeof obj${accessor} === "${type}") | ||
|  |                   ${nestedResult.code} | ||
|  |               `
 | ||
|  |               break | ||
|  |             } | ||
|  |           } | ||
|  |           laterCode = nestedResult.laterCode | ||
|  |         }) | ||
|  |         code += `
 | ||
|  |           else json+= null | ||
|  |         `
 | ||
|  |       } else { | ||
|  |         throw new Error(`${type} unsupported`) | ||
|  |       } | ||
|  |   } | ||
|  | 
 | ||
|  |   return { | ||
|  |     code, | ||
|  |     laterCode | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function isEmpty (schema) { | ||
|  |   // eslint-disable-next-line
 | ||
|  |   for (var key in schema) { | ||
|  |     if (schema.hasOwnProperty(key) && schema[key] !== undefined) { | ||
|  |       return false | ||
|  |     } | ||
|  |   } | ||
|  |   return true | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = build | ||
|  | 
 | ||
|  | module.exports.restore = function (debugModeStr, options = {}) { | ||
|  |   const dependencies = [debugModeStr] | ||
|  |   const args = [] | ||
|  |   if (debugModeStr.startsWith('ajv')) { | ||
|  |     dependencies.unshift('ajv') | ||
|  |     args.push(new Ajv(options.ajv)) | ||
|  |   } | ||
|  | 
 | ||
|  |   // eslint-disable-next-line
 | ||
|  |   return (Function.apply(null, ['ajv', debugModeStr]) | ||
|  |     .apply(null, args)) | ||
|  | } |