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.
		
		
		
		
		
			
		
			
				
					
					
						
							146 lines
						
					
					
						
							6.0 KiB
						
					
					
				
			
		
		
	
	
							146 lines
						
					
					
						
							6.0 KiB
						
					
					
				| "use strict";
 | |
| Object.defineProperty(exports, "__esModule", { value: true });
 | |
| exports.validateProperties = exports.error = void 0;
 | |
| const code_1 = require("../code");
 | |
| const util_1 = require("../../compile/util");
 | |
| const codegen_1 = require("../../compile/codegen");
 | |
| const metadata_1 = require("./metadata");
 | |
| const nullable_1 = require("./nullable");
 | |
| const error_1 = require("./error");
 | |
| var PropError;
 | |
| (function (PropError) {
 | |
|     PropError["Additional"] = "additional";
 | |
|     PropError["Missing"] = "missing";
 | |
| })(PropError || (PropError = {}));
 | |
| exports.error = {
 | |
|     message: (cxt) => {
 | |
|         const { params } = cxt;
 | |
|         return params.propError
 | |
|             ? params.propError === PropError.Additional
 | |
|                 ? "must NOT have additional properties"
 | |
|                 : `must have property '${params.missingProperty}'`
 | |
|             : (0, error_1.typeErrorMessage)(cxt, "object");
 | |
|     },
 | |
|     params: (cxt) => {
 | |
|         const { params } = cxt;
 | |
|         return params.propError
 | |
|             ? params.propError === PropError.Additional
 | |
|                 ? (0, codegen_1._) `{error: ${params.propError}, additionalProperty: ${params.additionalProperty}}`
 | |
|                 : (0, codegen_1._) `{error: ${params.propError}, missingProperty: ${params.missingProperty}}`
 | |
|             : (0, error_1.typeErrorParams)(cxt, "object");
 | |
|     },
 | |
| };
 | |
| const def = {
 | |
|     keyword: "properties",
 | |
|     schemaType: "object",
 | |
|     error: exports.error,
 | |
|     code: validateProperties,
 | |
| };
 | |
| // const error: KeywordErrorDefinition = {
 | |
| //   message: "should NOT have additional properties",
 | |
| //   params: ({params}) => _`{additionalProperty: ${params.additionalProperty}}`,
 | |
| // }
 | |
| function validateProperties(cxt) {
 | |
|     (0, metadata_1.checkMetadata)(cxt);
 | |
|     const { gen, data, parentSchema, it } = cxt;
 | |
|     const { additionalProperties, nullable } = parentSchema;
 | |
|     if (it.jtdDiscriminator && nullable)
 | |
|         throw new Error("JTD: nullable inside discriminator mapping");
 | |
|     if (commonProperties()) {
 | |
|         throw new Error("JTD: properties and optionalProperties have common members");
 | |
|     }
 | |
|     const [allProps, properties] = schemaProperties("properties");
 | |
|     const [allOptProps, optProperties] = schemaProperties("optionalProperties");
 | |
|     if (properties.length === 0 && optProperties.length === 0 && additionalProperties) {
 | |
|         return;
 | |
|     }
 | |
|     const [valid, cond] = it.jtdDiscriminator === undefined
 | |
|         ? (0, nullable_1.checkNullableObject)(cxt, data)
 | |
|         : [gen.let("valid", false), true];
 | |
|     gen.if(cond, () => gen.assign(valid, true).block(() => {
 | |
|         validateProps(properties, "properties", true);
 | |
|         validateProps(optProperties, "optionalProperties");
 | |
|         if (!additionalProperties)
 | |
|             validateAdditional();
 | |
|     }));
 | |
|     cxt.pass(valid);
 | |
|     function commonProperties() {
 | |
|         const props = parentSchema.properties;
 | |
|         const optProps = parentSchema.optionalProperties;
 | |
|         if (!(props && optProps))
 | |
|             return false;
 | |
|         for (const p in props) {
 | |
|             if (Object.prototype.hasOwnProperty.call(optProps, p))
 | |
|                 return true;
 | |
|         }
 | |
|         return false;
 | |
|     }
 | |
|     function schemaProperties(keyword) {
 | |
|         const schema = parentSchema[keyword];
 | |
|         const allPs = schema ? (0, code_1.allSchemaProperties)(schema) : [];
 | |
|         if (it.jtdDiscriminator && allPs.some((p) => p === it.jtdDiscriminator)) {
 | |
|             throw new Error(`JTD: discriminator tag used in ${keyword}`);
 | |
|         }
 | |
|         const ps = allPs.filter((p) => !(0, util_1.alwaysValidSchema)(it, schema[p]));
 | |
|         return [allPs, ps];
 | |
|     }
 | |
|     function validateProps(props, keyword, required) {
 | |
|         const _valid = gen.var("valid");
 | |
|         for (const prop of props) {
 | |
|             gen.if((0, code_1.propertyInData)(gen, data, prop, it.opts.ownProperties), () => applyPropertySchema(prop, keyword, _valid), () => missingProperty(prop));
 | |
|             cxt.ok(_valid);
 | |
|         }
 | |
|         function missingProperty(prop) {
 | |
|             if (required) {
 | |
|                 gen.assign(_valid, false);
 | |
|                 cxt.error(false, { propError: PropError.Missing, missingProperty: prop }, { schemaPath: prop });
 | |
|             }
 | |
|             else {
 | |
|                 gen.assign(_valid, true);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     function applyPropertySchema(prop, keyword, _valid) {
 | |
|         cxt.subschema({
 | |
|             keyword,
 | |
|             schemaProp: prop,
 | |
|             dataProp: prop,
 | |
|         }, _valid);
 | |
|     }
 | |
|     function validateAdditional() {
 | |
|         gen.forIn("key", data, (key) => {
 | |
|             const _allProps = it.jtdDiscriminator === undefined ? allProps : [it.jtdDiscriminator].concat(allProps);
 | |
|             const addProp = isAdditional(key, _allProps, "properties");
 | |
|             const addOptProp = isAdditional(key, allOptProps, "optionalProperties");
 | |
|             const extra = addProp === true ? addOptProp : addOptProp === true ? addProp : (0, codegen_1.and)(addProp, addOptProp);
 | |
|             gen.if(extra, () => {
 | |
|                 if (it.opts.removeAdditional) {
 | |
|                     gen.code((0, codegen_1._) `delete ${data}[${key}]`);
 | |
|                 }
 | |
|                 else {
 | |
|                     cxt.error(false, { propError: PropError.Additional, additionalProperty: key }, { instancePath: key, parentSchema: true });
 | |
|                     if (!it.opts.allErrors)
 | |
|                         gen.break();
 | |
|                 }
 | |
|             });
 | |
|         });
 | |
|     }
 | |
|     function isAdditional(key, props, keyword) {
 | |
|         let additional;
 | |
|         if (props.length > 8) {
 | |
|             // TODO maybe an option instead of hard-coded 8?
 | |
|             const propsSchema = (0, util_1.schemaRefOrVal)(it, parentSchema[keyword], keyword);
 | |
|             additional = (0, codegen_1.not)((0, code_1.isOwnProperty)(gen, propsSchema, key));
 | |
|         }
 | |
|         else if (props.length) {
 | |
|             additional = (0, codegen_1.and)(...props.map((p) => (0, codegen_1._) `${key} !== ${p}`));
 | |
|         }
 | |
|         else {
 | |
|             additional = true;
 | |
|         }
 | |
|         return additional;
 | |
|     }
 | |
| }
 | |
| exports.validateProperties = validateProperties;
 | |
| exports.default = def;
 | |
| //# sourceMappingURL=properties.js.map
 |