Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: ajv-validator/ajv
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v8.10.0
Choose a base ref
...
head repository: ajv-validator/ajv
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v8.11.0
Choose a head ref
  • 7 commits
  • 8 files changed
  • 7 contributors

Commits on Feb 24, 2022

  1. doc: add uriResolver (#1896)

    * doc: add uriResolver
    
    * update link
    
    Co-authored-by: Evgeny Poberezkin <2769109+epoberezkin@users.noreply.github.com>
    zekth and epoberezkin authored Feb 24, 2022
    Copy the full SHA
    6e53e43 View commit details
  2. Use root schemaEnv when resolving references in oneOf (#1901)

    * Use root schemaEnv when resolving references in oneOf
    
    * Update discriminator test name
    asprouse authored Feb 24, 2022
    Copy the full SHA
    95b15b6 View commit details

Commits on Mar 22, 2022

  1. Fix KeywordCxt links (#1931)

    kapouer authored Mar 22, 2022
    Copy the full SHA
    bd7cf15 View commit details
  2. fix: emit equal when needed - alternative to #1853 (#1922)

    * fix: emit equal when needed - alternative to #1853
    
    * fix: prettier errors
    
    * fix: tslint errors
    
    * Update lib/vocabularies/validation/enum.ts
    
    * Update lib/vocabularies/validation/enum.ts
    
    Co-authored-by: Evgeny Poberezkin <2769109+epoberezkin@users.noreply.github.com>
    bhvngt and epoberezkin authored Mar 22, 2022
    Copy the full SHA
    09f67f2 View commit details
  3. Copy the full SHA
    3ec588b View commit details
  4. Update limitProperties.ts (#1918)

    * Update limitProperties.ts
    
    in absence of condition pluralization of message, at least have item(s)
    
    * Update lib/vocabularies/validation/limitProperties.ts
    
    Co-authored-by: Evgeny Poberezkin <2769109+epoberezkin@users.noreply.github.com>
    TIBCOeddie and epoberezkin authored Mar 22, 2022
    Copy the full SHA
    a5119ef View commit details
  5. 8.11.0

    epoberezkin committed Mar 22, 2022
    1
    Copy the full SHA
    c067d6d View commit details
4 changes: 2 additions & 2 deletions docs/guide/formats.md
Original file line number Diff line number Diff line change
@@ -30,7 +30,7 @@ addFormats(ajv)

See [ajv-formats](https://github.com/ajv-validator/ajv-formats) documentation for further details.

It is recommended NOT to use "format" keyword implementations with untrusted data, as they may use potentially unsafe regular expressions (even though known issues are fixed) - see [ReDoS attack](./security.md#redos-attack).
It is recommended NOT to use "format" keyword implementations with untrusted data, as they may use potentially unsafe regular expressions (even though known issues are fixed) - see [ReDoS attack](../security.md#redos-attack).

::: danger Format validation of untrusted data
If you need to use "format" keyword to validate untrusted data, you MUST assess their suitability and safety for your validation scenarios.
@@ -61,7 +61,7 @@ JSON Schema draft-07 also defines formats `iri`, `iri-reference`, `idn-hostname`

## User-defined formats

You can add and replace any formats using [addFormat](./api.md#api-addformat) method:
You can add and replace any formats using [addFormat](../api.md#ajv-addformat-name-string-format-format-ajv) method:

```javascript
ajv.addFormat("identifier", /^a-z\$_[a-zA-Z$_0-9]*$/)
6 changes: 3 additions & 3 deletions docs/keywords.md
Original file line number Diff line number Diff line change
@@ -19,7 +19,7 @@ interface _KeywordDef {
}
```

Keyword definitions may have additional optional properties - see [types](https://github.com/ajv-validator/ajv/blob/master/lib/types/index.ts) and [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/context.ts).
Keyword definitions may have additional optional properties - see [types](https://github.com/ajv-validator/ajv/blob/master/lib/types/index.ts) and [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/validate/index.ts).

### Define keyword with code generation function <Badge text="recommended" />

@@ -90,7 +90,7 @@ ajv.addKeyword({

You can review pre-defined Ajv keywords in [validation](https://github.com/ajv-validator/ajv/blob/master/lib/validation) folder for more advanced examples - it is much easier to define code generation keywords than it was in the previous version of Ajv.

See [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/context.ts) and [SchemaCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/index.ts) type definitions for more information about properties you can use in your keywords.
See [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/validate/index.ts) and [SchemaCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/index.ts) type definitions for more information about properties you can use in your keywords.

### Define keyword with "validate" function

@@ -243,7 +243,7 @@ Macro keywords an be recursive - i.e. return schemas containing the same keyword

## Schema compilation context

Schema compilation context [SchemaCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/index.ts) is available in property `it` of [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/context.ts) (and it is also the 3rd parameter of `compile` and `macro` keyword functions). See types in the source code on the properties you can use in this object.
Schema compilation context [SchemaCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/index.ts) is available in property `it` of [KeywordCxt](https://github.com/ajv-validator/ajv/blob/master/lib/compile/validate/index.ts) (and it is also the 3rd parameter of `compile` and `macro` keyword functions). See types in the source code on the properties you can use in this object.

## Validation time variables

5 changes: 5 additions & 0 deletions docs/options.md
Original file line number Diff line number Diff line change
@@ -62,6 +62,7 @@ const defaultOptions = {
ownProperties: false,
multipleOfPrecision: undefined, // *
messages: true, // false with JTD
uriResolver: undefined,
code: {
// NEW
es5: false,
@@ -341,6 +342,10 @@ By default `multipleOf` keyword is validated by comparing the result of division

Include human-readable messages in errors. `true` by default. `false` can be passed when messages are generated outside of Ajv code (e.g. with [ajv-i18n](https://github.com/ajv-validator/ajv-i18n)).

### uriResolver

By default `uriResolver` is undefined and relies on the embedded uriResolver [uri-js](https://github.com/garycourt/uri-js). Pass an object that satisfies the interface [UriResolver](https://github.com/ajv-validator/ajv/blob/master/lib/types/index.ts) to be used in replacement. One alternative is [fast-uri](https://github.com/fastify/fast-uri).

### code <Badge text="v7" />

Code generation options:
2 changes: 1 addition & 1 deletion lib/vocabularies/discriminator/index.ts
Original file line number Diff line number Diff line change
@@ -66,7 +66,7 @@ const def: CodeKeywordDefinition = {
for (let i = 0; i < oneOf.length; i++) {
let sch = oneOf[i]
if (sch?.$ref && !schemaHasRulesButRef(sch, it.self.RULES)) {
sch = resolveRef.call(it.self, it.schemaEnv, it.baseId, sch?.$ref)
sch = resolveRef.call(it.self, it.schemaEnv.root, it.baseId, sch?.$ref)
if (sch instanceof SchemaEnv) sch = sch.schema
}
const propSch = sch?.properties?.[tagName]
8 changes: 5 additions & 3 deletions lib/vocabularies/validation/enum.ts
Original file line number Diff line number Diff line change
@@ -20,7 +20,9 @@ const def: CodeKeywordDefinition = {
const {gen, data, $data, schema, schemaCode, it} = cxt
if (!$data && schema.length === 0) throw new Error("enum must have non-empty array")
const useLoop = schema.length >= it.opts.loopEnum
const eql = useFunc(gen, equal)
let eql: Name | undefined
const getEql = (): Name => (eql ??= useFunc(gen, equal))

let valid: Code
if (useLoop || $data) {
valid = gen.let("valid")
@@ -36,14 +38,14 @@ const def: CodeKeywordDefinition = {
function loopEnum(): void {
gen.assign(valid, false)
gen.forOf("v", schemaCode as Code, (v) =>
gen.if(_`${eql}(${data}, ${v})`, () => gen.assign(valid, true).break())
gen.if(_`${getEql()}(${data}, ${v})`, () => gen.assign(valid, true).break())
)
}

function equalCode(vSchema: Name, i: number): Code {
const sch = schema[i]
return typeof sch === "object" && sch !== null
? _`${eql}(${data}, ${vSchema}[${i}])`
? _`${getEql()}(${data}, ${vSchema}[${i}])`
: _`${data} === ${sch}`
}
},
2 changes: 1 addition & 1 deletion lib/vocabularies/validation/limitProperties.ts
Original file line number Diff line number Diff line change
@@ -5,7 +5,7 @@ import {_, str, operators} from "../../compile/codegen"
const error: KeywordErrorDefinition = {
message({keyword, schemaCode}) {
const comp = keyword === "maxProperties" ? "more" : "fewer"
return str`must NOT have ${comp} than ${schemaCode} items`
return str`must NOT have ${comp} than ${schemaCode} properties`
},
params: ({schemaCode}) => _`{limit: ${schemaCode}}`,
}
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "ajv",
"version": "8.10.0",
"version": "8.11.0",
"description": "Another JSON Schema Validator",
"main": "dist/ajv.js",
"types": "dist/ajv.d.ts",
82 changes: 82 additions & 0 deletions spec/discriminator.spec.ts
Original file line number Diff line number Diff line change
@@ -159,6 +159,88 @@ describe("discriminator keyword", function () {
})
})

describe("validation with deeply referenced schemas", () => {
const schema = [
{
type: "object",
properties: {
container: {
$ref: "#/definitions/Container",
},
},
definitions: {
BlockA: {
type: "object",
properties: {
_type: {
type: "string",
enum: ["a"],
},
a: {type: "string"},
},
additionalProperties: false,
required: ["_type"],
title: "BlockA",
},
BlockB: {
type: "object",
properties: {
_type: {
type: "string",
enum: ["b"],
},
b: {type: "string"},
},
additionalProperties: false,
required: ["_type"],
title: "BlockB",
},
Container: {
type: "object",
properties: {
list: {
type: "array",
items: {
oneOf: [
{
$ref: "#/definitions/BlockA",
},
{
$ref: "#/definitions/BlockB",
},
],
discriminator: {
propertyName: "_type",
},
},
},
},
},
},
},
]

it("should validate data", () => {
assertValid(schema, {
container: {
list: [
{_type: "a", a: "foo"},
{_type: "b", b: "bar"},
],
},
})

assertInvalid(schema, {
container: {
list: [
{_type: "a", b: "foo"},
{_type: "b", b: "bar"},
],
},
})
})
})

describe("valid schemas", () => {
it("should have oneOf", () => {
invalidSchema(