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.
		
		
		
		
		
			
		
			
				
					52 lines
				
				1.8 KiB
			
		
		
			
		
	
	
					52 lines
				
				1.8 KiB
			| 
											3 years ago
										 | import type {CodeKeywordDefinition} from "../../types" | ||
|  | import type {KeywordCxt} from "../../compile/validate" | ||
|  | import {_, getProperty, Code, Name} from "../../compile/codegen" | ||
|  | import N from "../../compile/names" | ||
|  | import {callRef} from "../core/ref" | ||
|  | 
 | ||
|  | const def: CodeKeywordDefinition = { | ||
|  |   keyword: "$dynamicRef", | ||
|  |   schemaType: "string", | ||
|  |   code: (cxt) => dynamicRef(cxt, cxt.schema), | ||
|  | } | ||
|  | 
 | ||
|  | export function dynamicRef(cxt: KeywordCxt, ref: string): void { | ||
|  |   const {gen, keyword, it} = cxt | ||
|  |   if (ref[0] !== "#") throw new Error(`"${keyword}" only supports hash fragment reference`) | ||
|  |   const anchor = ref.slice(1) | ||
|  |   if (it.allErrors) { | ||
|  |     _dynamicRef() | ||
|  |   } else { | ||
|  |     const valid = gen.let("valid", false) | ||
|  |     _dynamicRef(valid) | ||
|  |     cxt.ok(valid) | ||
|  |   } | ||
|  | 
 | ||
|  |   function _dynamicRef(valid?: Name): void { | ||
|  |     // TODO the assumption here is that `recursiveRef: #` always points to the root
 | ||
|  |     // of the schema object, which is not correct, because there may be $id that
 | ||
|  |     // makes # point to it, and the target schema may not contain dynamic/recursiveAnchor.
 | ||
|  |     // Because of that 2 tests in recursiveRef.json fail.
 | ||
|  |     // This is a similar problem to #815 (`$id` doesn't alter resolution scope for `{ "$ref": "#" }`).
 | ||
|  |     // (This problem is not tested in JSON-Schema-Test-Suite)
 | ||
|  |     if (it.schemaEnv.root.dynamicAnchors[anchor]) { | ||
|  |       const v = gen.let("_v", _`${N.dynamicAnchors}${getProperty(anchor)}`) | ||
|  |       gen.if(v, _callRef(v, valid), _callRef(it.validateName, valid)) | ||
|  |     } else { | ||
|  |       _callRef(it.validateName, valid)() | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   function _callRef(validate: Code, valid?: Name): () => void { | ||
|  |     return valid | ||
|  |       ? () => | ||
|  |           gen.block(() => { | ||
|  |             callRef(cxt, validate) | ||
|  |             gen.let(valid, true) | ||
|  |           }) | ||
|  |       : () => callRef(cxt, validate) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | export default def |