Skip to content

Commit 32d1cab

Browse files
committed
feat: Accept multiple schemas if external definitions are used
* A schema can be a string|object or an array pf them. * The main schema, if there is such, should be passed as the first one to be immediately compiled. * Fix multivalue argument parsing to accept both comma-delimited items and multiple arguments. Unfortunately, data arguments have to be separated by "--". * Accept schemas either as strings or as already parsed objects. * Update the readme with the extended JSON Schema support.
1 parent d92083d commit 32d1cab

15 files changed

+280
-125
lines changed

README.md

+34-31
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,15 @@ This is a fork of the original project ([zaach/jsonlint](https://github.com/zaac
1616
* Optionally recognizes JavaScript-style comments (CJSON) and single quoted strings (JSON5).
1717
* Optionally ignores trailing commas and reports duplicate object keys as an error.
1818
* Optionally checks that also the expected format matches, including sorted object keys.
19-
* Supports [JSON Schema] drafts 04, 06 and 07.
19+
* Supports [JSON Schema] drafts 04, 06, 07, 2019-09 and 2020-12.
20+
* Supports [JSON Type Definition].
2021
* Offers pretty-printing including comment-stripping and object keys without quotes (JSON5).
21-
* Prefers the native JSON parser if possible to run [7x faster than the custom parser].
22+
* Prefers the native JSON parser if possible to run [10x faster than the custom parser].
2223
* Reports errors with rich additional information. From the JSON Schema validation too.
2324
* Consumes configuration from both command line and [configuration files](configuration).
2425
* Implements JavaScript modules using [UMD] to work in Node.js, in a browser, everywhere.
2526
* Depends on up-to-date npm modules with no installation warnings.
26-
* Small size - 18.7 kB minified, 6.54 kB gzipped, 5.16 kB brotlied.
27+
* Small size - 18.4 kB minified, 6.45 kB gzipped, 5.05 kB brotlied.
2728

2829
**Note:** In comparison with the original project, this package exports only the `parse` method; not the `Parser` object.
2930

@@ -113,50 +114,42 @@ The input can be checked not only to be a valid JSON, but also to be formatted a
113114

114115
### Usage
115116

116-
Usage: `jsonlint [options] [<file, directory, pattern> ...]`
117+
Usage: `jsonlint [options] [--] [<file, directory, pattern> ...]`
117118

118119
#### Options
119120

120-
-f, --config [file] read options from a custom configuration file
121-
-F, --no-config disable searching for configuration file
121+
-f, --config <file> read options from a custom configuration file
122+
-F, --no-config disable searching for configuration files
122123
-s, --sort-keys sort object keys (not when prettifying)
123-
-E, --extensions [ext] file extensions to process for directory walk
124-
(default: ["json","JSON"])
124+
-E, --extensions <ext...> file extensions to process for directory walk (default: ["json","JSON"])
125125
-i, --in-place overwrite the input files
126126
-j, --diff print difference instead of writing the output
127127
-k, --check check that the input is equal to the output
128-
-t, --indent [num|char] number of spaces or specific characters
129-
to use for indentation (default: 2)
128+
-t, --indent <num|char> number of spaces or specific characters to use for indentation (default: 2)
130129
-c, --compact compact error display
131-
-M, --mode [mode] set other parsing flags according to a format
132-
type (default: "json")
130+
-M, --mode <mode> set other parsing flags according to a format type (default: "json")
133131
-B, --bom ignore the leading UTF-8 byte-order mark
134132
-C, --comments recognize and ignore JavaScript-style comments
135133
-S, --single-quoted-strings support single quotes as string delimiters
136134
-T, --trailing-commas ignore trailing commas in objects and arrays
137135
-D, --no-duplicate-keys report duplicate object keys as an error
138-
-V, --validate [file] JSON Schema file to use for validation
139-
-e, --environment [env] which specification of JSON Schema the
140-
validation file uses
141-
-x, --context [num] line count used as the diff context (default: 3)
136+
-V, --validate <file...> JSON Schema file(s) to use for validation (default: [])
137+
-e, --environment <env> which specification of JSON Schema the validation file uses
138+
-x, --context <num> line count used as the diff context (default: 3)
142139
-l, --log-files print only the parsed file names to stdout
143140
-q, --quiet do not print the parsed json to stdout
144141
-n, --continue continue with other files if an error occurs
145-
-p, --pretty-print prettify the input instead of stringifying
146-
the parsed object
142+
-p, --pretty-print prettify the input instead of stringifying the parsed object
147143
-P, --pretty-print-invalid force pretty-printing even for invalid input
148144
-r, --trailing-newline ensure a line break at the end of the output
149145
-R, --no-trailing-newline ensure no line break at the end of the output
150146
--prune-comments omit comments from the prettified output
151-
--strip-object-keys strip quotes from object keys if possible
152-
(JSON5)
147+
--strip-object-keys strip quotes from object keys if possible (JSON5)
153148
--enforce-double-quotes surrounds all strings with double quotes
154-
--enforce-single-quotes surrounds all strings with single quotes
155-
(JSON5)
156-
--trim-trailing-commas omit trailing commas from objects and arrays
157-
(JSON5)
149+
--enforce-single-quotes surrounds all strings with single quotes (JSON5)
150+
--trim-trailing-commas omit trailing commas from objects and arrays (JSON5)
158151
-v, --version output the version number
159-
-h, --help output usage information
152+
-h, --help display help for command
160153

161154
You can use BASH patterns for including and excluding files (only files).
162155
Patterns are case-sensitive and have to use slashes as directory separators.
@@ -169,6 +162,9 @@ for JSON Schema validation are "draft-04", "draft-06", "draft-07",
169162
with "json-schema-". JSON Type Definition can be selected by "rfc8927",
170163
"json-type-definition" or "jtd". If not specified, it will be "draft-07".
171164

165+
If you specify schemas using the "-V" parameter, you will have to separate
166+
files to test with "--".
167+
172168
### Configuration
173169

174170
In addition to the command line parameters, the options can be supplied from the following files:
@@ -274,7 +270,7 @@ The `mode` parameter (string) sets parsing options to match a common format of i
274270

275271
### Schema Validation
276272

277-
You can validate the input against a JSON Schema using the `lib/validator` module. The `validate` method accepts either an earlier parsed JSON data or a string with the JSON input:
273+
You can validate the input against a JSON Schema using the `lib/validator` module. The `compile` method accepts either an earlier parsed JSON Schema or a string with it:
278274

279275
```js
280276
const { compile } = require('@prantlf/jsonlint/lib/validator')
@@ -283,12 +279,18 @@ const validate = compile('string with JSON Schema')
283279
const parsed = validate('string with JSON data')
284280
```
285281

286-
If a string is passed to the `validate` method, the same options as for parsing JSON data can be passed as the second parameter. Compiling JSON Schema supports the same options as parsing JSON data too (except for `reviver`). They can be passed as the second (object) parameter. The optional second `environment` parameter can be passed either as a string or as an additional property in the options object too:
282+
If a string is passed to the `compile` method, the same options as for parsing JSON data can be passed as the second parameter. Compiling JSON Schema supports the same options as parsing JSON data too (except for `reviver`). They can be passed as the second (object) parameter. The optional second `environment` parameter (the default value is `draft-07`) ) can be passed either as a string or as an additional property in the options object too:
287283

288284
```js
289285
const validate = compile('string with JSON Schema', { environment: 'draft-2020-12' })
290286
```
291287

288+
If you use external definitions in multiple schemas, you have to pass an array of all schemas to `compile`. The `$id` properties have to be set in each sub-schema according to the `$ref` references in the main schema. The main schema is usually sent as the first one to be compiled immediately, so that the errors in any sub-schema would be reported right away:
289+
290+
```js
291+
const validate = compile(['string with main schema', 'string with a sub-schema'])
292+
```
293+
292294
### Pretty-Printing
293295

294296
You can parse a JSON string to an array of tokens and print it back to a string with some changes applied. It can be unification of whitespace, reformatting or stripping comments, for example. (Raw token values must be enabled when tokenizing the JSON input.)
@@ -372,11 +374,11 @@ If you want to retain comments or whitespace for pretty-printing, for example, s
372374

373375
### Performance
374376

375-
This is a part of an output from the [parser benchmark], when parsing a 4.2 KB formatted string ([package.json](./package.json)) with Node.js 12.14.0:
377+
This is a part of an output from the [parser benchmark], when parsing a 4.68 KB formatted string ([package.json](./package.json)) with Node.js 18.14.2:
376378

377-
jsonlint using native JSON.parse x 97,109 ops/sec ±0.81% (93 runs sampled)
378-
jsonlint using hand-coded parser x 7,256 ops/sec ±0.54% (90 runs sampled)
379-
jsonlint using tokenising parser x 6,387 ops/sec ±0.44% (88 runs sampled)
379+
the standard jsonlint parser x 78,998 ops/sec ±0.48% (95 runs sampled)
380+
the extended jsonlint parser x 7,923 ops/sec ±0.51% (93 runs sampled)
381+
the tokenising jsonlint parser x 6,281 ops/sec ±0.71% (91 runs sampled)
380382

381383
A custom JSON parser is [a lot slower] than the built-in one. However, it is more important to have a [clear error reporting] than the highest speed in scenarios like parsing configuration files. (For better error-reporting, the speed can be preserved by using the native parser initially and re-parsing with another parser only in case of failure.) Features like comments or JSON5 are also helpful in configuration files. Tokens preserve the complete input and can be used for pretty-printing without losing the comments.
382384

@@ -429,6 +431,7 @@ Licensed under the [MIT License].
429431
[JSON]: https://tools.ietf.org/html/rfc8259
430432
[JSON5]: https://spec.json5.org
431433
[JSON Schema]: https://json-schema.org
434+
[JSON Type Definition]: https://jsontypedef.com/
432435
[UMD]: https://github.com/umdjs/umd
433436
[`Grunt`]: https://gruntjs.com/
434437
[`Gulp`]: http://gulpjs.com/

lib/cli.js

+22-16
Original file line numberDiff line numberDiff line change
@@ -10,30 +10,30 @@ const { sortObject } = require('./sorter')
1010
const { compile } = require('./validator')
1111
const { description, version } = require('../package')
1212

13-
const collectValues = extension => extension.split(',')
13+
const collectValues = (input, result) => result.concat(input.split(','))
1414

1515
const commander = require('commander')
1616
.name('jsonlint')
1717
.usage('[options] [<file, directory, pattern> ...]')
1818
.description(description)
19-
.option('-f, --config [file]', 'read options from a custom configuration file')
19+
.option('-f, --config <file>', 'read options from a custom configuration file')
2020
.option('-F, --no-config', 'disable searching for configuration files')
2121
.option('-s, --sort-keys', 'sort object keys (not when prettifying)')
22-
.option('-E, --extensions [ext]', 'file extensions to process for directory walk', collectValues, ['json', 'JSON'])
22+
.option('-E, --extensions <ext...>', 'file extensions to process for directory walk', collectValues, ['json', 'JSON'])
2323
.option('-i, --in-place', 'overwrite the input files')
2424
.option('-j, --diff', 'print difference instead of writing the output')
2525
.option('-k, --check', 'check that the input is equal to the output')
26-
.option('-t, --indent [num|char]', 'number of spaces or specific characters to use for indentation', 2)
26+
.option('-t, --indent <num|char>', 'number of spaces or specific characters to use for indentation', 2)
2727
.option('-c, --compact', 'compact error display')
28-
.option('-M, --mode [mode]', 'set other parsing flags according to a format type', 'json')
28+
.option('-M, --mode <mode>', 'set other parsing flags according to a format type', 'json')
2929
.option('-B, --bom', 'ignore the leading UTF-8 byte-order mark')
3030
.option('-C, --comments', 'recognize and ignore JavaScript-style comments')
3131
.option('-S, --single-quoted-strings', 'support single quotes as string delimiters')
3232
.option('-T, --trailing-commas', 'ignore trailing commas in objects and arrays')
3333
.option('-D, --no-duplicate-keys', 'report duplicate object keys as an error')
34-
.option('-V, --validate [file]', 'JSON Schema file to use for validation')
35-
.option('-e, --environment [env]', 'which specification of JSON Schema the validation file uses')
36-
.option('-x, --context [num]', 'line count used as the diff context', 3)
34+
.option('-V, --validate <file...>', 'JSON Schema file(s) to use for validation', collectValues, [])
35+
.option('-e, --environment <env>', 'which specification of JSON Schema the validation file uses')
36+
.option('-x, --context <num>', 'line count used as the diff context', 3)
3737
.option('-l, --log-files', 'print only the parsed file names to stdout')
3838
.option('-q, --quiet', 'do not print the parsed json to stdout')
3939
.option('-n, --continue', 'continue with other files if an error occurs')
@@ -59,6 +59,9 @@ const commander = require('commander')
5959
console.log('"draft-2019-09" or "draft-2020-12". The environment may be prefixed')
6060
console.log('with "json-schema-". JSON Type Definition can be selected by "rfc8927",')
6161
console.log('"json-type-definition" or "jtd". If not specified, it will be "draft-07".')
62+
console.log()
63+
console.log('If you specify schemas using the "-V" parameter, you will have to separate')
64+
console.log('files to test with "--".')
6265
})
6366
.parse(process.argv)
6467

@@ -144,16 +147,19 @@ function processContents (source, file) {
144147
allowSingleQuotedStrings: options.singleQuotedStrings,
145148
allowDuplicateObjectKeys: options.duplicateKeys
146149
}
147-
if (options.validate) {
148-
let validate
150+
if (options.validate.length) {
151+
const schemas = options.validate.map((file, index) => {
152+
try {
153+
return readFileSync(file, 'utf8')
154+
} catch (error) {
155+
throw new Error(`Loading the JSON Schema #${index + 1} failed: "${file}".\n${error.message}`)
156+
}
157+
})
158+
parserOptions.environment = options.environment
149159
try {
150-
const schema = readFileSync(normalize(options.validate), 'utf8')
151-
parserOptions.environment = options.environment
152-
validate = compile(schema, parserOptions)
160+
validate = compile(schemas, parserOptions)
153161
} catch (error) {
154-
const message = 'Loading the JSON Schema failed: "' +
155-
options.validate + '".\n' + error.message
156-
throw new Error(message)
162+
throw new Error(`Loading the JSON Schema failed:\n${error.message}`)
157163
}
158164
parsed = validate(source, parserOptions)
159165
} else {

lib/index.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -458,7 +458,8 @@ declare module '@prantlf/jsonlint/lib/validator' {
458458
* with multiple options
459459
* @returns the validator function
460460
*/
461-
function compile (schema: string, environmentOrOptions?: Environment | CompileOptions): Validator
461+
function compile (schema: string | string[] | Record<string, unknown> | Record<string, unknown>[],
462+
environmentOrOptions?: Environment | CompileOptions): Validator
462463
}
463464

464465
declare module '@prantlf/jsonlint/lib/printer' {

lib/validator.js

+16-11
Original file line numberDiff line numberDiff line change
@@ -123,28 +123,33 @@
123123
const Ajv = requireAjv('AjvJTD')
124124
ajv = new Ajv()
125125
} else {
126-
throw new RangeError('Unsupported environment for the JSON Schema validation: "' +
127-
environment + '".')
126+
throw new RangeError(`Unsupported environment for the JSON Schema validation: "${environment}".`)
128127
}
129128
return ajv
130129
}
131130

132131
function compileSchema (ajv, schema, parseOptions) {
133-
let parsed
134-
try {
135-
parsed = jsonlint.parse(schema, parseOptions)
136-
} catch (error) {
137-
error.message = 'Parsing the JSON Schema failed.\n' + error.message
138-
throw error
139-
}
132+
if (!Array.isArray(schema)) schema = [schema]
133+
const [main, ...others] = schema.map((schema, index) => {
134+
if (typeof schema !== 'string') return schema
135+
try {
136+
return jsonlint.parse(schema, parseOptions)
137+
} catch (error) {
138+
error.message = `Parsing the JSON Schema #${index + 1} failed.\n${error.message}`
139+
throw error
140+
}
141+
})
140142
try {
141-
return ajv.compile(parsed)
143+
for (const schema of others) {
144+
ajv.addSchema(schema)
145+
}
146+
return ajv.compile(main)
142147
} catch (originalError) {
143148
const errors = ajv.errors
144149
const betterError = errors
145150
? createError(errors, parsed, schema, parseOptions)
146151
: originalError
147-
betterError.message = 'Compiling the JSON Schema failed.\n' + betterError.message
152+
betterError.message = `Compiling the JSON Schema failed.\n${betterError.message}`
148153
throw betterError
149154
}
150155
}

0 commit comments

Comments
 (0)