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,3 @@
|
|
|
1
|
+
export { inferResource } from "./infer.js";
|
|
2
|
+
export { validateResource, validateResourceData } from "./validate.js";
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9yZXNvdXJjZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sWUFBWSxDQUFBO0FBQzFDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxvQkFBb0IsRUFBRSxNQUFNLGVBQWUsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IGluZmVyUmVzb3VyY2UgfSBmcm9tIFwiLi9pbmZlci50c1wiXG5leHBvcnQgeyB2YWxpZGF0ZVJlc291cmNlLCB2YWxpZGF0ZVJlc291cmNlRGF0YSB9IGZyb20gXCIuL3ZhbGlkYXRlLnRzXCJcbiJdfQ==
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import type { Resource } from "@frictionless-ts/metadata";
|
|
2
|
+
import type { InferDialectOptions } from "@frictionless-ts/table";
|
|
3
|
+
import type { InferSchemaOptions } from "@frictionless-ts/table";
|
|
4
|
+
export declare function inferResource(resource: Partial<Resource>, options?: InferDialectOptions & InferSchemaOptions): Promise<{
|
|
5
|
+
name: string;
|
|
6
|
+
$schema?: string | undefined;
|
|
7
|
+
path?: string | string[] | undefined;
|
|
8
|
+
data?: unknown | unknown[][] | Record<string, unknown>[];
|
|
9
|
+
type?: "table" | undefined;
|
|
10
|
+
format?: string | undefined;
|
|
11
|
+
mediatype?: string | undefined;
|
|
12
|
+
encoding?: string | undefined;
|
|
13
|
+
title?: string | undefined;
|
|
14
|
+
description?: string | undefined;
|
|
15
|
+
bytes?: number | undefined;
|
|
16
|
+
hash?: string | undefined;
|
|
17
|
+
sources?: import("@frictionless-ts/metadata").Source[] | undefined;
|
|
18
|
+
licenses?: import("@frictionless-ts/metadata").License[] | undefined;
|
|
19
|
+
dialect?: (string | import("@frictionless-ts/metadata").Dialect) | undefined;
|
|
20
|
+
schema?: (string | import("@frictionless-ts/metadata").Schema) | undefined;
|
|
21
|
+
jsonSchema?: (string | import("@frictionless-ts/metadata").Descriptor) | undefined;
|
|
22
|
+
}>;
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { prefetchFile } from "@frictionless-ts/dataset";
|
|
2
|
+
import { inferBytes, inferEncoding, inferHash } from "@frictionless-ts/dataset";
|
|
3
|
+
import { inferFormat, inferName } from "@frictionless-ts/metadata";
|
|
4
|
+
import { inferDialect } from "../dialect/index.js";
|
|
5
|
+
import { inferSchema } from "../schema/index.js";
|
|
6
|
+
export async function inferResource(resource, options) {
|
|
7
|
+
const result = {
|
|
8
|
+
...resource,
|
|
9
|
+
name: resource.name ?? inferName(resource),
|
|
10
|
+
};
|
|
11
|
+
if (!result.format) {
|
|
12
|
+
result.format = inferFormat(resource);
|
|
13
|
+
}
|
|
14
|
+
if (typeof resource.path === "string") {
|
|
15
|
+
const localPath = await prefetchFile(resource.path);
|
|
16
|
+
const localResource = { ...resource, path: localPath };
|
|
17
|
+
if (!result.encoding) {
|
|
18
|
+
const encoding = await inferEncoding(localResource);
|
|
19
|
+
if (encoding) {
|
|
20
|
+
result.encoding = encoding;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
if (!result.bytes) {
|
|
24
|
+
result.bytes = await inferBytes(localResource);
|
|
25
|
+
}
|
|
26
|
+
if (!result.hash) {
|
|
27
|
+
result.hash = await inferHash(localResource);
|
|
28
|
+
}
|
|
29
|
+
if (!result.dialect) {
|
|
30
|
+
try {
|
|
31
|
+
result.dialect = await inferDialect(localResource, options);
|
|
32
|
+
}
|
|
33
|
+
catch { }
|
|
34
|
+
}
|
|
35
|
+
if (!result.schema) {
|
|
36
|
+
try {
|
|
37
|
+
result.schema = await inferSchema(localResource, options);
|
|
38
|
+
}
|
|
39
|
+
catch { }
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
if (!result.type) {
|
|
43
|
+
if (result.schema) {
|
|
44
|
+
result.type = "table";
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return result;
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5mZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9yZXNvdXJjZS9pbmZlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMEJBQTBCLENBQUE7QUFDdkQsT0FBTyxFQUFFLFVBQVUsRUFBRSxhQUFhLEVBQUUsU0FBUyxFQUFFLE1BQU0sMEJBQTBCLENBQUE7QUFFL0UsT0FBTyxFQUFFLFdBQVcsRUFBRSxTQUFTLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQTtBQUdsRSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDbEQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBRWhELE1BQU0sQ0FBQyxLQUFLLFVBQVUsYUFBYSxDQUNqQyxRQUEyQixFQUMzQixPQUFrRDtJQUVsRCxNQUFNLE1BQU0sR0FBRztRQUNiLEdBQUcsUUFBUTtRQUNYLElBQUksRUFBRSxRQUFRLENBQUMsSUFBSSxJQUFJLFNBQVMsQ0FBQyxRQUFRLENBQUM7S0FDM0MsQ0FBQTtJQUVELElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDbkIsTUFBTSxDQUFDLE1BQU0sR0FBRyxXQUFXLENBQUMsUUFBUSxDQUFDLENBQUE7SUFDdkMsQ0FBQztJQUVELElBQUksT0FBTyxRQUFRLENBQUMsSUFBSSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3RDLE1BQU0sU0FBUyxHQUFHLE1BQU0sWUFBWSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQTtRQUNuRCxNQUFNLGFBQWEsR0FBRyxFQUFFLEdBQUcsUUFBUSxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsQ0FBQTtRQUV0RCxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO1lBQ3JCLE1BQU0sUUFBUSxHQUFHLE1BQU0sYUFBYSxDQUFDLGFBQWEsQ0FBQyxDQUFBO1lBQ25ELElBQUksUUFBUSxFQUFFLENBQUM7Z0JBQ2IsTUFBTSxDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUE7WUFDNUIsQ0FBQztRQUNILENBQUM7UUFFRCxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ2xCLE1BQU0sQ0FBQyxLQUFLLEdBQUcsTUFBTSxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUE7UUFDaEQsQ0FBQztRQUVELElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDakIsTUFBTSxDQUFDLElBQUksR0FBRyxNQUFNLFNBQVMsQ0FBQyxhQUFhLENBQUMsQ0FBQTtRQUM5QyxDQUFDO1FBRUQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNwQixJQUFJLENBQUM7Z0JBQ0gsTUFBTSxDQUFDLE9BQU8sR0FBRyxNQUFNLFlBQVksQ0FBQyxhQUFhLEVBQUUsT0FBTyxDQUFDLENBQUE7WUFDN0QsQ0FBQztZQUFDLE1BQU0sQ0FBQyxDQUFBLENBQUM7UUFDWixDQUFDO1FBRUQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNuQixJQUFJLENBQUM7Z0JBQ0gsTUFBTSxDQUFDLE1BQU0sR0FBRyxNQUFNLFdBQVcsQ0FBQyxhQUFhLEVBQUUsT0FBTyxDQUFDLENBQUE7WUFDM0QsQ0FBQztZQUFDLE1BQU0sQ0FBQyxDQUFBLENBQUM7UUFDWixDQUFDO0lBQ0gsQ0FBQztJQUVELElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDakIsSUFBSSxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDbEIsTUFBTSxDQUFDLElBQUksR0FBRyxPQUFPLENBQUE7UUFDdkIsQ0FBQztJQUNILENBQUM7SUFFRCxPQUFPLE1BQU0sQ0FBQTtBQUNmLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBwcmVmZXRjaEZpbGUgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9kYXRhc2V0XCJcbmltcG9ydCB7IGluZmVyQnl0ZXMsIGluZmVyRW5jb2RpbmcsIGluZmVySGFzaCB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL2RhdGFzZXRcIlxuaW1wb3J0IHR5cGUgeyBSZXNvdXJjZSB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL21ldGFkYXRhXCJcbmltcG9ydCB7IGluZmVyRm9ybWF0LCBpbmZlck5hbWUgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9tZXRhZGF0YVwiXG5pbXBvcnQgdHlwZSB7IEluZmVyRGlhbGVjdE9wdGlvbnMgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy90YWJsZVwiXG5pbXBvcnQgdHlwZSB7IEluZmVyU2NoZW1hT3B0aW9ucyB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL3RhYmxlXCJcbmltcG9ydCB7IGluZmVyRGlhbGVjdCB9IGZyb20gXCIuLi9kaWFsZWN0L2luZGV4LnRzXCJcbmltcG9ydCB7IGluZmVyU2NoZW1hIH0gZnJvbSBcIi4uL3NjaGVtYS9pbmRleC50c1wiXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpbmZlclJlc291cmNlKFxuICByZXNvdXJjZTogUGFydGlhbDxSZXNvdXJjZT4sXG4gIG9wdGlvbnM/OiBJbmZlckRpYWxlY3RPcHRpb25zICYgSW5mZXJTY2hlbWFPcHRpb25zLFxuKSB7XG4gIGNvbnN0IHJlc3VsdCA9IHtcbiAgICAuLi5yZXNvdXJjZSxcbiAgICBuYW1lOiByZXNvdXJjZS5uYW1lID8/IGluZmVyTmFtZShyZXNvdXJjZSksXG4gIH1cblxuICBpZiAoIXJlc3VsdC5mb3JtYXQpIHtcbiAgICByZXN1bHQuZm9ybWF0ID0gaW5mZXJGb3JtYXQocmVzb3VyY2UpXG4gIH1cblxuICBpZiAodHlwZW9mIHJlc291cmNlLnBhdGggPT09IFwic3RyaW5nXCIpIHtcbiAgICBjb25zdCBsb2NhbFBhdGggPSBhd2FpdCBwcmVmZXRjaEZpbGUocmVzb3VyY2UucGF0aClcbiAgICBjb25zdCBsb2NhbFJlc291cmNlID0geyAuLi5yZXNvdXJjZSwgcGF0aDogbG9jYWxQYXRoIH1cblxuICAgIGlmICghcmVzdWx0LmVuY29kaW5nKSB7XG4gICAgICBjb25zdCBlbmNvZGluZyA9IGF3YWl0IGluZmVyRW5jb2RpbmcobG9jYWxSZXNvdXJjZSlcbiAgICAgIGlmIChlbmNvZGluZykge1xuICAgICAgICByZXN1bHQuZW5jb2RpbmcgPSBlbmNvZGluZ1xuICAgICAgfVxuICAgIH1cblxuICAgIGlmICghcmVzdWx0LmJ5dGVzKSB7XG4gICAgICByZXN1bHQuYnl0ZXMgPSBhd2FpdCBpbmZlckJ5dGVzKGxvY2FsUmVzb3VyY2UpXG4gICAgfVxuXG4gICAgaWYgKCFyZXN1bHQuaGFzaCkge1xuICAgICAgcmVzdWx0Lmhhc2ggPSBhd2FpdCBpbmZlckhhc2gobG9jYWxSZXNvdXJjZSlcbiAgICB9XG5cbiAgICBpZiAoIXJlc3VsdC5kaWFsZWN0KSB7XG4gICAgICB0cnkge1xuICAgICAgICByZXN1bHQuZGlhbGVjdCA9IGF3YWl0IGluZmVyRGlhbGVjdChsb2NhbFJlc291cmNlLCBvcHRpb25zKVxuICAgICAgfSBjYXRjaCB7fVxuICAgIH1cblxuICAgIGlmICghcmVzdWx0LnNjaGVtYSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgcmVzdWx0LnNjaGVtYSA9IGF3YWl0IGluZmVyU2NoZW1hKGxvY2FsUmVzb3VyY2UsIG9wdGlvbnMpXG4gICAgICB9IGNhdGNoIHt9XG4gICAgfVxuICB9XG5cbiAgaWYgKCFyZXN1bHQudHlwZSkge1xuICAgIGlmIChyZXN1bHQuc2NoZW1hKSB7XG4gICAgICByZXN1bHQudHlwZSA9IFwidGFibGVcIlxuICAgIH1cbiAgfVxuXG4gIHJldHVybiByZXN1bHRcbn1cbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import { writeTempFile } from "@frictionless-ts/dataset";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { inferResource } from "./infer.js";
|
|
4
|
+
describe("inferResource", () => {
|
|
5
|
+
it("should infer name from path", async () => {
|
|
6
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
7
|
+
const resource = { path: csvPath, format: "csv" };
|
|
8
|
+
const result = await inferResource(resource);
|
|
9
|
+
expect(result.name).toBeDefined();
|
|
10
|
+
expect(typeof result.name).toBe("string");
|
|
11
|
+
});
|
|
12
|
+
it("should infer bytes (file size)", async () => {
|
|
13
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
14
|
+
const resource = { path: csvPath, format: "csv" };
|
|
15
|
+
const result = await inferResource(resource);
|
|
16
|
+
expect(result.bytes).toBeGreaterThan(0);
|
|
17
|
+
});
|
|
18
|
+
it("should infer hash", async () => {
|
|
19
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
20
|
+
const resource = { path: csvPath, format: "csv" };
|
|
21
|
+
const result = await inferResource(resource);
|
|
22
|
+
expect(result.hash).toBeDefined();
|
|
23
|
+
expect(typeof result.hash).toBe("string");
|
|
24
|
+
expect(result.hash?.length).toBeGreaterThan(0);
|
|
25
|
+
});
|
|
26
|
+
it("should infer dialect for CSV files", async () => {
|
|
27
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
28
|
+
const resource = { path: csvPath, format: "csv" };
|
|
29
|
+
const result = await inferResource(resource);
|
|
30
|
+
expect(result.dialect).toBeDefined();
|
|
31
|
+
});
|
|
32
|
+
it("should infer schema for tabular data", async () => {
|
|
33
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
34
|
+
const resource = { path: csvPath, format: "csv" };
|
|
35
|
+
const result = await inferResource(resource);
|
|
36
|
+
expect(result.schema).toBeDefined();
|
|
37
|
+
expect(typeof result.schema === "object" &&
|
|
38
|
+
"fields" in result.schema &&
|
|
39
|
+
result.schema.fields).toBeDefined();
|
|
40
|
+
expect(typeof result.schema === "object" &&
|
|
41
|
+
"fields" in result.schema &&
|
|
42
|
+
result.schema.fields?.length).toBeGreaterThan(0);
|
|
43
|
+
});
|
|
44
|
+
it("should set type to table when schema is inferred", async () => {
|
|
45
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
46
|
+
const resource = { path: csvPath, format: "csv" };
|
|
47
|
+
const result = await inferResource(resource);
|
|
48
|
+
expect(result.type).toBe("table");
|
|
49
|
+
});
|
|
50
|
+
it("should preserve existing properties", async () => {
|
|
51
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
52
|
+
const resource = {
|
|
53
|
+
path: csvPath,
|
|
54
|
+
format: "csv",
|
|
55
|
+
name: "custom-name",
|
|
56
|
+
title: "Custom Title",
|
|
57
|
+
description: "Custom description",
|
|
58
|
+
};
|
|
59
|
+
const result = await inferResource(resource);
|
|
60
|
+
expect(result.name).toBe("custom-name");
|
|
61
|
+
expect(result.title).toBe("Custom Title");
|
|
62
|
+
expect(result.description).toBe("Custom description");
|
|
63
|
+
});
|
|
64
|
+
it("should not override existing format", async () => {
|
|
65
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
66
|
+
const resource = {
|
|
67
|
+
path: csvPath,
|
|
68
|
+
format: "json",
|
|
69
|
+
};
|
|
70
|
+
const result = await inferResource(resource);
|
|
71
|
+
expect(result.format).toBe("json");
|
|
72
|
+
});
|
|
73
|
+
it("should handle inline data without path", async () => {
|
|
74
|
+
const resource = {
|
|
75
|
+
name: "test-resource",
|
|
76
|
+
data: [
|
|
77
|
+
{ id: 1, name: "alice" },
|
|
78
|
+
{ id: 2, name: "bob" },
|
|
79
|
+
],
|
|
80
|
+
};
|
|
81
|
+
const result = await inferResource(resource);
|
|
82
|
+
expect(result.name).toBe("test-resource");
|
|
83
|
+
expect(result.data).toEqual(resource.data);
|
|
84
|
+
expect(result.bytes).toBeUndefined();
|
|
85
|
+
expect(result.hash).toBeUndefined();
|
|
86
|
+
});
|
|
87
|
+
it("should infer dialect with custom options", async () => {
|
|
88
|
+
const csvPath = await writeTempFile("id|name\n1|alice\n2|bob");
|
|
89
|
+
const resource = { path: csvPath, format: "csv" };
|
|
90
|
+
const result = await inferResource(resource, { delimiter: "|" });
|
|
91
|
+
expect(result.dialect).toBeDefined();
|
|
92
|
+
});
|
|
93
|
+
it("should infer schema with custom delimiter", async () => {
|
|
94
|
+
const csvPath = await writeTempFile("id|name\n1|alice\n2|bob");
|
|
95
|
+
const resource = { path: csvPath, format: "csv" };
|
|
96
|
+
const result = await inferResource(resource, { delimiter: "|" });
|
|
97
|
+
expect(result.schema).toBeDefined();
|
|
98
|
+
expect(typeof result.schema === "object" &&
|
|
99
|
+
"fields" in result.schema &&
|
|
100
|
+
result.schema.fields).toBeDefined();
|
|
101
|
+
expect(typeof result.schema === "object" &&
|
|
102
|
+
"fields" in result.schema &&
|
|
103
|
+
result.schema.fields?.length).toBe(2);
|
|
104
|
+
});
|
|
105
|
+
it("should not override existing encoding", async () => {
|
|
106
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
107
|
+
const resource = {
|
|
108
|
+
path: csvPath,
|
|
109
|
+
format: "csv",
|
|
110
|
+
encoding: "iso-8859-1",
|
|
111
|
+
};
|
|
112
|
+
const result = await inferResource(resource);
|
|
113
|
+
expect(result.encoding).toBe("iso-8859-1");
|
|
114
|
+
});
|
|
115
|
+
it("should not override existing bytes", async () => {
|
|
116
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
117
|
+
const resource = {
|
|
118
|
+
path: csvPath,
|
|
119
|
+
format: "csv",
|
|
120
|
+
bytes: 12345,
|
|
121
|
+
};
|
|
122
|
+
const result = await inferResource(resource);
|
|
123
|
+
expect(result.bytes).toBe(12345);
|
|
124
|
+
});
|
|
125
|
+
it("should not override existing hash", async () => {
|
|
126
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
127
|
+
const resource = {
|
|
128
|
+
path: csvPath,
|
|
129
|
+
format: "csv",
|
|
130
|
+
hash: "custom-hash",
|
|
131
|
+
};
|
|
132
|
+
const result = await inferResource(resource);
|
|
133
|
+
expect(result.hash).toBe("custom-hash");
|
|
134
|
+
});
|
|
135
|
+
it("should not override existing dialect", async () => {
|
|
136
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
137
|
+
const resource = {
|
|
138
|
+
path: csvPath,
|
|
139
|
+
format: "csv",
|
|
140
|
+
dialect: { delimiter: ";" },
|
|
141
|
+
};
|
|
142
|
+
const result = await inferResource(resource);
|
|
143
|
+
expect(typeof result.dialect === "object" &&
|
|
144
|
+
"delimiter" in result.dialect &&
|
|
145
|
+
result.dialect.delimiter).toBe(";");
|
|
146
|
+
});
|
|
147
|
+
it("should not override existing schema", async () => {
|
|
148
|
+
const csvPath = await writeTempFile("id,name\n1,alice\n2,bob");
|
|
149
|
+
const customSchema = {
|
|
150
|
+
fields: [
|
|
151
|
+
{ name: "custom1", type: "string" },
|
|
152
|
+
{ name: "custom2", type: "string" },
|
|
153
|
+
],
|
|
154
|
+
};
|
|
155
|
+
const resource = {
|
|
156
|
+
path: csvPath,
|
|
157
|
+
format: "csv",
|
|
158
|
+
schema: customSchema,
|
|
159
|
+
};
|
|
160
|
+
const result = await inferResource(resource);
|
|
161
|
+
expect(result.schema).toEqual(customSchema);
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infer.spec.js","sourceRoot":"","sources":["../../resource/infer.spec.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAA;AACxD,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAC7C,OAAO,EAAE,aAAa,EAAE,MAAM,YAAY,CAAA;AAE1C,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,EAAE,CAAC,6BAA6B,EAAE,KAAK,IAAI,EAAE;QAC3C,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,WAAW,EAAE,CAAA;QACjC,MAAM,CAAC,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;IAC3C,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,gCAAgC,EAAE,KAAK,IAAI,EAAE;QAC9C,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;IACzC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,mBAAmB,EAAE,KAAK,IAAI,EAAE;QACjC,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,WAAW,EAAE,CAAA;QACjC,MAAM,CAAC,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QACzC,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;IAChD,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAA;IACtC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;QACpD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QACnC,MAAM,CACJ,OAAO,MAAM,CAAC,MAAM,KAAK,QAAQ;YAC/B,QAAQ,IAAI,MAAM,CAAC,MAAM;YACzB,MAAM,CAAC,MAAM,CAAC,MAAM,CACvB,CAAC,WAAW,EAAE,CAAA;QACf,MAAM,CACJ,OAAO,MAAM,CAAC,MAAM,KAAK,QAAQ;YAC/B,QAAQ,IAAI,MAAM,CAAC,MAAM;YACzB,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAC/B,CAAC,eAAe,CAAC,CAAC,CAAC,CAAA;IACtB,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,kDAAkD,EAAE,KAAK,IAAI,EAAE;QAChE,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;IACnC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;QACnD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,cAAc;YACrB,WAAW,EAAE,oBAAoB;SAClC,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;QACvC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAA;QACzC,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAA;IACvD,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;QACnD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,MAAe;SACxB,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;IACpC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,wCAAwC,EAAE,KAAK,IAAI,EAAE;QACtD,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,eAAe;YACrB,IAAI,EAAE;gBACJ,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE;gBACxB,EAAE,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE;aACvB;SACF,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAA;QACzC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAA;QAC1C,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,aAAa,EAAE,CAAA;QACpC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,aAAa,EAAE,CAAA;IACrC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,0CAA0C,EAAE,KAAK,IAAI,EAAE;QACxD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,GAAG,EAAE,CAAC,CAAA;QAEhE,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAA;IACtC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,2CAA2C,EAAE,KAAK,IAAI,EAAE;QACzD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,KAAc,EAAE,CAAA;QAE1D,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,GAAG,EAAE,CAAC,CAAA;QAEhE,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAA;QACnC,MAAM,CACJ,OAAO,MAAM,CAAC,MAAM,KAAK,QAAQ;YAC/B,QAAQ,IAAI,MAAM,CAAC,MAAM;YACzB,MAAM,CAAC,MAAM,CAAC,MAAM,CACvB,CAAC,WAAW,EAAE,CAAA;QACf,MAAM,CACJ,OAAO,MAAM,CAAC,MAAM,KAAK,QAAQ;YAC/B,QAAQ,IAAI,MAAM,CAAC,MAAM;YACzB,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAC/B,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACX,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,uCAAuC,EAAE,KAAK,IAAI,EAAE;QACrD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,QAAQ,EAAE,YAAY;SACvB,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAA;IAC5C,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,KAAK,EAAE,KAAK;SACb,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAClC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,mCAAmC,EAAE,KAAK,IAAI,EAAE;QACjD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,IAAI,EAAE,aAAa;SACpB,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;IACzC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;QACpD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,OAAO,EAAE,EAAE,SAAS,EAAE,GAAG,EAAE;SAC5B,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CACJ,OAAO,MAAM,CAAC,OAAO,KAAK,QAAQ;YAChC,WAAW,IAAI,MAAM,CAAC,OAAO;YAC7B,MAAM,CAAC,OAAO,CAAC,SAAS,CAC3B,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;IACb,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;QACnD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,yBAAyB,CAAC,CAAA;QAC9D,MAAM,YAAY,GAAG;YACnB,MAAM,EAAE;gBACN,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAiB,EAAE;gBAC5C,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAiB,EAAE;aAC7C;SACF,CAAA;QACD,MAAM,QAAQ,GAAG;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE,KAAc;YACtB,MAAM,EAAE,YAAY;SACrB,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC,CAAA;QAE5C,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAA;IAC7C,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA","sourcesContent":["import { writeTempFile } from \"@frictionless-ts/dataset\"\nimport { describe, expect, it } from \"vitest\"\nimport { inferResource } from \"./infer.ts\"\n\ndescribe(\"inferResource\", () => {\n  it(\"should infer name from path\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.name).toBeDefined()\n    expect(typeof result.name).toBe(\"string\")\n  })\n\n  it(\"should infer bytes (file size)\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.bytes).toBeGreaterThan(0)\n  })\n\n  it(\"should infer hash\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.hash).toBeDefined()\n    expect(typeof result.hash).toBe(\"string\")\n    expect(result.hash?.length).toBeGreaterThan(0)\n  })\n\n  it(\"should infer dialect for CSV files\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.dialect).toBeDefined()\n  })\n\n  it(\"should infer schema for tabular data\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.schema).toBeDefined()\n    expect(\n      typeof result.schema === \"object\" &&\n        \"fields\" in result.schema &&\n        result.schema.fields,\n    ).toBeDefined()\n    expect(\n      typeof result.schema === \"object\" &&\n        \"fields\" in result.schema &&\n        result.schema.fields?.length,\n    ).toBeGreaterThan(0)\n  })\n\n  it(\"should set type to table when schema is inferred\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource)\n\n    expect(result.type).toBe(\"table\")\n  })\n\n  it(\"should preserve existing properties\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      name: \"custom-name\",\n      title: \"Custom Title\",\n      description: \"Custom description\",\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.name).toBe(\"custom-name\")\n    expect(result.title).toBe(\"Custom Title\")\n    expect(result.description).toBe(\"Custom description\")\n  })\n\n  it(\"should not override existing format\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"json\" as const,\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.format).toBe(\"json\")\n  })\n\n  it(\"should handle inline data without path\", async () => {\n    const resource = {\n      name: \"test-resource\",\n      data: [\n        { id: 1, name: \"alice\" },\n        { id: 2, name: \"bob\" },\n      ],\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.name).toBe(\"test-resource\")\n    expect(result.data).toEqual(resource.data)\n    expect(result.bytes).toBeUndefined()\n    expect(result.hash).toBeUndefined()\n  })\n\n  it(\"should infer dialect with custom options\", async () => {\n    const csvPath = await writeTempFile(\"id|name\\n1|alice\\n2|bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource, { delimiter: \"|\" })\n\n    expect(result.dialect).toBeDefined()\n  })\n\n  it(\"should infer schema with custom delimiter\", async () => {\n    const csvPath = await writeTempFile(\"id|name\\n1|alice\\n2|bob\")\n    const resource = { path: csvPath, format: \"csv\" as const }\n\n    const result = await inferResource(resource, { delimiter: \"|\" })\n\n    expect(result.schema).toBeDefined()\n    expect(\n      typeof result.schema === \"object\" &&\n        \"fields\" in result.schema &&\n        result.schema.fields,\n    ).toBeDefined()\n    expect(\n      typeof result.schema === \"object\" &&\n        \"fields\" in result.schema &&\n        result.schema.fields?.length,\n    ).toBe(2)\n  })\n\n  it(\"should not override existing encoding\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      encoding: \"iso-8859-1\",\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.encoding).toBe(\"iso-8859-1\")\n  })\n\n  it(\"should not override existing bytes\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      bytes: 12345,\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.bytes).toBe(12345)\n  })\n\n  it(\"should not override existing hash\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      hash: \"custom-hash\",\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.hash).toBe(\"custom-hash\")\n  })\n\n  it(\"should not override existing dialect\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      dialect: { delimiter: \";\" },\n    }\n\n    const result = await inferResource(resource)\n\n    expect(\n      typeof result.dialect === \"object\" &&\n        \"delimiter\" in result.dialect &&\n        result.dialect.delimiter,\n    ).toBe(\";\")\n  })\n\n  it(\"should not override existing schema\", async () => {\n    const csvPath = await writeTempFile(\"id,name\\n1,alice\\n2,bob\")\n    const customSchema = {\n      fields: [\n        { name: \"custom1\", type: \"string\" as const },\n        { name: \"custom2\", type: \"string\" as const },\n      ],\n    }\n    const resource = {\n      path: csvPath,\n      format: \"csv\" as const,\n      schema: customSchema,\n    }\n\n    const result = await inferResource(resource)\n\n    expect(result.schema).toEqual(customSchema)\n  })\n})\n"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { Descriptor, Resource } from "@frictionless-ts/metadata";
|
|
2
|
+
import type { InferSchemaOptions } from "@frictionless-ts/table";
|
|
3
|
+
export declare function validateResource(source: string | Descriptor | Partial<Resource>, options?: InferSchemaOptions & {
|
|
4
|
+
basepath?: string;
|
|
5
|
+
}): Promise<{
|
|
6
|
+
errors: import("@frictionless-ts/metadata").UnboundError[];
|
|
7
|
+
valid: boolean;
|
|
8
|
+
}>;
|
|
9
|
+
export declare function validateResourceData(resource: Partial<Resource>, options?: InferSchemaOptions): Promise<{
|
|
10
|
+
errors: import("@frictionless-ts/metadata").UnboundError[];
|
|
11
|
+
valid: boolean;
|
|
12
|
+
}>;
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { validateFile } from "@frictionless-ts/dataset";
|
|
2
|
+
import { validateDocument } from "@frictionless-ts/document";
|
|
3
|
+
import { createReport } from "@frictionless-ts/metadata";
|
|
4
|
+
import { loadDescriptor, validateResourceMetadata, } from "@frictionless-ts/metadata";
|
|
5
|
+
import { resolveBasepath } from "@frictionless-ts/metadata";
|
|
6
|
+
import { validateTable } from "../table/index.js";
|
|
7
|
+
export async function validateResource(source, options) {
|
|
8
|
+
let descriptor = source;
|
|
9
|
+
let basepath = options?.basepath;
|
|
10
|
+
if (typeof descriptor === "string") {
|
|
11
|
+
basepath = await resolveBasepath(descriptor);
|
|
12
|
+
descriptor = await loadDescriptor(descriptor);
|
|
13
|
+
}
|
|
14
|
+
const report = await validateResourceMetadata(descriptor, { basepath });
|
|
15
|
+
if (!report.resource) {
|
|
16
|
+
return report;
|
|
17
|
+
}
|
|
18
|
+
return await validateResourceData(report.resource, options);
|
|
19
|
+
}
|
|
20
|
+
export async function validateResourceData(resource, options) {
|
|
21
|
+
const fileReport = await validateFile(resource);
|
|
22
|
+
if (!fileReport.valid) {
|
|
23
|
+
return fileReport;
|
|
24
|
+
}
|
|
25
|
+
const tableReport = await validateTable(resource, options);
|
|
26
|
+
if (!tableReport.valid) {
|
|
27
|
+
return tableReport;
|
|
28
|
+
}
|
|
29
|
+
const documentReport = await validateDocument(resource);
|
|
30
|
+
if (!documentReport.valid) {
|
|
31
|
+
return documentReport;
|
|
32
|
+
}
|
|
33
|
+
return createReport();
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9yZXNvdXJjZS92YWxpZGF0ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMEJBQTBCLENBQUE7QUFDdkQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sMkJBQTJCLENBQUE7QUFFNUQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFBO0FBQ3hELE9BQU8sRUFDTCxjQUFjLEVBQ2Qsd0JBQXdCLEdBQ3pCLE1BQU0sMkJBQTJCLENBQUE7QUFDbEMsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDJCQUEyQixDQUFBO0FBRTNELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQTtBQUVqRCxNQUFNLENBQUMsS0FBSyxVQUFVLGdCQUFnQixDQUNwQyxNQUErQyxFQUMvQyxPQUFvRDtJQUVwRCxJQUFJLFVBQVUsR0FBRyxNQUFNLENBQUE7SUFDdkIsSUFBSSxRQUFRLEdBQUcsT0FBTyxFQUFFLFFBQVEsQ0FBQTtJQUVoQyxJQUFJLE9BQU8sVUFBVSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ25DLFFBQVEsR0FBRyxNQUFNLGVBQWUsQ0FBQyxVQUFVLENBQUMsQ0FBQTtRQUM1QyxVQUFVLEdBQUcsTUFBTSxjQUFjLENBQUMsVUFBVSxDQUFDLENBQUE7SUFDL0MsQ0FBQztJQUVELE1BQU0sTUFBTSxHQUFHLE1BQU0sd0JBQXdCLENBQUMsVUFBVSxFQUFFLEVBQUUsUUFBUSxFQUFFLENBQUMsQ0FBQTtJQUV2RSxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ3JCLE9BQU8sTUFBTSxDQUFBO0lBQ2YsQ0FBQztJQUVELE9BQU8sTUFBTSxvQkFBb0IsQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLE9BQU8sQ0FBQyxDQUFBO0FBQzdELENBQUM7QUFFRCxNQUFNLENBQUMsS0FBSyxVQUFVLG9CQUFvQixDQUN4QyxRQUEyQixFQUMzQixPQUE0QjtJQUU1QixNQUFNLFVBQVUsR0FBRyxNQUFNLFlBQVksQ0FBQyxRQUFRLENBQUMsQ0FBQTtJQUMvQyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ3RCLE9BQU8sVUFBVSxDQUFBO0lBQ25CLENBQUM7SUFFRCxNQUFNLFdBQVcsR0FBRyxNQUFNLGFBQWEsQ0FBQyxRQUFRLEVBQUUsT0FBTyxDQUFDLENBQUE7SUFDMUQsSUFBSSxDQUFDLFdBQVcsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUN2QixPQUFPLFdBQVcsQ0FBQTtJQUNwQixDQUFDO0lBRUQsTUFBTSxjQUFjLEdBQUcsTUFBTSxnQkFBZ0IsQ0FBQyxRQUFRLENBQUMsQ0FBQTtJQUN2RCxJQUFJLENBQUMsY0FBYyxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQzFCLE9BQU8sY0FBYyxDQUFBO0lBQ3ZCLENBQUM7SUFFRCxPQUFPLFlBQVksRUFBRSxDQUFBO0FBQ3ZCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB2YWxpZGF0ZUZpbGUgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9kYXRhc2V0XCJcbmltcG9ydCB7IHZhbGlkYXRlRG9jdW1lbnQgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy9kb2N1bWVudFwiXG5pbXBvcnQgdHlwZSB7IERlc2NyaXB0b3IsIFJlc291cmNlIH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvbWV0YWRhdGFcIlxuaW1wb3J0IHsgY3JlYXRlUmVwb3J0IH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvbWV0YWRhdGFcIlxuaW1wb3J0IHtcbiAgbG9hZERlc2NyaXB0b3IsXG4gIHZhbGlkYXRlUmVzb3VyY2VNZXRhZGF0YSxcbn0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvbWV0YWRhdGFcIlxuaW1wb3J0IHsgcmVzb2x2ZUJhc2VwYXRoIH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvbWV0YWRhdGFcIlxuaW1wb3J0IHR5cGUgeyBJbmZlclNjaGVtYU9wdGlvbnMgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy90YWJsZVwiXG5pbXBvcnQgeyB2YWxpZGF0ZVRhYmxlIH0gZnJvbSBcIi4uL3RhYmxlL2luZGV4LnRzXCJcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHZhbGlkYXRlUmVzb3VyY2UoXG4gIHNvdXJjZTogc3RyaW5nIHwgRGVzY3JpcHRvciB8IFBhcnRpYWw8UmVzb3VyY2U+LFxuICBvcHRpb25zPzogSW5mZXJTY2hlbWFPcHRpb25zICYgeyBiYXNlcGF0aD86IHN0cmluZyB9LFxuKSB7XG4gIGxldCBkZXNjcmlwdG9yID0gc291cmNlXG4gIGxldCBiYXNlcGF0aCA9IG9wdGlvbnM/LmJhc2VwYXRoXG5cbiAgaWYgKHR5cGVvZiBkZXNjcmlwdG9yID09PSBcInN0cmluZ1wiKSB7XG4gICAgYmFzZXBhdGggPSBhd2FpdCByZXNvbHZlQmFzZXBhdGgoZGVzY3JpcHRvcilcbiAgICBkZXNjcmlwdG9yID0gYXdhaXQgbG9hZERlc2NyaXB0b3IoZGVzY3JpcHRvcilcbiAgfVxuXG4gIGNvbnN0IHJlcG9ydCA9IGF3YWl0IHZhbGlkYXRlUmVzb3VyY2VNZXRhZGF0YShkZXNjcmlwdG9yLCB7IGJhc2VwYXRoIH0pXG5cbiAgaWYgKCFyZXBvcnQucmVzb3VyY2UpIHtcbiAgICByZXR1cm4gcmVwb3J0XG4gIH1cblxuICByZXR1cm4gYXdhaXQgdmFsaWRhdGVSZXNvdXJjZURhdGEocmVwb3J0LnJlc291cmNlLCBvcHRpb25zKVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gdmFsaWRhdGVSZXNvdXJjZURhdGEoXG4gIHJlc291cmNlOiBQYXJ0aWFsPFJlc291cmNlPixcbiAgb3B0aW9ucz86IEluZmVyU2NoZW1hT3B0aW9ucyxcbikge1xuICBjb25zdCBmaWxlUmVwb3J0ID0gYXdhaXQgdmFsaWRhdGVGaWxlKHJlc291cmNlKVxuICBpZiAoIWZpbGVSZXBvcnQudmFsaWQpIHtcbiAgICByZXR1cm4gZmlsZVJlcG9ydFxuICB9XG5cbiAgY29uc3QgdGFibGVSZXBvcnQgPSBhd2FpdCB2YWxpZGF0ZVRhYmxlKHJlc291cmNlLCBvcHRpb25zKVxuICBpZiAoIXRhYmxlUmVwb3J0LnZhbGlkKSB7XG4gICAgcmV0dXJuIHRhYmxlUmVwb3J0XG4gIH1cblxuICBjb25zdCBkb2N1bWVudFJlcG9ydCA9IGF3YWl0IHZhbGlkYXRlRG9jdW1lbnQocmVzb3VyY2UpXG4gIGlmICghZG9jdW1lbnRSZXBvcnQudmFsaWQpIHtcbiAgICByZXR1cm4gZG9jdW1lbnRSZXBvcnRcbiAgfVxuXG4gIHJldHVybiBjcmVhdGVSZXBvcnQoKVxufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { validateResource } from "./validate.js";
|
|
3
|
+
describe("validateResource", () => {
|
|
4
|
+
it("should catch validation errors for invalid tabular data", async () => {
|
|
5
|
+
const resource = {
|
|
6
|
+
name: "test",
|
|
7
|
+
type: "table",
|
|
8
|
+
data: [
|
|
9
|
+
{ id: 1, name: "Alice", active: true },
|
|
10
|
+
{ id: 2, name: "Bob", active: 123 },
|
|
11
|
+
],
|
|
12
|
+
schema: {
|
|
13
|
+
fields: [
|
|
14
|
+
{ name: "id", type: "integer" },
|
|
15
|
+
{ name: "name", type: "string" },
|
|
16
|
+
{ name: "active", type: "boolean" },
|
|
17
|
+
],
|
|
18
|
+
},
|
|
19
|
+
};
|
|
20
|
+
const report = await validateResource(resource);
|
|
21
|
+
expect(report.valid).toBe(false);
|
|
22
|
+
expect(report.errors.length).toEqual(1);
|
|
23
|
+
});
|
|
24
|
+
it("should validate correct tabular data", async () => {
|
|
25
|
+
const resource = {
|
|
26
|
+
name: "test",
|
|
27
|
+
type: "table",
|
|
28
|
+
data: [
|
|
29
|
+
{ name: "Alice", active: true },
|
|
30
|
+
{ name: "Bob", active: false },
|
|
31
|
+
],
|
|
32
|
+
schema: {
|
|
33
|
+
fields: [
|
|
34
|
+
{ name: "name", type: "string" },
|
|
35
|
+
{ name: "active", type: "boolean" },
|
|
36
|
+
],
|
|
37
|
+
},
|
|
38
|
+
};
|
|
39
|
+
const report = await validateResource(resource);
|
|
40
|
+
expect(report.valid).toBe(true);
|
|
41
|
+
expect(report.errors).toEqual([]);
|
|
42
|
+
});
|
|
43
|
+
it("should catch multiple validation errors", async () => {
|
|
44
|
+
const resource = {
|
|
45
|
+
name: "test",
|
|
46
|
+
type: "table",
|
|
47
|
+
data: [
|
|
48
|
+
{ id: 1, name: "Alice", age: 25 },
|
|
49
|
+
{ id: "not-a-number", name: "Bob", age: "not-a-number" },
|
|
50
|
+
{ id: 3, name: "Charlie", age: -5 },
|
|
51
|
+
],
|
|
52
|
+
schema: {
|
|
53
|
+
fields: [
|
|
54
|
+
{ name: "id", type: "integer" },
|
|
55
|
+
{ name: "name", type: "string" },
|
|
56
|
+
{
|
|
57
|
+
name: "age",
|
|
58
|
+
type: "integer",
|
|
59
|
+
constraints: { minimum: 0 },
|
|
60
|
+
},
|
|
61
|
+
],
|
|
62
|
+
},
|
|
63
|
+
};
|
|
64
|
+
const report = await validateResource(resource);
|
|
65
|
+
expect(report.valid).toBe(false);
|
|
66
|
+
expect(report.errors.length).toEqual(3);
|
|
67
|
+
});
|
|
68
|
+
it("should catch missing table error when schema is defined but table cannot be loaded", async () => {
|
|
69
|
+
const resource = {
|
|
70
|
+
name: "test",
|
|
71
|
+
path: "table.bad",
|
|
72
|
+
schema: {
|
|
73
|
+
fields: [
|
|
74
|
+
{ name: "id", type: "integer" },
|
|
75
|
+
{ name: "name", type: "string" },
|
|
76
|
+
],
|
|
77
|
+
},
|
|
78
|
+
};
|
|
79
|
+
const report = await validateResource(resource);
|
|
80
|
+
expect(report.valid).toBe(false);
|
|
81
|
+
expect(report.errors).toEqual([
|
|
82
|
+
{
|
|
83
|
+
type: "data",
|
|
84
|
+
message: "missing test table",
|
|
85
|
+
},
|
|
86
|
+
]);
|
|
87
|
+
});
|
|
88
|
+
it("should validate document with jsonSchema", async () => {
|
|
89
|
+
const resource = {
|
|
90
|
+
name: "test-document",
|
|
91
|
+
data: {
|
|
92
|
+
name: "test-package",
|
|
93
|
+
version: "1.0.0",
|
|
94
|
+
author: {
|
|
95
|
+
name: "John Doe",
|
|
96
|
+
email: "john@example.com",
|
|
97
|
+
},
|
|
98
|
+
},
|
|
99
|
+
jsonSchema: {
|
|
100
|
+
type: "object",
|
|
101
|
+
required: ["name", "version", "author"],
|
|
102
|
+
properties: {
|
|
103
|
+
name: { type: "string" },
|
|
104
|
+
version: { type: "string" },
|
|
105
|
+
author: {
|
|
106
|
+
type: "object",
|
|
107
|
+
required: ["name", "email"],
|
|
108
|
+
properties: {
|
|
109
|
+
name: { type: "string" },
|
|
110
|
+
email: { type: "string" },
|
|
111
|
+
},
|
|
112
|
+
},
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
};
|
|
116
|
+
const report = await validateResource(resource);
|
|
117
|
+
expect(report.valid).toBe(true);
|
|
118
|
+
expect(report.errors).toEqual([]);
|
|
119
|
+
});
|
|
120
|
+
it("should catch validation errors for document with invalid jsonSchema data", async () => {
|
|
121
|
+
const resource = {
|
|
122
|
+
name: "test-document",
|
|
123
|
+
data: {
|
|
124
|
+
name: "test-package",
|
|
125
|
+
version: 123,
|
|
126
|
+
},
|
|
127
|
+
jsonSchema: {
|
|
128
|
+
type: "object",
|
|
129
|
+
required: ["name", "version"],
|
|
130
|
+
properties: {
|
|
131
|
+
name: { type: "string" },
|
|
132
|
+
version: { type: "string" },
|
|
133
|
+
},
|
|
134
|
+
},
|
|
135
|
+
};
|
|
136
|
+
const report = await validateResource(resource);
|
|
137
|
+
expect(report.valid).toBe(false);
|
|
138
|
+
expect(report.errors).toEqual([
|
|
139
|
+
{
|
|
140
|
+
type: "document/json",
|
|
141
|
+
pointer: "/version",
|
|
142
|
+
message: "must be string",
|
|
143
|
+
},
|
|
144
|
+
]);
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validate.spec.js","sourceRoot":"","sources":["../../resource/validate.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { describe, expect, it } from \"vitest\"\nimport { validateResource } from \"./validate.ts\"\n\ndescribe(\"validateResource\", () => {\n  it(\"should catch validation errors for invalid tabular data\", async () => {\n    const resource = {\n      name: \"test\",\n      type: \"table\" as const,\n      data: [\n        { id: 1, name: \"Alice\", active: true },\n        { id: 2, name: \"Bob\", active: 123 },\n      ],\n      schema: {\n        fields: [\n          { name: \"id\", type: \"integer\" as const },\n          { name: \"name\", type: \"string\" as const },\n          { name: \"active\", type: \"boolean\" as const },\n        ],\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors.length).toEqual(1)\n  })\n\n  it(\"should validate correct tabular data\", async () => {\n    const resource = {\n      name: \"test\",\n      type: \"table\" as const,\n      data: [\n        { name: \"Alice\", active: true },\n        { name: \"Bob\", active: false },\n      ],\n      schema: {\n        fields: [\n          { name: \"name\", type: \"string\" as const },\n          { name: \"active\", type: \"boolean\" as const },\n        ],\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(true)\n    expect(report.errors).toEqual([])\n  })\n\n  it(\"should catch multiple validation errors\", async () => {\n    const resource = {\n      name: \"test\",\n      type: \"table\" as const,\n      data: [\n        { id: 1, name: \"Alice\", age: 25 },\n        { id: \"not-a-number\", name: \"Bob\", age: \"not-a-number\" },\n        { id: 3, name: \"Charlie\", age: -5 },\n      ],\n      schema: {\n        fields: [\n          { name: \"id\", type: \"integer\" as const },\n          { name: \"name\", type: \"string\" as const },\n          {\n            name: \"age\",\n            type: \"integer\" as const,\n            constraints: { minimum: 0 },\n          },\n        ],\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors.length).toEqual(3)\n  })\n\n  it(\"should catch missing table error when schema is defined but table cannot be loaded\", async () => {\n    const resource = {\n      name: \"test\",\n      path: \"table.bad\",\n      schema: {\n        fields: [\n          { name: \"id\", type: \"integer\" as const },\n          { name: \"name\", type: \"string\" as const },\n        ],\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors).toEqual([\n      {\n        type: \"data\",\n        message: \"missing test table\",\n      },\n    ])\n  })\n\n  it(\"should validate document with jsonSchema\", async () => {\n    const resource = {\n      name: \"test-document\",\n      data: {\n        name: \"test-package\",\n        version: \"1.0.0\",\n        author: {\n          name: \"John Doe\",\n          email: \"john@example.com\",\n        },\n      },\n      jsonSchema: {\n        type: \"object\",\n        required: [\"name\", \"version\", \"author\"],\n        properties: {\n          name: { type: \"string\" },\n          version: { type: \"string\" },\n          author: {\n            type: \"object\",\n            required: [\"name\", \"email\"],\n            properties: {\n              name: { type: \"string\" },\n              email: { type: \"string\" },\n            },\n          },\n        },\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(true)\n    expect(report.errors).toEqual([])\n  })\n\n  it(\"should catch validation errors for document with invalid jsonSchema data\", async () => {\n    const resource = {\n      name: \"test-document\",\n      data: {\n        name: \"test-package\",\n        version: 123,\n      },\n      jsonSchema: {\n        type: \"object\",\n        required: [\"name\", \"version\"],\n        properties: {\n          name: { type: \"string\" },\n          version: { type: \"string\" },\n        },\n      },\n    }\n\n    const report = await validateResource(resource)\n\n    expect(report.valid).toBe(false)\n    expect(report.errors).toEqual([\n      {\n        type: \"document/json\",\n        pointer: \"/version\",\n        message: \"must be string\",\n      },\n    ])\n  })\n})\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { inferSchema } from "./infer.ts";
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export { inferSchema } from "./infer.js";
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zY2hlbWEvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLFlBQVksQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IGluZmVyU2NoZW1hIH0gZnJvbSBcIi4vaW5mZXIudHNcIlxuIl19
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
import type { Resource } from "@frictionless-ts/metadata";
|
|
2
|
+
import type { InferSchemaOptions } from "@frictionless-ts/table";
|
|
3
|
+
export declare function inferSchema(resource: Partial<Resource>, options?: InferSchemaOptions): Promise<import("@frictionless-ts/metadata").Schema | undefined>;
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { inferSchemaFromTable } from "@frictionless-ts/table";
|
|
2
|
+
import { system } from "../system.js";
|
|
3
|
+
import { loadTable } from "../table/index.js";
|
|
4
|
+
export async function inferSchema(resource, options) {
|
|
5
|
+
for (const plugin of system.plugins) {
|
|
6
|
+
const schema = await plugin.inferSchema?.(resource, options);
|
|
7
|
+
if (schema) {
|
|
8
|
+
return schema;
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
const table = await loadTable(resource, { denormalized: true });
|
|
12
|
+
if (!table) {
|
|
13
|
+
return undefined;
|
|
14
|
+
}
|
|
15
|
+
const schema = await inferSchemaFromTable(table, options);
|
|
16
|
+
return schema;
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5mZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zY2hlbWEvaW5mZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0JBQXdCLENBQUE7QUFDN0QsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGNBQWMsQ0FBQTtBQUNyQyxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sbUJBQW1CLENBQUE7QUFFN0MsTUFBTSxDQUFDLEtBQUssVUFBVSxXQUFXLENBQy9CLFFBQTJCLEVBQzNCLE9BQTRCO0lBRTVCLEtBQUssTUFBTSxNQUFNLElBQUksTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3BDLE1BQU0sTUFBTSxHQUFHLE1BQU0sTUFBTSxDQUFDLFdBQVcsRUFBRSxDQUFDLFFBQVEsRUFBRSxPQUFPLENBQUMsQ0FBQTtRQUM1RCxJQUFJLE1BQU0sRUFBRSxDQUFDO1lBQ1gsT0FBTyxNQUFNLENBQUE7UUFDZixDQUFDO0lBQ0gsQ0FBQztJQUVELE1BQU0sS0FBSyxHQUFHLE1BQU0sU0FBUyxDQUFDLFFBQVEsRUFBRSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFBO0lBQy9ELElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUNYLE9BQU8sU0FBUyxDQUFBO0lBQ2xCLENBQUM7SUFFRCxNQUFNLE1BQU0sR0FBRyxNQUFNLG9CQUFvQixDQUFDLEtBQUssRUFBRSxPQUFPLENBQUMsQ0FBQTtJQUN6RCxPQUFPLE1BQU0sQ0FBQTtBQUNmLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFJlc291cmNlIH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvbWV0YWRhdGFcIlxuaW1wb3J0IHR5cGUgeyBJbmZlclNjaGVtYU9wdGlvbnMgfSBmcm9tIFwiQGZyaWN0aW9ubGVzcy10cy90YWJsZVwiXG5pbXBvcnQgeyBpbmZlclNjaGVtYUZyb21UYWJsZSB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL3RhYmxlXCJcbmltcG9ydCB7IHN5c3RlbSB9IGZyb20gXCIuLi9zeXN0ZW0udHNcIlxuaW1wb3J0IHsgbG9hZFRhYmxlIH0gZnJvbSBcIi4uL3RhYmxlL2luZGV4LnRzXCJcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGluZmVyU2NoZW1hKFxuICByZXNvdXJjZTogUGFydGlhbDxSZXNvdXJjZT4sXG4gIG9wdGlvbnM/OiBJbmZlclNjaGVtYU9wdGlvbnMsXG4pIHtcbiAgZm9yIChjb25zdCBwbHVnaW4gb2Ygc3lzdGVtLnBsdWdpbnMpIHtcbiAgICBjb25zdCBzY2hlbWEgPSBhd2FpdCBwbHVnaW4uaW5mZXJTY2hlbWE/LihyZXNvdXJjZSwgb3B0aW9ucylcbiAgICBpZiAoc2NoZW1hKSB7XG4gICAgICByZXR1cm4gc2NoZW1hXG4gICAgfVxuICB9XG5cbiAgY29uc3QgdGFibGUgPSBhd2FpdCBsb2FkVGFibGUocmVzb3VyY2UsIHsgZGVub3JtYWxpemVkOiB0cnVlIH0pXG4gIGlmICghdGFibGUpIHtcbiAgICByZXR1cm4gdW5kZWZpbmVkXG4gIH1cblxuICBjb25zdCBzY2hlbWEgPSBhd2FpdCBpbmZlclNjaGVtYUZyb21UYWJsZSh0YWJsZSwgb3B0aW9ucylcbiAgcmV0dXJuIHNjaGVtYVxufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|