Skip to content

Commit da7f368

Browse files
authored
Handle generic validation error (#103)
Currently we don't return validation errors when we don't have valid corresponding pointers. However we wouldn't always have valid pointers e.g. if the data isn't a valid JSON object (see #98) Here I just `pushRoot` in those cases. We can consider adding the default root `#` pointer but the effect will be the same (i.e. by default all entries will _at least_ match the root pointer and so will set the range to the root)
1 parent 5e95143 commit da7f368

File tree

3 files changed

+23
-9
lines changed

3 files changed

+23
-9
lines changed

.changeset/weak-bobcats-scream.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"codemirror-json-schema": patch
3+
---
4+
5+
Handle generic validation error

src/index.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,19 @@
1-
export { jsonCompletion } from "./json-completion.js";
1+
export {
2+
jsonCompletion,
3+
JSONCompletion,
4+
type JSONCompletionOptions,
5+
} from "./json-completion.js";
26

37
export {
48
jsonSchemaLinter,
9+
JSONValidation,
510
type JSONValidationOptions,
611
handleRefresh,
712
} from "./json-validation.js";
813

914
export {
1015
jsonSchemaHover,
16+
JSONHover,
1117
type HoverOptions,
1218
type FoundCursorData,
1319
type CursorData,

src/json-validation.ts

+11-8
Original file line numberDiff line numberDiff line change
@@ -4,14 +4,14 @@ import { Draft04, type Draft, type JsonError } from "json-schema-library";
44

55
import { getJSONSchema, schemaStateField } from "./state.js";
66
import { joinWithOr } from "./utils/formatting.js";
7-
import { JSONMode, JSONPointerData } from "./types.js";
7+
import { JSONMode, JSONPointerData, RequiredPick } from "./types.js";
88
import { parseJSONDocumentState } from "./utils/parseJSONDocument.js";
9-
import { RequiredPick } from "./types.js";
109
import { el } from "./utils/dom.js";
1110
import { renderMarkdown } from "./utils/markdown.js";
1211
import { MODES } from "./constants.js";
1312
import { parseYAMLDocumentState } from "./utils/parse-yaml-document.js";
1413
import { parseJSON5DocumentState } from "./utils/parseJSON5Document.js";
14+
import { debug } from "./utils/debug.js";
1515

1616
const getDefaultParser = (mode: JSONMode): typeof parseJSONDocumentState => {
1717
switch (mode) {
@@ -96,7 +96,7 @@ export class JSONValidation {
9696
const errors = errorData?.errors as string[];
9797
if (error.code === "one-of-error" && errors?.length) {
9898
return `Expected one of ${joinWithOr(
99-
errors as string[],
99+
errors,
100100
(data) => data.data.expected
101101
)}`;
102102
}
@@ -110,6 +110,7 @@ export class JSONValidation {
110110
const message = error.message
111111
// don't mention root object
112112
.replaceAll("in `#` ", "")
113+
.replaceAll("at `#`", "")
113114
.replaceAll("/", ".")
114115
.replaceAll("#.", "");
115116
return message;
@@ -135,9 +136,10 @@ export class JSONValidation {
135136
try {
136137
errors = this.schema.validate(json.data);
137138
} catch {}
139+
debug.log("xxx", "validation errors", errors, json.data);
138140
if (!errors.length) return [];
139141
// reduce() because we want to filter out errors that don't have a pointer
140-
return errors.reduce((acc, error) => {
142+
return errors.reduce<Diagnostic[]>((acc, error) => {
141143
const pushRoot = () => {
142144
const errorString = this.rewriteError(error);
143145
acc.push({
@@ -156,12 +158,11 @@ export class JSONValidation {
156158
const errorPath = getErrorPath(error);
157159
const pointer = json.pointers.get(errorPath) as JSONPointerData;
158160
if (
159-
error.name === "MaxPropertiesError" ??
161+
error.name === "MaxPropertiesError" ||
160162
error.name === "MinPropertiesError"
161163
) {
162164
pushRoot();
163-
}
164-
if (pointer) {
165+
} else if (pointer) {
165166
// if the error is a property error, use the key position
166167
const isKeyError = positionalErrors.includes(error.name);
167168
const errorString = this.rewriteError(error);
@@ -182,8 +183,10 @@ export class JSONValidation {
182183
source: this.schemaTitle,
183184
});
184185
}
186+
} else {
187+
pushRoot();
185188
}
186189
return acc;
187-
}, [] as Diagnostic[]);
190+
}, []);
188191
}
189192
}

0 commit comments

Comments
 (0)