Skip to content

Commit

Permalink
Merge branch 'fix/track-pattern-properties-evaluation' of https://git…
Browse files Browse the repository at this point in the history
…hub.com/stoplightio/ajv into stoplightio-fix/track-pattern-properties-evaluation
  • Loading branch information
epoberezkin committed Jun 6, 2021
2 parents 85ae956 + 73bf4b4 commit 3e285b0
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 15 deletions.
42 changes: 27 additions & 15 deletions lib/vocabularies/applicator/patternProperties.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
import type {CodeKeywordDefinition} from "../../types"
import type {KeywordCxt} from "../../compile/validate"
import {schemaProperties, usePattern} from "../code"
import {allSchemaProperties, usePattern} from "../code"
import {_, not, Name} from "../../compile/codegen"
import {checkStrictMode} from "../../compile/util"
import {alwaysValidSchema, checkStrictMode} from "../../compile/util"
import {evaluatedPropsToName, Type} from "../../compile/util"
import {AnySchema} from "../../types"

const def: CodeKeywordDefinition = {
keyword: "patternProperties",
Expand All @@ -12,9 +13,16 @@ const def: CodeKeywordDefinition = {
code(cxt: KeywordCxt) {
const {gen, schema, data, parentSchema, it} = cxt
const {opts} = it
const patterns = schemaProperties(it, schema)
// TODO mark properties matching patterns with always valid schemas as evaluated
if (patterns.length === 0) return
const patterns = allSchemaProperties(schema)
const validPatterns = patterns.filter((p) => !alwaysValidSchema(it, schema[p] as AnySchema))
const checkForUnevaluatedProperties =
parentSchema.unevaluatedProperties !== true &&
parentSchema.unevaluatedProperties !== undefined

if (patterns.length === 0 && (validPatterns.length === 0 || !checkForUnevaluatedProperties)) {
return
}

const checkProperties =
opts.strictSchema && !opts.allowMatchingProperties && parentSchema.properties
const valid = gen.name("valid")
Expand Down Expand Up @@ -51,18 +59,22 @@ const def: CodeKeywordDefinition = {
function validateProperties(pat: string): void {
gen.forIn("key", data, (key) => {
gen.if(_`${usePattern(cxt, pat)}.test(${key})`, () => {
cxt.subschema(
{
keyword: "patternProperties",
schemaProp: pat,
dataProp: key,
dataPropType: Type.Str,
},
valid
)
const alwaysValid = !validPatterns.includes(pat)
if (!alwaysValid) {
cxt.subschema(
{
keyword: "patternProperties",
schemaProp: pat,
dataProp: key,
dataPropType: Type.Str,
},
valid
)
}

if (it.opts.unevaluated && props !== true) {
gen.assign(_`${props}[${key}]`, true)
} else if (!it.allErrors) {
} else if (!alwaysValid && !it.allErrors) {
// can short-circuit if `unevaluatedProperties` is not supported (opts.next === false)
// or if all properties were evaluated (props === true)
gen.if(not(valid), () => gen.break())
Expand Down
20 changes: 20 additions & 0 deletions spec/issues/1625_evaluated_truthy_pattern_properties.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import _Ajv from "../ajv2020"
import * as assert from "assert"

describe("tracking evaluated properties with pattern properties of schema = true", () => {
it("should initialize evaluated properties", () => {
const ajv = new _Ajv()

const schema = {
type: "object",
patternProperties: {
"^x-": true,
},
unevaluatedProperties: false,
}

const validate = ajv.compile(schema)
assert.strictEqual(validate({bar: 1}), false)
assert.strictEqual(validate({"x-bar": false}), true)
})
})

0 comments on commit 3e285b0

Please sign in to comment.