@dpkit/table
Version: 
Data Package implementation in TypeScript.
202 lines • 24.7 kB
JavaScript
import { DataFrame } from "nodejs-polars";
import { describe, expect, it } from "vitest";
import { normalizeTable } from "./normalize.js";
describe("normalizeTable", () => {
    it("should work with schema", async () => {
        const table = DataFrame({
            id: [1, 2],
            name: ["english", "中文"],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work with less fields in data", async () => {
        const table = DataFrame({
            id: [1, 2],
            name: ["english", "中文"],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
                { name: "other", type: "boolean" },
            ],
        };
        const records = [
            { id: 1, name: "english", other: null },
            { id: 2, name: "中文", other: null },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work with more fields in data", async () => {
        const table = DataFrame({
            id: [1, 2],
            name: ["english", "中文"],
            other: [true, false],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work based on fields order", async () => {
        const table = DataFrame({
            field1: [1, 2],
            field2: ["english", "中文"],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work based on field names (equal)", async () => {
        const table = DataFrame({
            name: ["english", "中文"],
            id: [1, 2],
        }).lazy();
        const schema = {
            fieldsMatch: "equal",
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work based on field names (subset)", async () => {
        const table = DataFrame({
            name: ["english", "中文"],
            id: [1, 2],
        }).lazy();
        const schema = {
            fieldsMatch: "subset",
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work based on field names (superset)", async () => {
        const table = DataFrame({
            name: ["english", "中文"],
            id: [1, 2],
        }).lazy();
        const schema = {
            fieldsMatch: "superset",
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should work based on field names (partial)", async () => {
        const table = DataFrame({
            name: ["english", "中文"],
            id: [1, 2],
        }).lazy();
        const schema = {
            fieldsMatch: "partial",
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should parse string columns", async () => {
        const table = DataFrame({
            id: ["1", "2"],
            name: ["english", "中文"],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "string" },
            ],
        };
        const records = [
            { id: 1, name: "english" },
            { id: 2, name: "中文" },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
    it("should read type errors as nulls", async () => {
        const table = DataFrame({
            id: [1, 2],
            name: ["english", "中文"],
        }).lazy();
        const schema = {
            fields: [
                { name: "id", type: "integer" },
                { name: "name", type: "integer" },
            ],
        };
        const records = [
            { id: 1, name: null },
            { id: 2, name: null },
        ];
        const ldf = await normalizeTable(table, schema);
        const df = await ldf.collect();
        expect(df.toRecords()).toEqual(records);
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"normalize.spec.js","sourceRoot":"","sources":["../../table/normalize.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Schema } from \"@dpkit/core\"\nimport { DataFrame } from \"nodejs-polars\"\nimport { describe, expect, it } from \"vitest\"\nimport { normalizeTable } from \"./normalize.ts\"\n\ndescribe(\"normalizeTable\", () => {\n  it(\"should work with schema\", async () => {\n    const table = DataFrame({\n      id: [1, 2],\n      name: [\"english\", \"中文\"],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work with less fields in data\", async () => {\n    const table = DataFrame({\n      id: [1, 2],\n      name: [\"english\", \"中文\"],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n        { name: \"other\", type: \"boolean\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\", other: null },\n      { id: 2, name: \"中文\", other: null },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work with more fields in data\", async () => {\n    const table = DataFrame({\n      id: [1, 2],\n      name: [\"english\", \"中文\"],\n      other: [true, false],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work based on fields order\", async () => {\n    const table = DataFrame({\n      field1: [1, 2],\n      field2: [\"english\", \"中文\"],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work based on field names (equal)\", async () => {\n    const table = DataFrame({\n      name: [\"english\", \"中文\"],\n      id: [1, 2],\n    }).lazy()\n\n    const schema: Schema = {\n      fieldsMatch: \"equal\",\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work based on field names (subset)\", async () => {\n    const table = DataFrame({\n      name: [\"english\", \"中文\"],\n      id: [1, 2],\n    }).lazy()\n\n    const schema: Schema = {\n      fieldsMatch: \"subset\",\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work based on field names (superset)\", async () => {\n    const table = DataFrame({\n      name: [\"english\", \"中文\"],\n      id: [1, 2],\n    }).lazy()\n\n    const schema: Schema = {\n      fieldsMatch: \"superset\",\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should work based on field names (partial)\", async () => {\n    const table = DataFrame({\n      name: [\"english\", \"中文\"],\n      id: [1, 2],\n    }).lazy()\n\n    const schema: Schema = {\n      fieldsMatch: \"partial\",\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should parse string columns\", async () => {\n    const table = DataFrame({\n      id: [\"1\", \"2\"],\n      name: [\"english\", \"中文\"],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"string\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: \"english\" },\n      { id: 2, name: \"中文\" },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n\n  it(\"should read type errors as nulls\", async () => {\n    const table = DataFrame({\n      id: [1, 2],\n      name: [\"english\", \"中文\"],\n    }).lazy()\n\n    const schema: Schema = {\n      fields: [\n        { name: \"id\", type: \"integer\" },\n        { name: \"name\", type: \"integer\" },\n      ],\n    }\n\n    const records = [\n      { id: 1, name: null },\n      { id: 2, name: null },\n    ]\n\n    const ldf = await normalizeTable(table, schema)\n    const df = await ldf.collect()\n    expect(df.toRecords()).toEqual(records)\n  })\n})\n"]}