frictionless-ts 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +9 -0
- package/README.md +3 -0
- package/build/dialect/index.d.ts +1 -0
- package/build/dialect/index.js +2 -0
- package/build/dialect/infer.d.ts +3 -0
- package/build/dialect/infer.js +13 -0
- package/build/dialect/infer.spec.d.ts +1 -0
- package/build/dialect/infer.spec.js +89 -0
- package/build/index.d.ts +19 -0
- package/build/index.js +19 -0
- package/build/package/index.d.ts +4 -0
- package/build/package/index.js +5 -0
- package/build/package/infer.d.ts +40 -0
- package/build/package/infer.js +13 -0
- package/build/package/infer.spec.d.ts +1 -0
- package/build/package/infer.spec.js +121 -0
- package/build/package/integrity.d.ts +8 -0
- package/build/package/integrity.js +69 -0
- package/build/package/integrity.spec.d.ts +1 -0
- package/build/package/integrity.spec.js +575 -0
- package/build/package/load.d.ts +1 -0
- package/build/package/load.js +10 -0
- package/build/package/load.spec.d.ts +1 -0
- package/build/package/load.spec.js +117 -0
- package/build/package/save.d.ts +5 -0
- package/build/package/save.js +13 -0
- package/build/package/save.spec.d.ts +1 -0
- package/build/package/save.spec.js +138 -0
- package/build/package/validate.d.ts +20 -0
- package/build/package/validate.js +62 -0
- package/build/package/validate.spec.d.ts +1 -0
- package/build/package/validate.spec.js +172 -0
- package/build/plugin.d.ts +2 -0
- package/build/plugin.js +2 -0
- package/build/resource/index.d.ts +2 -0
- package/build/resource/index.js +3 -0
- package/build/resource/infer.d.ts +22 -0
- package/build/resource/infer.js +49 -0
- package/build/resource/infer.spec.d.ts +1 -0
- package/build/resource/infer.spec.js +164 -0
- package/build/resource/validate.d.ts +12 -0
- package/build/resource/validate.js +35 -0
- package/build/resource/validate.spec.d.ts +1 -0
- package/build/resource/validate.spec.js +147 -0
- package/build/schema/index.d.ts +1 -0
- package/build/schema/index.js +2 -0
- package/build/schema/infer.d.ts +3 -0
- package/build/schema/infer.js +18 -0
- package/build/schema/infer.spec.d.ts +1 -0
- package/build/schema/infer.spec.js +136 -0
- package/build/system.d.ts +6 -0
- package/build/system.js +41 -0
- package/build/table/index.d.ts +3 -0
- package/build/table/index.js +4 -0
- package/build/table/infer.d.ts +6 -0
- package/build/table/infer.js +20 -0
- package/build/table/infer.spec.d.ts +1 -0
- package/build/table/infer.spec.js +177 -0
- package/build/table/load.d.ts +3 -0
- package/build/table/load.js +11 -0
- package/build/table/load.spec.d.ts +1 -0
- package/build/table/load.spec.js +121 -0
- package/build/table/save.d.ts +2 -0
- package/build/table/save.js +11 -0
- package/build/table/save.spec.d.ts +1 -0
- package/build/table/save.spec.js +189 -0
- package/build/table/validate.d.ts +9 -0
- package/build/table/validate.js +26 -0
- package/package.json +38 -0
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { system } from "../system.js";
|
|
2
|
+
export async function loadPackage(source) {
|
|
3
|
+
for (const plugin of system.plugins) {
|
|
4
|
+
const result = await plugin.loadPackage?.(source);
|
|
5
|
+
if (result)
|
|
6
|
+
return result;
|
|
7
|
+
}
|
|
8
|
+
throw new Error(`No plugin can load the package: ${source}`);
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3BhY2thZ2UvbG9hZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sY0FBYyxDQUFBO0FBRXJDLE1BQU0sQ0FBQyxLQUFLLFVBQVUsV0FBVyxDQUFDLE1BQWM7SUFDOUMsS0FBSyxNQUFNLE1BQU0sSUFBSSxNQUFNLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDcEMsTUFBTSxNQUFNLEdBQUcsTUFBTSxNQUFNLENBQUMsV0FBVyxFQUFFLENBQUMsTUFBTSxDQUFDLENBQUE7UUFDakQsSUFBSSxNQUFNO1lBQUUsT0FBTyxNQUFNLENBQUE7SUFDM0IsQ0FBQztJQUVELE1BQU0sSUFBSSxLQUFLLENBQUMsbUNBQW1DLE1BQU0sRUFBRSxDQUFDLENBQUE7QUFDOUQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHN5c3RlbSB9IGZyb20gXCIuLi9zeXN0ZW0udHNcIlxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbG9hZFBhY2thZ2Uoc291cmNlOiBzdHJpbmcpIHtcbiAgZm9yIChjb25zdCBwbHVnaW4gb2Ygc3lzdGVtLnBsdWdpbnMpIHtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBwbHVnaW4ubG9hZFBhY2thZ2U/Lihzb3VyY2UpXG4gICAgaWYgKHJlc3VsdCkgcmV0dXJuIHJlc3VsdFxuICB9XG5cbiAgdGhyb3cgbmV3IEVycm9yKGBObyBwbHVnaW4gY2FuIGxvYWQgdGhlIHBhY2thZ2U6ICR7c291cmNlfWApXG59XG4iXX0=
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { basename } from "node:path";
|
|
2
|
+
import { writeTempFile } from "@frictionless-ts/dataset";
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
import { loadPackage } from "./load.js";
|
|
5
|
+
describe("loadPackage", () => {
|
|
6
|
+
it("should load package from JSON file", async () => {
|
|
7
|
+
const packageContent = JSON.stringify({
|
|
8
|
+
name: "test-package",
|
|
9
|
+
resources: [
|
|
10
|
+
{
|
|
11
|
+
name: "test-resource",
|
|
12
|
+
data: [{ id: 1, name: "alice" }],
|
|
13
|
+
},
|
|
14
|
+
],
|
|
15
|
+
});
|
|
16
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
17
|
+
filename: "datapackage.json",
|
|
18
|
+
});
|
|
19
|
+
const dataPackage = await loadPackage(packagePath);
|
|
20
|
+
expect(dataPackage).toBeDefined();
|
|
21
|
+
expect(dataPackage.name).toBe("test-package");
|
|
22
|
+
expect(dataPackage.resources).toBeDefined();
|
|
23
|
+
expect(dataPackage.resources.length).toBe(1);
|
|
24
|
+
});
|
|
25
|
+
it("should load package with multiple resources", async () => {
|
|
26
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
27
|
+
const packageContent = JSON.stringify({
|
|
28
|
+
name: "test-package",
|
|
29
|
+
resources: [
|
|
30
|
+
{
|
|
31
|
+
name: "resource-1",
|
|
32
|
+
path: basename(csvPath),
|
|
33
|
+
format: "csv",
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
name: "resource-2",
|
|
37
|
+
data: [{ id: 1, value: 100 }],
|
|
38
|
+
},
|
|
39
|
+
],
|
|
40
|
+
});
|
|
41
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
42
|
+
filename: "datapackage.json",
|
|
43
|
+
});
|
|
44
|
+
const dataPackage = await loadPackage(packagePath);
|
|
45
|
+
expect(dataPackage).toBeDefined();
|
|
46
|
+
expect(dataPackage.name).toBe("test-package");
|
|
47
|
+
expect(dataPackage.resources.length).toBe(2);
|
|
48
|
+
});
|
|
49
|
+
it("should load package with metadata", async () => {
|
|
50
|
+
const packageContent = JSON.stringify({
|
|
51
|
+
name: "test-package",
|
|
52
|
+
title: "Test Package",
|
|
53
|
+
description: "A test data package",
|
|
54
|
+
version: "1.0.0",
|
|
55
|
+
resources: [
|
|
56
|
+
{
|
|
57
|
+
name: "test-resource",
|
|
58
|
+
data: [{ id: 1 }],
|
|
59
|
+
},
|
|
60
|
+
],
|
|
61
|
+
});
|
|
62
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
63
|
+
filename: "datapackage.json",
|
|
64
|
+
});
|
|
65
|
+
const dataPackage = await loadPackage(packagePath);
|
|
66
|
+
expect(dataPackage.name).toBe("test-package");
|
|
67
|
+
expect(dataPackage.title).toBe("Test Package");
|
|
68
|
+
expect(dataPackage.description).toBe("A test data package");
|
|
69
|
+
expect(dataPackage.version).toBe("1.0.0");
|
|
70
|
+
});
|
|
71
|
+
it("should load package with inline data resources", async () => {
|
|
72
|
+
const packageContent = JSON.stringify({
|
|
73
|
+
name: "test-package",
|
|
74
|
+
resources: [
|
|
75
|
+
{
|
|
76
|
+
name: "test-resource",
|
|
77
|
+
data: [
|
|
78
|
+
{ id: 1, name: "alice" },
|
|
79
|
+
{ id: 2, name: "bob" },
|
|
80
|
+
],
|
|
81
|
+
},
|
|
82
|
+
],
|
|
83
|
+
});
|
|
84
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
85
|
+
filename: "datapackage.json",
|
|
86
|
+
});
|
|
87
|
+
const dataPackage = await loadPackage(packagePath);
|
|
88
|
+
expect(dataPackage).toBeDefined();
|
|
89
|
+
expect(dataPackage.resources?.[0]?.data).toBeDefined();
|
|
90
|
+
});
|
|
91
|
+
it("should load package with schema", async () => {
|
|
92
|
+
const packageContent = JSON.stringify({
|
|
93
|
+
name: "test-package",
|
|
94
|
+
resources: [
|
|
95
|
+
{
|
|
96
|
+
name: "test-resource",
|
|
97
|
+
data: [{ id: 1, name: "alice" }],
|
|
98
|
+
schema: {
|
|
99
|
+
fields: [
|
|
100
|
+
{ name: "id", type: "integer" },
|
|
101
|
+
{ name: "name", type: "string" },
|
|
102
|
+
],
|
|
103
|
+
},
|
|
104
|
+
},
|
|
105
|
+
],
|
|
106
|
+
});
|
|
107
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
108
|
+
filename: "datapackage.json",
|
|
109
|
+
});
|
|
110
|
+
const dataPackage = await loadPackage(packagePath);
|
|
111
|
+
expect(dataPackage.resources?.[0]?.schema).toBeDefined();
|
|
112
|
+
expect(typeof dataPackage.resources?.[0]?.schema === "object" &&
|
|
113
|
+
"fields" in dataPackage.resources[0].schema &&
|
|
114
|
+
dataPackage.resources[0].schema.fields?.length).toBe(2);
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"load.spec.js","sourceRoot":"","sources":["../../package/load.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { basename } from \"node:path\"\nimport { writeTempFile } from \"@frictionless-ts/dataset\"\nimport { describe, expect, it } from \"vitest\"\nimport { loadPackage } from \"./load.ts\"\n\ndescribe(\"loadPackage\", () => {\n  it(\"should load package from JSON file\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1, name: \"alice\" }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n\n    expect(dataPackage).toBeDefined()\n    expect(dataPackage.name).toBe(\"test-package\")\n    expect(dataPackage.resources).toBeDefined()\n    expect(dataPackage.resources.length).toBe(1)\n  })\n\n  it(\"should load package with multiple resources\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"resource-1\",\n          path: basename(csvPath),\n          format: \"csv\",\n        },\n        {\n          name: \"resource-2\",\n          data: [{ id: 1, value: 100 }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n\n    expect(dataPackage).toBeDefined()\n    expect(dataPackage.name).toBe(\"test-package\")\n    expect(dataPackage.resources.length).toBe(2)\n  })\n\n  it(\"should load package with metadata\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      title: \"Test Package\",\n      description: \"A test data package\",\n      version: \"1.0.0\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1 }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n\n    expect(dataPackage.name).toBe(\"test-package\")\n    expect(dataPackage.title).toBe(\"Test Package\")\n    expect(dataPackage.description).toBe(\"A test data package\")\n    expect(dataPackage.version).toBe(\"1.0.0\")\n  })\n\n  it(\"should load package with inline data resources\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [\n            { id: 1, name: \"alice\" },\n            { id: 2, name: \"bob\" },\n          ],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n\n    expect(dataPackage).toBeDefined()\n    expect(dataPackage.resources?.[0]?.data).toBeDefined()\n  })\n\n  it(\"should load package with schema\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1, name: \"alice\" }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"integer\" },\n              { name: \"name\", type: \"string\" },\n            ],\n          },\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n\n    expect(dataPackage.resources?.[0]?.schema).toBeDefined()\n    expect(\n      typeof dataPackage.resources?.[0]?.schema === \"object\" &&\n        \"fields\" in dataPackage.resources[0].schema &&\n        dataPackage.resources[0].schema.fields?.length,\n    ).toBe(2)\n  })\n})\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { system } from "../system.js";
|
|
2
|
+
export async function savePackage(dataPackage, options) {
|
|
3
|
+
for (const plugin of system.plugins) {
|
|
4
|
+
const result = await plugin.savePackage?.(dataPackage, {
|
|
5
|
+
plugins: system.plugins,
|
|
6
|
+
...options,
|
|
7
|
+
});
|
|
8
|
+
if (result)
|
|
9
|
+
return result;
|
|
10
|
+
}
|
|
11
|
+
throw new Error(`No plugin can save the package: ${options.target}`);
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2F2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3BhY2thZ2Uvc2F2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sY0FBYyxDQUFBO0FBRXJDLE1BQU0sQ0FBQyxLQUFLLFVBQVUsV0FBVyxDQUMvQixXQUFvQixFQUNwQixPQUEyQjtJQUUzQixLQUFLLE1BQU0sTUFBTSxJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUNwQyxNQUFNLE1BQU0sR0FBRyxNQUFNLE1BQU0sQ0FBQyxXQUFXLEVBQUUsQ0FBQyxXQUFXLEVBQUU7WUFDckQsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPO1lBQ3ZCLEdBQUcsT0FBTztTQUNYLENBQUMsQ0FBQTtRQUVGLElBQUksTUFBTTtZQUFFLE9BQU8sTUFBTSxDQUFBO0lBQzNCLENBQUM7SUFFRCxNQUFNLElBQUksS0FBSyxDQUFDLG1DQUFtQyxPQUFPLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQTtBQUN0RSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBTYXZlUGFja2FnZU9wdGlvbnMgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9kYXRhc2V0XCJcbmltcG9ydCB0eXBlIHsgUGFja2FnZSB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL21ldGFkYXRhXCJcbmltcG9ydCB7IHN5c3RlbSB9IGZyb20gXCIuLi9zeXN0ZW0udHNcIlxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gc2F2ZVBhY2thZ2UoXG4gIGRhdGFQYWNrYWdlOiBQYWNrYWdlLFxuICBvcHRpb25zOiBTYXZlUGFja2FnZU9wdGlvbnMsXG4pIHtcbiAgZm9yIChjb25zdCBwbHVnaW4gb2Ygc3lzdGVtLnBsdWdpbnMpIHtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBwbHVnaW4uc2F2ZVBhY2thZ2U/LihkYXRhUGFja2FnZSwge1xuICAgICAgcGx1Z2luczogc3lzdGVtLnBsdWdpbnMsXG4gICAgICAuLi5vcHRpb25zLFxuICAgIH0pXG5cbiAgICBpZiAocmVzdWx0KSByZXR1cm4gcmVzdWx0XG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoYE5vIHBsdWdpbiBjYW4gc2F2ZSB0aGUgcGFja2FnZTogJHtvcHRpb25zLnRhcmdldH1gKVxufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import { access } from "node:fs/promises";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { writeTempFile } from "@frictionless-ts/dataset";
|
|
4
|
+
import { getTempFolderPath } from "@frictionless-ts/dataset";
|
|
5
|
+
import { describe, expect, it } from "vitest";
|
|
6
|
+
import { loadPackage } from "./load.js";
|
|
7
|
+
import { savePackage } from "./save.js";
|
|
8
|
+
describe("savePackage", () => {
|
|
9
|
+
it("should save package to datapackage.json file", async () => {
|
|
10
|
+
const packageContent = JSON.stringify({
|
|
11
|
+
name: "test-package",
|
|
12
|
+
resources: [
|
|
13
|
+
{
|
|
14
|
+
name: "test-resource",
|
|
15
|
+
data: [{ id: 1, name: "alice" }],
|
|
16
|
+
},
|
|
17
|
+
],
|
|
18
|
+
});
|
|
19
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
20
|
+
filename: "datapackage.json",
|
|
21
|
+
});
|
|
22
|
+
const dataPackage = await loadPackage(packagePath);
|
|
23
|
+
const tempDir = getTempFolderPath();
|
|
24
|
+
const targetPath = join(tempDir, "datapackage.json");
|
|
25
|
+
await savePackage(dataPackage, {
|
|
26
|
+
target: targetPath,
|
|
27
|
+
});
|
|
28
|
+
expect(await access(targetPath)
|
|
29
|
+
.then(() => true)
|
|
30
|
+
.catch(() => false)).toBe(true);
|
|
31
|
+
});
|
|
32
|
+
it("should save package with multiple resources", async () => {
|
|
33
|
+
const packageContent = JSON.stringify({
|
|
34
|
+
name: "test-package",
|
|
35
|
+
resources: [
|
|
36
|
+
{
|
|
37
|
+
name: "resource-1",
|
|
38
|
+
data: [{ id: 1, name: "alice" }],
|
|
39
|
+
},
|
|
40
|
+
{
|
|
41
|
+
name: "resource-2",
|
|
42
|
+
data: [{ id: 2, value: 100 }],
|
|
43
|
+
},
|
|
44
|
+
],
|
|
45
|
+
});
|
|
46
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
47
|
+
filename: "datapackage.json",
|
|
48
|
+
});
|
|
49
|
+
const dataPackage = await loadPackage(packagePath);
|
|
50
|
+
const tempDir = getTempFolderPath();
|
|
51
|
+
const targetPath = join(tempDir, "datapackage.json");
|
|
52
|
+
await savePackage(dataPackage, {
|
|
53
|
+
target: targetPath,
|
|
54
|
+
});
|
|
55
|
+
expect(await access(targetPath)
|
|
56
|
+
.then(() => true)
|
|
57
|
+
.catch(() => false)).toBe(true);
|
|
58
|
+
});
|
|
59
|
+
it("should save and reload package with same structure", async () => {
|
|
60
|
+
const packageContent = JSON.stringify({
|
|
61
|
+
name: "test-package",
|
|
62
|
+
title: "Test Package",
|
|
63
|
+
resources: [
|
|
64
|
+
{
|
|
65
|
+
name: "test-resource",
|
|
66
|
+
data: [{ id: 1, name: "alice" }],
|
|
67
|
+
},
|
|
68
|
+
],
|
|
69
|
+
});
|
|
70
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
71
|
+
filename: "datapackage.json",
|
|
72
|
+
});
|
|
73
|
+
const originalPackage = await loadPackage(packagePath);
|
|
74
|
+
const tempDir = getTempFolderPath();
|
|
75
|
+
const targetPath = join(tempDir, "datapackage.json");
|
|
76
|
+
await savePackage(originalPackage, { target: targetPath });
|
|
77
|
+
const reloadedPackage = await loadPackage(targetPath);
|
|
78
|
+
expect(reloadedPackage).toBeDefined();
|
|
79
|
+
expect(reloadedPackage.name).toBe("test-package");
|
|
80
|
+
expect(reloadedPackage.title).toBe("Test Package");
|
|
81
|
+
});
|
|
82
|
+
it("should save package with metadata", async () => {
|
|
83
|
+
const packageContent = JSON.stringify({
|
|
84
|
+
name: "test-package",
|
|
85
|
+
title: "Test Package",
|
|
86
|
+
description: "A test package",
|
|
87
|
+
version: "1.0.0",
|
|
88
|
+
resources: [
|
|
89
|
+
{
|
|
90
|
+
name: "test-resource",
|
|
91
|
+
data: [{ id: 1 }],
|
|
92
|
+
},
|
|
93
|
+
],
|
|
94
|
+
});
|
|
95
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
96
|
+
filename: "datapackage.json",
|
|
97
|
+
});
|
|
98
|
+
const dataPackage = await loadPackage(packagePath);
|
|
99
|
+
const tempDir = getTempFolderPath();
|
|
100
|
+
const targetPath = join(tempDir, "datapackage.json");
|
|
101
|
+
await savePackage(dataPackage, {
|
|
102
|
+
target: targetPath,
|
|
103
|
+
});
|
|
104
|
+
expect(await access(targetPath)
|
|
105
|
+
.then(() => true)
|
|
106
|
+
.catch(() => false)).toBe(true);
|
|
107
|
+
});
|
|
108
|
+
it("should save package with schema", async () => {
|
|
109
|
+
const packageContent = JSON.stringify({
|
|
110
|
+
name: "test-package",
|
|
111
|
+
resources: [
|
|
112
|
+
{
|
|
113
|
+
name: "test-resource",
|
|
114
|
+
data: [{ id: 1, name: "alice" }],
|
|
115
|
+
schema: {
|
|
116
|
+
fields: [
|
|
117
|
+
{ name: "id", type: "integer" },
|
|
118
|
+
{ name: "name", type: "string" },
|
|
119
|
+
],
|
|
120
|
+
},
|
|
121
|
+
},
|
|
122
|
+
],
|
|
123
|
+
});
|
|
124
|
+
const packagePath = await writeTempFile(packageContent, {
|
|
125
|
+
filename: "datapackage.json",
|
|
126
|
+
});
|
|
127
|
+
const dataPackage = await loadPackage(packagePath);
|
|
128
|
+
const tempDir = getTempFolderPath();
|
|
129
|
+
const targetPath = join(tempDir, "datapackage.json");
|
|
130
|
+
await savePackage(dataPackage, {
|
|
131
|
+
target: targetPath,
|
|
132
|
+
});
|
|
133
|
+
expect(await access(targetPath)
|
|
134
|
+
.then(() => true)
|
|
135
|
+
.catch(() => false)).toBe(true);
|
|
136
|
+
});
|
|
137
|
+
});
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"save.spec.js","sourceRoot":"","sources":["../../package/save.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { access } from \"node:fs/promises\"\nimport { join } from \"node:path\"\nimport { writeTempFile } from \"@frictionless-ts/dataset\"\nimport { getTempFolderPath } from \"@frictionless-ts/dataset\"\nimport { describe, expect, it } from \"vitest\"\nimport { loadPackage } from \"./load.ts\"\nimport { savePackage } from \"./save.ts\"\n\ndescribe(\"savePackage\", () => {\n  it(\"should save package to datapackage.json file\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1, name: \"alice\" }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n    const tempDir = getTempFolderPath()\n    const targetPath = join(tempDir, \"datapackage.json\")\n\n    await savePackage(dataPackage, {\n      target: targetPath,\n    })\n\n    expect(\n      await access(targetPath)\n        .then(() => true)\n        .catch(() => false),\n    ).toBe(true)\n  })\n\n  it(\"should save package with multiple resources\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"resource-1\",\n          data: [{ id: 1, name: \"alice\" }],\n        },\n        {\n          name: \"resource-2\",\n          data: [{ id: 2, value: 100 }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n    const tempDir = getTempFolderPath()\n    const targetPath = join(tempDir, \"datapackage.json\")\n\n    await savePackage(dataPackage, {\n      target: targetPath,\n    })\n\n    expect(\n      await access(targetPath)\n        .then(() => true)\n        .catch(() => false),\n    ).toBe(true)\n  })\n\n  it(\"should save and reload package with same structure\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      title: \"Test Package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1, name: \"alice\" }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const originalPackage = await loadPackage(packagePath)\n    const tempDir = getTempFolderPath()\n    const targetPath = join(tempDir, \"datapackage.json\")\n\n    await savePackage(originalPackage, { target: targetPath })\n    const reloadedPackage = await loadPackage(targetPath)\n\n    expect(reloadedPackage).toBeDefined()\n    expect(reloadedPackage.name).toBe(\"test-package\")\n    expect(reloadedPackage.title).toBe(\"Test Package\")\n  })\n\n  it(\"should save package with metadata\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      title: \"Test Package\",\n      description: \"A test package\",\n      version: \"1.0.0\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1 }],\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n    const tempDir = getTempFolderPath()\n    const targetPath = join(tempDir, \"datapackage.json\")\n\n    await savePackage(dataPackage, {\n      target: targetPath,\n    })\n\n    expect(\n      await access(targetPath)\n        .then(() => true)\n        .catch(() => false),\n    ).toBe(true)\n  })\n\n  it(\"should save package with schema\", async () => {\n    const packageContent = JSON.stringify({\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          data: [{ id: 1, name: \"alice\" }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"integer\" },\n              { name: \"name\", type: \"string\" },\n            ],\n          },\n        },\n      ],\n    })\n\n    const packagePath = await writeTempFile(packageContent, {\n      filename: \"datapackage.json\",\n    })\n\n    const dataPackage = await loadPackage(packagePath)\n    const tempDir = getTempFolderPath()\n    const targetPath = join(tempDir, \"datapackage.json\")\n\n    await savePackage(dataPackage, {\n      target: targetPath,\n    })\n\n    expect(\n      await access(targetPath)\n        .then(() => true)\n        .catch(() => false),\n    ).toBe(true)\n  })\n})\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { BoundError } from "@frictionless-ts/metadata";
|
|
2
|
+
import type { Descriptor, Package } from "@frictionless-ts/metadata";
|
|
3
|
+
export declare function validatePackage(source: string | Descriptor | Partial<Package>, options?: {
|
|
4
|
+
basepath?: string;
|
|
5
|
+
}): Promise<{
|
|
6
|
+
errors: BoundError[];
|
|
7
|
+
valid: boolean;
|
|
8
|
+
} | {
|
|
9
|
+
valid: boolean;
|
|
10
|
+
errors: {
|
|
11
|
+
resource: undefined;
|
|
12
|
+
type: "metadata";
|
|
13
|
+
pointer: string;
|
|
14
|
+
message: string;
|
|
15
|
+
}[];
|
|
16
|
+
}>;
|
|
17
|
+
export declare function validatePackageData(dataPackage: Package): Promise<{
|
|
18
|
+
errors: BoundError[];
|
|
19
|
+
valid: boolean;
|
|
20
|
+
}>;
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import os from "node:os";
|
|
2
|
+
import { createReport } from "@frictionless-ts/metadata";
|
|
3
|
+
import { loadDescriptor, validatePackageMetadata, } from "@frictionless-ts/metadata";
|
|
4
|
+
import { resolveBasepath } from "@frictionless-ts/metadata";
|
|
5
|
+
import pAll from "p-all";
|
|
6
|
+
import { validateResourceData } from "../resource/index.js";
|
|
7
|
+
import { system } from "../system.js";
|
|
8
|
+
import { validatePackageIntegrity } from "./integrity.js";
|
|
9
|
+
export async function validatePackage(source, options) {
|
|
10
|
+
let descriptor;
|
|
11
|
+
let basepath = options?.basepath;
|
|
12
|
+
if (typeof source !== "string") {
|
|
13
|
+
descriptor = source;
|
|
14
|
+
}
|
|
15
|
+
else {
|
|
16
|
+
for (const plugin of system.plugins) {
|
|
17
|
+
const result = await plugin.loadPackage?.(source);
|
|
18
|
+
if (result) {
|
|
19
|
+
descriptor = result;
|
|
20
|
+
break;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
if (!descriptor) {
|
|
24
|
+
basepath = await resolveBasepath(source);
|
|
25
|
+
descriptor = await loadDescriptor(source);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
const metadataReport = await validatePackageMetadata(descriptor, {
|
|
29
|
+
basepath,
|
|
30
|
+
});
|
|
31
|
+
if (!metadataReport.dataPackage) {
|
|
32
|
+
return {
|
|
33
|
+
valid: metadataReport.valid,
|
|
34
|
+
errors: metadataReport.errors.map(error => ({
|
|
35
|
+
...error,
|
|
36
|
+
resource: undefined,
|
|
37
|
+
})),
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
const dataReport = await validatePackageData(metadataReport.dataPackage);
|
|
41
|
+
const integrityReport = await validatePackageIntegrity(metadataReport.dataPackage);
|
|
42
|
+
const errors = [...dataReport.errors, ...integrityReport.errors];
|
|
43
|
+
return createReport(errors);
|
|
44
|
+
}
|
|
45
|
+
export async function validatePackageData(dataPackage) {
|
|
46
|
+
const concurrency = os.cpus().length;
|
|
47
|
+
const errors = (await pAll(dataPackage.resources.map(resource => async () => {
|
|
48
|
+
try {
|
|
49
|
+
const report = await validateResourceData(resource);
|
|
50
|
+
return report.errors.map(error => ({
|
|
51
|
+
...error,
|
|
52
|
+
resource: resource.name,
|
|
53
|
+
}));
|
|
54
|
+
}
|
|
55
|
+
catch (error) {
|
|
56
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
57
|
+
throw new Error(`[${resource.name}] ${message}`);
|
|
58
|
+
}
|
|
59
|
+
}), { concurrency })).flat();
|
|
60
|
+
return createReport(errors);
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9wYWNrYWdlL3ZhbGlkYXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLFNBQVMsQ0FBQTtBQUV4QixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMkJBQTJCLENBQUE7QUFFeEQsT0FBTyxFQUNMLGNBQWMsRUFDZCx1QkFBdUIsR0FDeEIsTUFBTSwyQkFBMkIsQ0FBQTtBQUNsQyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sMkJBQTJCLENBQUE7QUFDM0QsT0FBTyxJQUFJLE1BQU0sT0FBTyxDQUFBO0FBQ3hCLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHNCQUFzQixDQUFBO0FBQzNELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxjQUFjLENBQUE7QUFDckMsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sZ0JBQWdCLENBQUE7QUFFekQsTUFBTSxDQUFDLEtBQUssVUFBVSxlQUFlLENBQ25DLE1BQThDLEVBQzlDLE9BQStCO0lBRS9CLElBQUksVUFBa0MsQ0FBQTtJQUN0QyxJQUFJLFFBQVEsR0FBRyxPQUFPLEVBQUUsUUFBUSxDQUFBO0lBRWhDLElBQUksT0FBTyxNQUFNLEtBQUssUUFBUSxFQUFFLENBQUM7UUFDL0IsVUFBVSxHQUFHLE1BQU0sQ0FBQTtJQUNyQixDQUFDO1NBQU0sQ0FBQztRQUNOLEtBQUssTUFBTSxNQUFNLElBQUksTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3BDLE1BQU0sTUFBTSxHQUFHLE1BQU0sTUFBTSxDQUFDLFdBQVcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFBO1lBQ2pELElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ1gsVUFBVSxHQUFHLE1BQStCLENBQUE7Z0JBQzVDLE1BQUs7WUFDUCxDQUFDO1FBQ0gsQ0FBQztRQUVELElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUNoQixRQUFRLEdBQUcsTUFBTSxlQUFlLENBQUMsTUFBTSxDQUFDLENBQUE7WUFDeEMsVUFBVSxHQUFHLE1BQU0sY0FBYyxDQUFDLE1BQU0sQ0FBQyxDQUFBO1FBQzNDLENBQUM7SUFDSCxDQUFDO0lBRUQsTUFBTSxjQUFjLEdBQUcsTUFBTSx1QkFBdUIsQ0FBQyxVQUFVLEVBQUU7UUFDL0QsUUFBUTtLQUNULENBQUMsQ0FBQTtJQUVGLElBQUksQ0FBQyxjQUFjLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDaEMsT0FBTztZQUNMLEtBQUssRUFBRSxjQUFjLENBQUMsS0FBSztZQUMzQixNQUFNLEVBQUUsY0FBYyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDO2dCQUMxQyxHQUFHLEtBQUs7Z0JBQ1IsUUFBUSxFQUFFLFNBQVM7YUFDcEIsQ0FBQyxDQUFDO1NBQ0osQ0FBQTtJQUNILENBQUM7SUFFRCxNQUFNLFVBQVUsR0FBRyxNQUFNLG1CQUFtQixDQUFDLGNBQWMsQ0FBQyxXQUFXLENBQUMsQ0FBQTtJQUN4RSxNQUFNLGVBQWUsR0FBRyxNQUFNLHdCQUF3QixDQUNwRCxjQUFjLENBQUMsV0FBVyxDQUMzQixDQUFBO0lBRUQsTUFBTSxNQUFNLEdBQUcsQ0FBQyxHQUFHLFVBQVUsQ0FBQyxNQUFNLEVBQUUsR0FBRyxlQUFlLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDaEUsT0FBTyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUE7QUFDN0IsQ0FBQztBQUVELE1BQU0sQ0FBQyxLQUFLLFVBQVUsbUJBQW1CLENBQUMsV0FBb0I7SUFDNUQsTUFBTSxXQUFXLEdBQUcsRUFBRSxDQUFDLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQTtJQUVwQyxNQUFNLE1BQU0sR0FBaUIsQ0FDM0IsTUFBTSxJQUFJLENBQ1IsV0FBVyxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxLQUFLLElBQUksRUFBRTtRQUMvQyxJQUFJLENBQUM7WUFDSCxNQUFNLE1BQU0sR0FBRyxNQUFNLG9CQUFvQixDQUFDLFFBQVEsQ0FBQyxDQUFBO1lBQ25ELE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDO2dCQUNqQyxHQUFHLEtBQUs7Z0JBQ1IsUUFBUSxFQUFFLFFBQVEsQ0FBQyxJQUFJO2FBQ3hCLENBQUMsQ0FBQyxDQUFBO1FBQ0wsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLE9BQU8sR0FBRyxLQUFLLFlBQVksS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUE7WUFDdEUsTUFBTSxJQUFJLEtBQUssQ0FBQyxJQUFJLFFBQVEsQ0FBQyxJQUFJLEtBQUssT0FBTyxFQUFFLENBQUMsQ0FBQTtRQUNsRCxDQUFDO0lBQ0gsQ0FBQyxDQUFDLEVBQ0YsRUFBRSxXQUFXLEVBQUUsQ0FDaEIsQ0FDRixDQUFDLElBQUksRUFBRSxDQUFBO0lBRVIsT0FBTyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUE7QUFDN0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBvcyBmcm9tIFwibm9kZTpvc1wiXG5pbXBvcnQgdHlwZSB7IEJvdW5kRXJyb3IgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9tZXRhZGF0YVwiXG5pbXBvcnQgeyBjcmVhdGVSZXBvcnQgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9tZXRhZGF0YVwiXG5pbXBvcnQgdHlwZSB7IERlc2NyaXB0b3IsIFBhY2thZ2UgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9tZXRhZGF0YVwiXG5pbXBvcnQge1xuICBsb2FkRGVzY3JpcHRvcixcbiAgdmFsaWRhdGVQYWNrYWdlTWV0YWRhdGEsXG59IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL21ldGFkYXRhXCJcbmltcG9ydCB7IHJlc29sdmVCYXNlcGF0aCB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL21ldGFkYXRhXCJcbmltcG9ydCBwQWxsIGZyb20gXCJwLWFsbFwiXG5pbXBvcnQgeyB2YWxpZGF0ZVJlc291cmNlRGF0YSB9IGZyb20gXCIuLi9yZXNvdXJjZS9pbmRleC50c1wiXG5pbXBvcnQgeyBzeXN0ZW0gfSBmcm9tIFwiLi4vc3lzdGVtLnRzXCJcbmltcG9ydCB7IHZhbGlkYXRlUGFja2FnZUludGVncml0eSB9IGZyb20gXCIuL2ludGVncml0eS50c1wiXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB2YWxpZGF0ZVBhY2thZ2UoXG4gIHNvdXJjZTogc3RyaW5nIHwgRGVzY3JpcHRvciB8IFBhcnRpYWw8UGFja2FnZT4sXG4gIG9wdGlvbnM/OiB7IGJhc2VwYXRoPzogc3RyaW5nIH0sXG4pIHtcbiAgbGV0IGRlc2NyaXB0b3I6IERlc2NyaXB0b3IgfCB1bmRlZmluZWRcbiAgbGV0IGJhc2VwYXRoID0gb3B0aW9ucz8uYmFzZXBhdGhcblxuICBpZiAodHlwZW9mIHNvdXJjZSAhPT0gXCJzdHJpbmdcIikge1xuICAgIGRlc2NyaXB0b3IgPSBzb3VyY2VcbiAgfSBlbHNlIHtcbiAgICBmb3IgKGNvbnN0IHBsdWdpbiBvZiBzeXN0ZW0ucGx1Z2lucykge1xuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcGx1Z2luLmxvYWRQYWNrYWdlPy4oc291cmNlKVxuICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICBkZXNjcmlwdG9yID0gcmVzdWx0IGFzIHVua25vd24gYXMgRGVzY3JpcHRvclxuICAgICAgICBicmVha1xuICAgICAgfVxuICAgIH1cblxuICAgIGlmICghZGVzY3JpcHRvcikge1xuICAgICAgYmFzZXBhdGggPSBhd2FpdCByZXNvbHZlQmFzZXBhdGgoc291cmNlKVxuICAgICAgZGVzY3JpcHRvciA9IGF3YWl0IGxvYWREZXNjcmlwdG9yKHNvdXJjZSlcbiAgICB9XG4gIH1cblxuICBjb25zdCBtZXRhZGF0YVJlcG9ydCA9IGF3YWl0IHZhbGlkYXRlUGFja2FnZU1ldGFkYXRhKGRlc2NyaXB0b3IsIHtcbiAgICBiYXNlcGF0aCxcbiAgfSlcblxuICBpZiAoIW1ldGFkYXRhUmVwb3J0LmRhdGFQYWNrYWdlKSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZhbGlkOiBtZXRhZGF0YVJlcG9ydC52YWxpZCxcbiAgICAgIGVycm9yczogbWV0YWRhdGFSZXBvcnQuZXJyb3JzLm1hcChlcnJvciA9PiAoe1xuICAgICAgICAuLi5lcnJvcixcbiAgICAgICAgcmVzb3VyY2U6IHVuZGVmaW5lZCxcbiAgICAgIH0pKSxcbiAgICB9XG4gIH1cblxuICBjb25zdCBkYXRhUmVwb3J0ID0gYXdhaXQgdmFsaWRhdGVQYWNrYWdlRGF0YShtZXRhZGF0YVJlcG9ydC5kYXRhUGFja2FnZSlcbiAgY29uc3QgaW50ZWdyaXR5UmVwb3J0ID0gYXdhaXQgdmFsaWRhdGVQYWNrYWdlSW50ZWdyaXR5KFxuICAgIG1ldGFkYXRhUmVwb3J0LmRhdGFQYWNrYWdlLFxuICApXG5cbiAgY29uc3QgZXJyb3JzID0gWy4uLmRhdGFSZXBvcnQuZXJyb3JzLCAuLi5pbnRlZ3JpdHlSZXBvcnQuZXJyb3JzXVxuICByZXR1cm4gY3JlYXRlUmVwb3J0KGVycm9ycylcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHZhbGlkYXRlUGFja2FnZURhdGEoZGF0YVBhY2thZ2U6IFBhY2thZ2UpIHtcbiAgY29uc3QgY29uY3VycmVuY3kgPSBvcy5jcHVzKCkubGVuZ3RoXG5cbiAgY29uc3QgZXJyb3JzOiBCb3VuZEVycm9yW10gPSAoXG4gICAgYXdhaXQgcEFsbChcbiAgICAgIGRhdGFQYWNrYWdlLnJlc291cmNlcy5tYXAocmVzb3VyY2UgPT4gYXN5bmMgKCkgPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGNvbnN0IHJlcG9ydCA9IGF3YWl0IHZhbGlkYXRlUmVzb3VyY2VEYXRhKHJlc291cmNlKVxuICAgICAgICAgIHJldHVybiByZXBvcnQuZXJyb3JzLm1hcChlcnJvciA9PiAoe1xuICAgICAgICAgICAgLi4uZXJyb3IsXG4gICAgICAgICAgICByZXNvdXJjZTogcmVzb3VyY2UubmFtZSxcbiAgICAgICAgICB9KSlcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICBjb25zdCBtZXNzYWdlID0gZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpXG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBbJHtyZXNvdXJjZS5uYW1lfV0gJHttZXNzYWdlfWApXG4gICAgICAgIH1cbiAgICAgIH0pLFxuICAgICAgeyBjb25jdXJyZW5jeSB9LFxuICAgIClcbiAgKS5mbGF0KClcblxuICByZXR1cm4gY3JlYXRlUmVwb3J0KGVycm9ycylcbn1cbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { join } from "node:path";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { validatePackage } from "./validate.js";
|
|
4
|
+
describe("validatePackage", () => {
|
|
5
|
+
it("should validate a valid package with inline data", async () => {
|
|
6
|
+
const dataPackage = {
|
|
7
|
+
name: "test-package",
|
|
8
|
+
resources: [
|
|
9
|
+
{
|
|
10
|
+
name: "test-resource",
|
|
11
|
+
type: "table",
|
|
12
|
+
data: [
|
|
13
|
+
{ id: 1, name: "Alice" },
|
|
14
|
+
{ id: 2, name: "Bob" },
|
|
15
|
+
],
|
|
16
|
+
schema: {
|
|
17
|
+
fields: [
|
|
18
|
+
{ name: "id", type: "number" },
|
|
19
|
+
{ name: "name", type: "string" },
|
|
20
|
+
],
|
|
21
|
+
},
|
|
22
|
+
},
|
|
23
|
+
],
|
|
24
|
+
};
|
|
25
|
+
const report = await validatePackage(dataPackage);
|
|
26
|
+
expect(report.valid).toBe(true);
|
|
27
|
+
expect(report.errors).toEqual([]);
|
|
28
|
+
});
|
|
29
|
+
it("should detect invalid resource data", async () => {
|
|
30
|
+
const dataPackage = {
|
|
31
|
+
name: "test-package",
|
|
32
|
+
resources: [
|
|
33
|
+
{
|
|
34
|
+
name: "test-resource",
|
|
35
|
+
type: "table",
|
|
36
|
+
data: [
|
|
37
|
+
{ id: 1, name: "Alice" },
|
|
38
|
+
{ id: "not-a-number", name: "Bob" },
|
|
39
|
+
],
|
|
40
|
+
schema: {
|
|
41
|
+
fields: [
|
|
42
|
+
{ name: "id", type: "number" },
|
|
43
|
+
{ name: "name", type: "string" },
|
|
44
|
+
],
|
|
45
|
+
},
|
|
46
|
+
},
|
|
47
|
+
],
|
|
48
|
+
};
|
|
49
|
+
const report = await validatePackage(dataPackage);
|
|
50
|
+
expect(report.valid).toBe(false);
|
|
51
|
+
expect(report.errors.length).toBeGreaterThan(0);
|
|
52
|
+
expect(report.errors?.[0]?.resource).toBe("test-resource");
|
|
53
|
+
});
|
|
54
|
+
it("should validate multiple resources", async () => {
|
|
55
|
+
const dataPackage = {
|
|
56
|
+
name: "test-package",
|
|
57
|
+
resources: [
|
|
58
|
+
{
|
|
59
|
+
name: "resource-1",
|
|
60
|
+
type: "table",
|
|
61
|
+
data: [{ id: 1, name: "Alice" }],
|
|
62
|
+
schema: {
|
|
63
|
+
fields: [
|
|
64
|
+
{ name: "id", type: "number" },
|
|
65
|
+
{ name: "name", type: "string" },
|
|
66
|
+
],
|
|
67
|
+
},
|
|
68
|
+
},
|
|
69
|
+
{
|
|
70
|
+
name: "resource-2",
|
|
71
|
+
type: "table",
|
|
72
|
+
data: [{ id: 2, value: 100 }],
|
|
73
|
+
schema: {
|
|
74
|
+
fields: [
|
|
75
|
+
{ name: "id", type: "number" },
|
|
76
|
+
{ name: "value", type: "number" },
|
|
77
|
+
],
|
|
78
|
+
},
|
|
79
|
+
},
|
|
80
|
+
],
|
|
81
|
+
};
|
|
82
|
+
const report = await validatePackage(dataPackage);
|
|
83
|
+
expect(report.valid).toBe(true);
|
|
84
|
+
expect(report.errors).toEqual([]);
|
|
85
|
+
});
|
|
86
|
+
it("should detect errors in multiple resources", async () => {
|
|
87
|
+
const dataPackage = {
|
|
88
|
+
name: "test-package",
|
|
89
|
+
resources: [
|
|
90
|
+
{
|
|
91
|
+
name: "resource-1",
|
|
92
|
+
type: "table",
|
|
93
|
+
data: [{ id: "invalid", name: "Alice" }],
|
|
94
|
+
schema: {
|
|
95
|
+
fields: [
|
|
96
|
+
{ name: "id", type: "number" },
|
|
97
|
+
{ name: "name", type: "string" },
|
|
98
|
+
],
|
|
99
|
+
},
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
name: "resource-2",
|
|
103
|
+
type: "table",
|
|
104
|
+
data: [{ id: 2, value: "invalid" }],
|
|
105
|
+
schema: {
|
|
106
|
+
fields: [
|
|
107
|
+
{ name: "id", type: "number" },
|
|
108
|
+
{ name: "value", type: "number" },
|
|
109
|
+
],
|
|
110
|
+
},
|
|
111
|
+
},
|
|
112
|
+
],
|
|
113
|
+
};
|
|
114
|
+
const report = await validatePackage(dataPackage);
|
|
115
|
+
expect(report.valid).toBe(false);
|
|
116
|
+
expect(report.errors.length).toBeGreaterThan(1);
|
|
117
|
+
expect(report.errors.some(e => e.resource === "resource-1")).toBe(true);
|
|
118
|
+
expect(report.errors.some(e => e.resource === "resource-2")).toBe(true);
|
|
119
|
+
});
|
|
120
|
+
it("should reject package with no resources", async () => {
|
|
121
|
+
const dataPackage = {
|
|
122
|
+
name: "test-package",
|
|
123
|
+
resources: [],
|
|
124
|
+
};
|
|
125
|
+
const report = await validatePackage(dataPackage);
|
|
126
|
+
expect(report.valid).toBe(false);
|
|
127
|
+
expect(report.errors.length).toBeGreaterThan(0);
|
|
128
|
+
const firstError = report.errors?.[0];
|
|
129
|
+
if (firstError && "message" in firstError) {
|
|
130
|
+
expect(firstError.message).toContain("must NOT have fewer than 1 items");
|
|
131
|
+
}
|
|
132
|
+
});
|
|
133
|
+
it("should tag errors with resource name", async () => {
|
|
134
|
+
const dataPackage = {
|
|
135
|
+
name: "test-package",
|
|
136
|
+
resources: [
|
|
137
|
+
{
|
|
138
|
+
name: "error-resource",
|
|
139
|
+
type: "table",
|
|
140
|
+
data: [{ id: "invalid", name: 123 }],
|
|
141
|
+
schema: {
|
|
142
|
+
fields: [
|
|
143
|
+
{ name: "id", type: "number" },
|
|
144
|
+
{ name: "name", type: "string" },
|
|
145
|
+
],
|
|
146
|
+
},
|
|
147
|
+
},
|
|
148
|
+
],
|
|
149
|
+
};
|
|
150
|
+
const report = await validatePackage(dataPackage);
|
|
151
|
+
expect(report.valid).toBe(false);
|
|
152
|
+
report.errors.forEach(error => {
|
|
153
|
+
expect(error.resource).toBe("error-resource");
|
|
154
|
+
});
|
|
155
|
+
});
|
|
156
|
+
it("should detect bad cell type (issue-153)", async () => {
|
|
157
|
+
const dataPackage = join(import.meta.dirname, "fixtures/issue-153/datapackage.json");
|
|
158
|
+
const report = await validatePackage(dataPackage);
|
|
159
|
+
expect(report.valid).toBe(false);
|
|
160
|
+
expect(report.errors).toEqual([
|
|
161
|
+
{
|
|
162
|
+
rowNumber: 3,
|
|
163
|
+
type: "cell/type",
|
|
164
|
+
fieldName: "longitude",
|
|
165
|
+
fieldType: "number",
|
|
166
|
+
cell: "bad",
|
|
167
|
+
resource: "deployments",
|
|
168
|
+
},
|
|
169
|
+
]);
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validate.spec.js","sourceRoot":"","sources":["../../package/validate.spec.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAA;AAChC,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAC7C,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAA;AAE/C,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;IAC/B,EAAE,CAAC,kDAAkD,EAAE,KAAK,IAAI,EAAE;QAChE,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE;gBACT;oBACE,IAAI,EAAE,eAAe;oBACrB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE;wBACJ,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE;wBACxB,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE;qBACvB;oBACD,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC1C;qBACF;iBACF;aACF;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC/B,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAA;IACnC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;QACnD,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE;gBACT;oBACE,IAAI,EAAE,eAAe;oBACrB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE;wBACJ,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE;wBACxB,EAAE,EAAE,EAAE,cAAc,EAAE,IAAI,EAAE,KAAK,EAAE;qBACpC;oBACD,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC1C;qBACF;iBACF;aACF;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAChC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;QAC/C,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAA;IAC5D,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE;gBACT;oBACE,IAAI,EAAE,YAAY;oBAClB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;oBAChC,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC1C;qBACF;iBACF;gBACD;oBACE,IAAI,EAAE,YAAY;oBAClB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;oBAC7B,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC3C;qBACF;iBACF;aACF;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC/B,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAA;IACnC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,4CAA4C,EAAE,KAAK,IAAI,EAAE;QAC1D,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE;gBACT;oBACE,IAAI,EAAE,YAAY;oBAClB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;oBACxC,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC1C;qBACF;iBACF;gBACD;oBACE,IAAI,EAAE,YAAY;oBAClB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC;oBACnC,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC3C;qBACF;iBACF;aACF;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAChC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;QAC/C,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACvE,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;IACzE,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,yCAAyC,EAAE,KAAK,IAAI,EAAE;QACvD,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE,EAAE;SACd,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAChC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;QAC/C,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,CAAA;QACrC,IAAI,UAAU,IAAI,SAAS,IAAI,UAAU,EAAE,CAAC;YAC1C,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,kCAAkC,CAAC,CAAA;QAC1E,CAAC;IACH,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;QACpD,MAAM,WAAW,GAAG;YAClB,IAAI,EAAE,cAAc;YACpB,SAAS,EAAE;gBACT;oBACE,IAAI,EAAE,gBAAgB;oBACtB,IAAI,EAAE,OAAgB;oBACtB,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC;oBACpC,MAAM,EAAE;wBACN,MAAM,EAAE;4BACN,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAiB,EAAE;4BACvC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAiB,EAAE;yBAC1C;qBACF;iBACF;aACF;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAChC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YAC5B,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAA;QAC/C,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,yCAAyC,EAAE,KAAK,IAAI,EAAE;QACvD,MAAM,WAAW,GAAG,IAAI,CACtB,MAAM,CAAC,IAAI,CAAC,OAAO,EACnB,qCAAqC,CACtC,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,WAAW,CAAC,CAAA;QAEjD,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAChC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;YAC5B;gBACE,SAAS,EAAE,CAAC;gBACZ,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,WAAW;gBACtB,SAAS,EAAE,QAAQ;gBACnB,IAAI,EAAE,KAAK;gBACX,QAAQ,EAAE,aAAa;aACxB;SACF,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA","sourcesContent":["import { join } from \"node:path\"\nimport { describe, expect, it } from \"vitest\"\nimport { validatePackage } from \"./validate.ts\"\n\ndescribe(\"validatePackage\", () => {\n  it(\"should validate a valid package with inline data\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          type: \"table\" as const,\n          data: [\n            { id: 1, name: \"Alice\" },\n            { id: 2, name: \"Bob\" },\n          ],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"name\", type: \"string\" as const },\n            ],\n          },\n        },\n      ],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(true)\n    expect(report.errors).toEqual([])\n  })\n\n  it(\"should detect invalid resource data\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"test-resource\",\n          type: \"table\" as const,\n          data: [\n            { id: 1, name: \"Alice\" },\n            { id: \"not-a-number\", name: \"Bob\" },\n          ],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"name\", type: \"string\" as const },\n            ],\n          },\n        },\n      ],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors.length).toBeGreaterThan(0)\n    expect(report.errors?.[0]?.resource).toBe(\"test-resource\")\n  })\n\n  it(\"should validate multiple resources\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"resource-1\",\n          type: \"table\" as const,\n          data: [{ id: 1, name: \"Alice\" }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"name\", type: \"string\" as const },\n            ],\n          },\n        },\n        {\n          name: \"resource-2\",\n          type: \"table\" as const,\n          data: [{ id: 2, value: 100 }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"value\", type: \"number\" as const },\n            ],\n          },\n        },\n      ],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(true)\n    expect(report.errors).toEqual([])\n  })\n\n  it(\"should detect errors in multiple resources\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"resource-1\",\n          type: \"table\" as const,\n          data: [{ id: \"invalid\", name: \"Alice\" }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"name\", type: \"string\" as const },\n            ],\n          },\n        },\n        {\n          name: \"resource-2\",\n          type: \"table\" as const,\n          data: [{ id: 2, value: \"invalid\" }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"value\", type: \"number\" as const },\n            ],\n          },\n        },\n      ],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors.length).toBeGreaterThan(1)\n    expect(report.errors.some(e => e.resource === \"resource-1\")).toBe(true)\n    expect(report.errors.some(e => e.resource === \"resource-2\")).toBe(true)\n  })\n\n  it(\"should reject package with no resources\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors.length).toBeGreaterThan(0)\n    const firstError = report.errors?.[0]\n    if (firstError && \"message\" in firstError) {\n      expect(firstError.message).toContain(\"must NOT have fewer than 1 items\")\n    }\n  })\n\n  it(\"should tag errors with resource name\", async () => {\n    const dataPackage = {\n      name: \"test-package\",\n      resources: [\n        {\n          name: \"error-resource\",\n          type: \"table\" as const,\n          data: [{ id: \"invalid\", name: 123 }],\n          schema: {\n            fields: [\n              { name: \"id\", type: \"number\" as const },\n              { name: \"name\", type: \"string\" as const },\n            ],\n          },\n        },\n      ],\n    }\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(false)\n    report.errors.forEach(error => {\n      expect(error.resource).toBe(\"error-resource\")\n    })\n  })\n\n  it(\"should detect bad cell type (issue-153)\", async () => {\n    const dataPackage = join(\n      import.meta.dirname,\n      \"fixtures/issue-153/datapackage.json\",\n    )\n\n    const report = await validatePackage(dataPackage)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors).toEqual([\n      {\n        rowNumber: 3,\n        type: \"cell/type\",\n        fieldName: \"longitude\",\n        fieldType: \"number\",\n        cell: \"bad\",\n        resource: \"deployments\",\n      },\n    ])\n  })\n})\n"]}
|
package/build/plugin.js
ADDED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vcGx1Z2luLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFRhYmxlUGx1Z2luIH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvdGFibGVcIlxuXG5leHBvcnQgdHlwZSBQbHVnaW4gPSBUYWJsZVBsdWdpblxuIl19
|