@frictionless-ts/database 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/adapters/base.d.ts +20 -0
- package/build/adapters/base.js +66 -0
- package/build/adapters/create.d.ts +5 -0
- package/build/adapters/create.js +17 -0
- package/build/adapters/mysql.d.ts +10 -0
- package/build/adapters/mysql.js +90 -0
- package/build/adapters/mysql.spec.d.ts +1 -0
- package/build/adapters/mysql.spec.js +170 -0
- package/build/adapters/postgresql.d.ts +10 -0
- package/build/adapters/postgresql.js +100 -0
- package/build/adapters/postgresql.spec.d.ts +1 -0
- package/build/adapters/postgresql.spec.js +170 -0
- package/build/adapters/sqlite.bun.d.ts +2 -0
- package/build/adapters/sqlite.bun.js +7 -0
- package/build/adapters/sqlite.d.ts +11 -0
- package/build/adapters/sqlite.js +57 -0
- package/build/adapters/sqlite.node.d.ts +2 -0
- package/build/adapters/sqlite.node.js +43 -0
- package/build/adapters/sqlite.spec.d.ts +1 -0
- package/build/adapters/sqlite.spec.js +252 -0
- package/build/field/Field.d.ts +3 -0
- package/build/field/Field.js +2 -0
- package/build/field/Type.d.ts +2 -0
- package/build/field/Type.js +2 -0
- package/build/field/index.d.ts +2 -0
- package/build/field/index.js +2 -0
- package/build/index.d.ts +6 -0
- package/build/index.js +7 -0
- package/build/package/index.d.ts +2 -0
- package/build/package/index.js +3 -0
- package/build/package/load.d.ts +7 -0
- package/build/package/load.js +30 -0
- package/build/package/save.d.ts +10 -0
- package/build/package/save.js +28 -0
- package/build/plugin.d.ts +15 -0
- package/build/plugin.js +54 -0
- package/build/plugin.spec.d.ts +1 -0
- package/build/plugin.spec.js +328 -0
- package/build/resource/Format.d.ts +1 -0
- package/build/resource/Format.js +2 -0
- package/build/resource/index.d.ts +1 -0
- package/build/resource/index.js +2 -0
- package/build/schema/Schema.d.ts +4 -0
- package/build/schema/Schema.js +2 -0
- package/build/schema/index.d.ts +2 -0
- package/build/schema/index.js +2 -0
- package/build/schema/infer.d.ts +4 -0
- package/build/schema/infer.js +24 -0
- package/build/schema/infer.spec.d.ts +1 -0
- package/build/schema/infer.spec.js +25 -0
- package/build/table/index.d.ts +2 -0
- package/build/table/index.js +3 -0
- package/build/table/load.d.ts +6 -0
- package/build/table/load.js +29 -0
- package/build/table/load.spec.d.ts +1 -0
- package/build/table/load.spec.js +24 -0
- package/build/table/save.d.ts +4 -0
- package/build/table/save.js +57 -0
- package/build/table/save.spec.d.ts +1 -0
- package/build/table/save.spec.js +20 -0
- package/package.json +45 -0
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import * as pl from "nodejs-polars";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { useRecording } from "vitest-polly";
|
|
4
|
+
import { loadPackageFromDatabase } from "../package/index.js";
|
|
5
|
+
import { inferDatabaseSchema } from "../schema/index.js";
|
|
6
|
+
import { loadDatabaseTable, saveDatabaseTable } from "../table/index.js";
|
|
7
|
+
import { createAdapter } from "./create.js";
|
|
8
|
+
useRecording();
|
|
9
|
+
const path = process.env.DPKIT_POSTGRESQL_URL;
|
|
10
|
+
// Vitest runs in-file tests sequentially so we can use the same table
|
|
11
|
+
const dialect = { table: "dpkit" };
|
|
12
|
+
const record1 = { id: 1, name: "english" };
|
|
13
|
+
const record2 = { id: 2, name: "中文" };
|
|
14
|
+
describe.skipIf(!path)("PostgresqlAdapter", () => {
|
|
15
|
+
if (!path)
|
|
16
|
+
return;
|
|
17
|
+
it("should infer schema", async () => {
|
|
18
|
+
const source = pl
|
|
19
|
+
.DataFrame([
|
|
20
|
+
pl.Series("string", ["string"], pl.Utf8),
|
|
21
|
+
pl.Series("integer", [1], pl.Int32),
|
|
22
|
+
pl.Series("number", [1.1], pl.Float64),
|
|
23
|
+
])
|
|
24
|
+
.lazy();
|
|
25
|
+
await saveDatabaseTable(source, {
|
|
26
|
+
path,
|
|
27
|
+
dialect,
|
|
28
|
+
format: "postgresql",
|
|
29
|
+
overwrite: true,
|
|
30
|
+
});
|
|
31
|
+
const schema = await inferDatabaseSchema({
|
|
32
|
+
path,
|
|
33
|
+
dialect,
|
|
34
|
+
format: "postgresql",
|
|
35
|
+
});
|
|
36
|
+
expect(schema).toEqual({
|
|
37
|
+
fields: [
|
|
38
|
+
{ name: "string", type: "string" },
|
|
39
|
+
{ name: "integer", type: "integer" },
|
|
40
|
+
{ name: "number", type: "number" },
|
|
41
|
+
],
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
it("should save/load table", async () => {
|
|
45
|
+
const source = pl.DataFrame([record1, record2]).lazy();
|
|
46
|
+
await saveDatabaseTable(source, {
|
|
47
|
+
path,
|
|
48
|
+
dialect,
|
|
49
|
+
format: "postgresql",
|
|
50
|
+
overwrite: true,
|
|
51
|
+
});
|
|
52
|
+
const target = await loadDatabaseTable({
|
|
53
|
+
path,
|
|
54
|
+
dialect,
|
|
55
|
+
format: "postgresql",
|
|
56
|
+
});
|
|
57
|
+
expect((await target.collect()).toRecords()).toEqual([record1, record2]);
|
|
58
|
+
});
|
|
59
|
+
it("should save/load table with various data types", async () => {
|
|
60
|
+
const source = pl
|
|
61
|
+
.DataFrame([
|
|
62
|
+
pl.Series("array", ["[1, 2, 3]"], pl.String),
|
|
63
|
+
pl.Series("boolean", [true], pl.Bool),
|
|
64
|
+
pl.Series("date", [new Date(Date.UTC(2025, 0, 1))], pl.Date),
|
|
65
|
+
pl.Series("datetime", [new Date(Date.UTC(2025, 0, 1))], pl.Datetime),
|
|
66
|
+
pl.Series("duration", ["P23DT23H"], pl.String),
|
|
67
|
+
pl.Series("geojson", ['{"value": 1}'], pl.String),
|
|
68
|
+
pl.Series("geopoint", [[40.0, 50.0]], pl.List(pl.Float32)),
|
|
69
|
+
pl.Series("integer", [1], pl.Int32),
|
|
70
|
+
pl.Series("list", [[1.0, 2.0, 3.0]], pl.List(pl.Float32)),
|
|
71
|
+
pl.Series("number", [1.1], pl.Float64),
|
|
72
|
+
pl.Series("object", ['{"value": 1}']),
|
|
73
|
+
pl.Series("string", ["string"], pl.String),
|
|
74
|
+
pl.Series("time", [new Date(Date.UTC(2025, 0, 1))], pl.Time),
|
|
75
|
+
pl.Series("year", [2025], pl.Int32),
|
|
76
|
+
pl.Series("yearmonth", [[2025, 1]], pl.List(pl.Int16)),
|
|
77
|
+
])
|
|
78
|
+
.lazy();
|
|
79
|
+
await saveDatabaseTable(source, {
|
|
80
|
+
path,
|
|
81
|
+
dialect,
|
|
82
|
+
format: "postgresql",
|
|
83
|
+
overwrite: true,
|
|
84
|
+
fieldTypes: {
|
|
85
|
+
geojson: "geojson",
|
|
86
|
+
geopoint: "geopoint",
|
|
87
|
+
list: "list",
|
|
88
|
+
object: "object",
|
|
89
|
+
// TODO: Remove time after:
|
|
90
|
+
// https://github.com/pola-rs/nodejs-polars/issues/364
|
|
91
|
+
time: "time",
|
|
92
|
+
year: "year",
|
|
93
|
+
yearmonth: "yearmonth",
|
|
94
|
+
},
|
|
95
|
+
});
|
|
96
|
+
const target = await loadDatabaseTable({ path, dialect, format: "postgresql" }, { denormalized: true });
|
|
97
|
+
expect((await target.collect()).toRecords()).toEqual([
|
|
98
|
+
{
|
|
99
|
+
array: "[1, 2, 3]",
|
|
100
|
+
boolean: true,
|
|
101
|
+
date: "2025-01-01",
|
|
102
|
+
datetime: new Date(Date.UTC(2025, 0, 1)),
|
|
103
|
+
duration: "P23DT23H",
|
|
104
|
+
geojson: '{"value": 1}',
|
|
105
|
+
geopoint: "40.0,50.0",
|
|
106
|
+
integer: 1,
|
|
107
|
+
list: "1.0,2.0,3.0",
|
|
108
|
+
number: 1.1,
|
|
109
|
+
object: '{"value": 1}',
|
|
110
|
+
string: "string",
|
|
111
|
+
time: "00:00:00",
|
|
112
|
+
year: 2025,
|
|
113
|
+
yearmonth: "2025-01",
|
|
114
|
+
},
|
|
115
|
+
]);
|
|
116
|
+
});
|
|
117
|
+
it("should load package from database", async () => {
|
|
118
|
+
const adapter = createAdapter("postgresql");
|
|
119
|
+
const database = await adapter.connectDatabase(path);
|
|
120
|
+
await database.schema.dropTable("table1").ifExists().execute();
|
|
121
|
+
await database.schema
|
|
122
|
+
.createTable("table1")
|
|
123
|
+
.ifNotExists()
|
|
124
|
+
.addColumn("id", "integer", column => column.notNull())
|
|
125
|
+
.addColumn("name", "text")
|
|
126
|
+
.execute();
|
|
127
|
+
await database.schema.dropTable("table2").ifExists().execute();
|
|
128
|
+
await database.schema
|
|
129
|
+
.createTable("table2")
|
|
130
|
+
.ifNotExists()
|
|
131
|
+
.addColumn("id", "integer", column => column.notNull())
|
|
132
|
+
.addColumn("number", "numeric")
|
|
133
|
+
.addColumn("datetime", "timestamp")
|
|
134
|
+
.execute();
|
|
135
|
+
const datapackage = await loadPackageFromDatabase(path, {
|
|
136
|
+
format: "postgresql",
|
|
137
|
+
includeTables: ["table1", "table2"],
|
|
138
|
+
});
|
|
139
|
+
expect(datapackage).toEqual({
|
|
140
|
+
resources: [
|
|
141
|
+
{
|
|
142
|
+
path,
|
|
143
|
+
name: "table1",
|
|
144
|
+
format: "postgresql",
|
|
145
|
+
dialect: { table: "table1" },
|
|
146
|
+
schema: {
|
|
147
|
+
fields: [
|
|
148
|
+
{ name: "id", type: "integer", constraints: { required: true } },
|
|
149
|
+
{ name: "name", type: "string" },
|
|
150
|
+
],
|
|
151
|
+
},
|
|
152
|
+
},
|
|
153
|
+
{
|
|
154
|
+
path,
|
|
155
|
+
name: "table2",
|
|
156
|
+
format: "postgresql",
|
|
157
|
+
dialect: { table: "table2" },
|
|
158
|
+
schema: {
|
|
159
|
+
fields: [
|
|
160
|
+
{ name: "id", type: "integer", constraints: { required: true } },
|
|
161
|
+
{ name: "number", type: "number" },
|
|
162
|
+
{ name: "datetime", type: "datetime" },
|
|
163
|
+
],
|
|
164
|
+
},
|
|
165
|
+
},
|
|
166
|
+
],
|
|
167
|
+
});
|
|
168
|
+
});
|
|
169
|
+
});
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"postgresql.spec.js","sourceRoot":"","sources":["../../adapters/postgresql.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as pl from \"nodejs-polars\"\nimport { describe, expect, it } from \"vitest\"\nimport { useRecording } from \"vitest-polly\"\nimport { loadPackageFromDatabase } from \"../package/index.ts\"\nimport { inferDatabaseSchema } from \"../schema/index.ts\"\nimport { loadDatabaseTable, saveDatabaseTable } from \"../table/index.ts\"\nimport { createAdapter } from \"./create.ts\"\n\nuseRecording()\n\nconst path = process.env.DPKIT_POSTGRESQL_URL\n\n// Vitest runs in-file tests sequentially so we can use the same table\nconst dialect = { table: \"dpkit\" }\nconst record1 = { id: 1, name: \"english\" }\nconst record2 = { id: 2, name: \"中文\" }\n\ndescribe.skipIf(!path)(\"PostgresqlAdapter\", () => {\n  if (!path) return\n\n  it(\"should infer schema\", async () => {\n    const source = pl\n      .DataFrame([\n        pl.Series(\"string\", [\"string\"], pl.Utf8),\n        pl.Series(\"integer\", [1], pl.Int32),\n        pl.Series(\"number\", [1.1], pl.Float64),\n      ])\n      .lazy()\n\n    await saveDatabaseTable(source, {\n      path,\n      dialect,\n      format: \"postgresql\",\n      overwrite: true,\n    })\n\n    const schema = await inferDatabaseSchema({\n      path,\n      dialect,\n      format: \"postgresql\",\n    })\n\n    expect(schema).toEqual({\n      fields: [\n        { name: \"string\", type: \"string\" },\n        { name: \"integer\", type: \"integer\" },\n        { name: \"number\", type: \"number\" },\n      ],\n    })\n  })\n\n  it(\"should save/load table\", async () => {\n    const source = pl.DataFrame([record1, record2]).lazy()\n\n    await saveDatabaseTable(source, {\n      path,\n      dialect,\n      format: \"postgresql\",\n      overwrite: true,\n    })\n\n    const target = await loadDatabaseTable({\n      path,\n      dialect,\n      format: \"postgresql\",\n    })\n\n    expect((await target.collect()).toRecords()).toEqual([record1, record2])\n  })\n\n  it(\"should save/load table with various data types\", async () => {\n    const source = pl\n      .DataFrame([\n        pl.Series(\"array\", [\"[1, 2, 3]\"], pl.String),\n        pl.Series(\"boolean\", [true], pl.Bool),\n        pl.Series(\"date\", [new Date(Date.UTC(2025, 0, 1))], pl.Date),\n        pl.Series(\"datetime\", [new Date(Date.UTC(2025, 0, 1))], pl.Datetime),\n        pl.Series(\"duration\", [\"P23DT23H\"], pl.String),\n        pl.Series(\"geojson\", ['{\"value\": 1}'], pl.String),\n        pl.Series(\"geopoint\", [[40.0, 50.0]], pl.List(pl.Float32)),\n        pl.Series(\"integer\", [1], pl.Int32),\n        pl.Series(\"list\", [[1.0, 2.0, 3.0]], pl.List(pl.Float32)),\n        pl.Series(\"number\", [1.1], pl.Float64),\n        pl.Series(\"object\", ['{\"value\": 1}']),\n        pl.Series(\"string\", [\"string\"], pl.String),\n        pl.Series(\"time\", [new Date(Date.UTC(2025, 0, 1))], pl.Time),\n        pl.Series(\"year\", [2025], pl.Int32),\n        pl.Series(\"yearmonth\", [[2025, 1]], pl.List(pl.Int16)),\n      ])\n      .lazy()\n\n    await saveDatabaseTable(source, {\n      path,\n      dialect,\n      format: \"postgresql\",\n      overwrite: true,\n      fieldTypes: {\n        geojson: \"geojson\",\n        geopoint: \"geopoint\",\n        list: \"list\",\n        object: \"object\",\n        // TODO: Remove time after:\n        // https://github.com/pola-rs/nodejs-polars/issues/364\n        time: \"time\",\n        year: \"year\",\n        yearmonth: \"yearmonth\",\n      },\n    })\n\n    const target = await loadDatabaseTable(\n      { path, dialect, format: \"postgresql\" },\n      { denormalized: true },\n    )\n\n    expect((await target.collect()).toRecords()).toEqual([\n      {\n        array: \"[1, 2, 3]\",\n        boolean: true,\n        date: \"2025-01-01\",\n        datetime: new Date(Date.UTC(2025, 0, 1)),\n        duration: \"P23DT23H\",\n        geojson: '{\"value\": 1}',\n        geopoint: \"40.0,50.0\",\n        integer: 1,\n        list: \"1.0,2.0,3.0\",\n        number: 1.1,\n        object: '{\"value\": 1}',\n        string: \"string\",\n        time: \"00:00:00\",\n        year: 2025,\n        yearmonth: \"2025-01\",\n      },\n    ])\n  })\n\n  it(\"should load package from database\", async () => {\n    const adapter = createAdapter(\"postgresql\")\n    const database = await adapter.connectDatabase(path)\n\n    await database.schema.dropTable(\"table1\").ifExists().execute()\n    await database.schema\n      .createTable(\"table1\")\n      .ifNotExists()\n      .addColumn(\"id\", \"integer\", column => column.notNull())\n      .addColumn(\"name\", \"text\")\n      .execute()\n\n    await database.schema.dropTable(\"table2\").ifExists().execute()\n    await database.schema\n      .createTable(\"table2\")\n      .ifNotExists()\n      .addColumn(\"id\", \"integer\", column => column.notNull())\n      .addColumn(\"number\", \"numeric\")\n      .addColumn(\"datetime\", \"timestamp\")\n      .execute()\n\n    const datapackage = await loadPackageFromDatabase(path, {\n      format: \"postgresql\",\n      includeTables: [\"table1\", \"table2\"],\n    })\n\n    expect(datapackage).toEqual({\n      resources: [\n        {\n          path,\n          name: \"table1\",\n          format: \"postgresql\",\n          dialect: { table: \"table1\" },\n          schema: {\n            fields: [\n              { name: \"id\", type: \"integer\", constraints: { required: true } },\n              { name: \"name\", type: \"string\" },\n            ],\n          },\n        },\n        {\n          path,\n          name: \"table2\",\n          format: \"postgresql\",\n          dialect: { table: \"table2\" },\n          schema: {\n            fields: [\n              { name: \"id\", type: \"integer\", constraints: { required: true } },\n              { name: \"number\", type: \"number\" },\n              { name: \"datetime\", type: \"datetime\" },\n            ],\n          },\n        },\n      ],\n    })\n  })\n})\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { BunSqliteDialect } from "kysely-bun-sqlite";
|
|
2
|
+
export async function createBunSqliteDialect(path) {
|
|
3
|
+
// @ts-ignore
|
|
4
|
+
const { Database } = await import("bun:sqlite");
|
|
5
|
+
return new BunSqliteDialect({ database: new Database(path) });
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3FsaXRlLmJ1bi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2FkYXB0ZXJzL3NxbGl0ZS5idW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sbUJBQW1CLENBQUE7QUFFcEQsTUFBTSxDQUFDLEtBQUssVUFBVSxzQkFBc0IsQ0FBQyxJQUFZO0lBQ3ZELGFBQWE7SUFDYixNQUFNLEVBQUUsUUFBUSxFQUFFLEdBQUcsTUFBTSxNQUFNLENBQUMsWUFBWSxDQUFDLENBQUE7SUFDL0MsT0FBTyxJQUFJLGdCQUFnQixDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksUUFBUSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQTtBQUMvRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQnVuU3FsaXRlRGlhbGVjdCB9IGZyb20gXCJreXNlbHktYnVuLXNxbGl0ZVwiXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjcmVhdGVCdW5TcWxpdGVEaWFsZWN0KHBhdGg6IHN0cmluZykge1xuICAvLyBAdHMtaWdub3JlXG4gIGNvbnN0IHsgRGF0YWJhc2UgfSA9IGF3YWl0IGltcG9ydChcImJ1bjpzcWxpdGVcIilcbiAgcmV0dXJuIG5ldyBCdW5TcWxpdGVEaWFsZWN0KHsgZGF0YWJhc2U6IG5ldyBEYXRhYmFzZShwYXRoKSB9KVxufVxuIl19
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { FieldType } from "@frictionless-ts/metadata";
|
|
2
|
+
import type { DatabaseType } from "../field/index.ts";
|
|
3
|
+
import { BaseAdapter } from "./base.ts";
|
|
4
|
+
export declare class SqliteAdapter extends BaseAdapter {
|
|
5
|
+
nativeTypes: ("string" | "number" | "integer" | "year")[];
|
|
6
|
+
createDialect(path: string, options?: {
|
|
7
|
+
create?: boolean;
|
|
8
|
+
}): Promise<import("kysely-bun-sqlite").BunSqliteDialect | import("kysely-generic-sqlite").GenericSqliteDialect>;
|
|
9
|
+
normalizeType(databaseType: DatabaseType): FieldType;
|
|
10
|
+
denormalizeType(fieldType: FieldType): DatabaseType;
|
|
11
|
+
}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
import { isLocalPathExist } from "@frictionless-ts/dataset";
|
|
2
|
+
import { BaseAdapter } from "./base.js";
|
|
3
|
+
// TODO: Currently, the solution is not optimal / hacky
|
|
4
|
+
// We need to rebase on proper sqlite dialect when it will be available
|
|
5
|
+
// - https://github.com/kysely-org/kysely/issues/1292
|
|
6
|
+
// - https://github.com/oven-sh/bun/issues/20412
|
|
7
|
+
export class SqliteAdapter extends BaseAdapter {
|
|
8
|
+
nativeTypes = ["integer", "number", "string", "year"];
|
|
9
|
+
async createDialect(path, options) {
|
|
10
|
+
path = path.replace(/^sqlite:\/\//, "");
|
|
11
|
+
if (!options?.create) {
|
|
12
|
+
const isExist = await isLocalPathExist(path);
|
|
13
|
+
if (!isExist) {
|
|
14
|
+
throw new Error(`Database file "${path}" does not exist`);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
// @ts-ignore
|
|
18
|
+
if (typeof Bun !== "undefined") {
|
|
19
|
+
const { createBunSqliteDialect } = await import("./sqlite.bun.js");
|
|
20
|
+
return await createBunSqliteDialect(path);
|
|
21
|
+
}
|
|
22
|
+
const { createNodeSqliteDialect } = await import("./sqlite.node.js");
|
|
23
|
+
return await createNodeSqliteDialect(path);
|
|
24
|
+
}
|
|
25
|
+
normalizeType(databaseType) {
|
|
26
|
+
switch (databaseType.toLowerCase()) {
|
|
27
|
+
case "blob":
|
|
28
|
+
return "string";
|
|
29
|
+
case "text":
|
|
30
|
+
return "string";
|
|
31
|
+
case "integer":
|
|
32
|
+
return "integer";
|
|
33
|
+
case "numeric":
|
|
34
|
+
case "real":
|
|
35
|
+
return "number";
|
|
36
|
+
default:
|
|
37
|
+
return "string";
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
denormalizeType(fieldType) {
|
|
41
|
+
switch (fieldType) {
|
|
42
|
+
case "boolean":
|
|
43
|
+
return "integer";
|
|
44
|
+
case "integer":
|
|
45
|
+
return "integer";
|
|
46
|
+
case "number":
|
|
47
|
+
return "real";
|
|
48
|
+
case "string":
|
|
49
|
+
return "text";
|
|
50
|
+
case "year":
|
|
51
|
+
return "integer";
|
|
52
|
+
default:
|
|
53
|
+
return "text";
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3FsaXRlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vYWRhcHRlcnMvc3FsaXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDBCQUEwQixDQUFBO0FBRzNELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxXQUFXLENBQUE7QUFFdkMsdURBQXVEO0FBQ3ZELHVFQUF1RTtBQUN2RSxxREFBcUQ7QUFDckQsZ0RBQWdEO0FBRWhELE1BQU0sT0FBTyxhQUFjLFNBQVEsV0FBVztJQUM1QyxXQUFXLEdBQUcsQ0FBQyxTQUFTLEVBQUUsUUFBUSxFQUFFLFFBQVEsRUFBRSxNQUFNLENBQXVCLENBQUE7SUFFM0UsS0FBSyxDQUFDLGFBQWEsQ0FBQyxJQUFZLEVBQUUsT0FBOEI7UUFDOUQsSUFBSSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsY0FBYyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRXZDLElBQUksQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLENBQUM7WUFDckIsTUFBTSxPQUFPLEdBQUcsTUFBTSxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQTtZQUM1QyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7Z0JBQ2IsTUFBTSxJQUFJLEtBQUssQ0FBQyxrQkFBa0IsSUFBSSxrQkFBa0IsQ0FBQyxDQUFBO1lBQzNELENBQUM7UUFDSCxDQUFDO1FBRUQsYUFBYTtRQUNiLElBQUksT0FBTyxHQUFHLEtBQUssV0FBVyxFQUFFLENBQUM7WUFDL0IsTUFBTSxFQUFFLHNCQUFzQixFQUFFLEdBQUcsTUFBTSxNQUFNLENBQUMsaUJBQWlCLENBQUMsQ0FBQTtZQUNsRSxPQUFPLE1BQU0sc0JBQXNCLENBQUMsSUFBSSxDQUFDLENBQUE7UUFDM0MsQ0FBQztRQUVELE1BQU0sRUFBRSx1QkFBdUIsRUFBRSxHQUFHLE1BQU0sTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUE7UUFDcEUsT0FBTyxNQUFNLHVCQUF1QixDQUFDLElBQUksQ0FBQyxDQUFBO0lBQzVDLENBQUM7SUFFRCxhQUFhLENBQUMsWUFBMEI7UUFDdEMsUUFBUSxZQUFZLENBQUMsV0FBVyxFQUFFLEVBQUUsQ0FBQztZQUNuQyxLQUFLLE1BQU07Z0JBQ1QsT0FBTyxRQUFRLENBQUE7WUFDakIsS0FBSyxNQUFNO2dCQUNULE9BQU8sUUFBUSxDQUFBO1lBQ2pCLEtBQUssU0FBUztnQkFDWixPQUFPLFNBQVMsQ0FBQTtZQUNsQixLQUFLLFNBQVMsQ0FBQztZQUNmLEtBQUssTUFBTTtnQkFDVCxPQUFPLFFBQVEsQ0FBQTtZQUNqQjtnQkFDRSxPQUFPLFFBQVEsQ0FBQTtRQUNuQixDQUFDO0lBQ0gsQ0FBQztJQUVELGVBQWUsQ0FBQyxTQUFvQjtRQUNsQyxRQUFRLFNBQVMsRUFBRSxDQUFDO1lBQ2xCLEtBQUssU0FBUztnQkFDWixPQUFPLFNBQVMsQ0FBQTtZQUNsQixLQUFLLFNBQVM7Z0JBQ1osT0FBTyxTQUFTLENBQUE7WUFDbEIsS0FBSyxRQUFRO2dCQUNYLE9BQU8sTUFBTSxDQUFBO1lBQ2YsS0FBSyxRQUFRO2dCQUNYLE9BQU8sTUFBTSxDQUFBO1lBQ2YsS0FBSyxNQUFNO2dCQUNULE9BQU8sU0FBUyxDQUFBO1lBQ2xCO2dCQUNFLE9BQU8sTUFBTSxDQUFBO1FBQ2pCLENBQUM7SUFDSCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpc0xvY2FsUGF0aEV4aXN0IH0gZnJvbSBcIkBmcmljdGlvbmxlc3MtdHMvZGF0YXNldFwiXG5pbXBvcnQgdHlwZSB7IEZpZWxkVHlwZSB9IGZyb20gXCJAZnJpY3Rpb25sZXNzLXRzL21ldGFkYXRhXCJcbmltcG9ydCB0eXBlIHsgRGF0YWJhc2VUeXBlIH0gZnJvbSBcIi4uL2ZpZWxkL2luZGV4LnRzXCJcbmltcG9ydCB7IEJhc2VBZGFwdGVyIH0gZnJvbSBcIi4vYmFzZS50c1wiXG5cbi8vIFRPRE86IEN1cnJlbnRseSwgdGhlIHNvbHV0aW9uIGlzIG5vdCBvcHRpbWFsIC8gaGFja3lcbi8vIFdlIG5lZWQgdG8gcmViYXNlIG9uIHByb3BlciBzcWxpdGUgZGlhbGVjdCB3aGVuIGl0IHdpbGwgYmUgYXZhaWxhYmxlXG4vLyAtIGh0dHBzOi8vZ2l0aHViLmNvbS9reXNlbHktb3JnL2t5c2VseS9pc3N1ZXMvMTI5MlxuLy8gLSBodHRwczovL2dpdGh1Yi5jb20vb3Zlbi1zaC9idW4vaXNzdWVzLzIwNDEyXG5cbmV4cG9ydCBjbGFzcyBTcWxpdGVBZGFwdGVyIGV4dGVuZHMgQmFzZUFkYXB0ZXIge1xuICBuYXRpdmVUeXBlcyA9IFtcImludGVnZXJcIiwgXCJudW1iZXJcIiwgXCJzdHJpbmdcIiwgXCJ5ZWFyXCJdIHNhdGlzZmllcyBGaWVsZFR5cGVbXVxuXG4gIGFzeW5jIGNyZWF0ZURpYWxlY3QocGF0aDogc3RyaW5nLCBvcHRpb25zPzogeyBjcmVhdGU/OiBib29sZWFuIH0pIHtcbiAgICBwYXRoID0gcGF0aC5yZXBsYWNlKC9ec3FsaXRlOlxcL1xcLy8sIFwiXCIpXG5cbiAgICBpZiAoIW9wdGlvbnM/LmNyZWF0ZSkge1xuICAgICAgY29uc3QgaXNFeGlzdCA9IGF3YWl0IGlzTG9jYWxQYXRoRXhpc3QocGF0aClcbiAgICAgIGlmICghaXNFeGlzdCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYERhdGFiYXNlIGZpbGUgXCIke3BhdGh9XCIgZG9lcyBub3QgZXhpc3RgKVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIEB0cy1pZ25vcmVcbiAgICBpZiAodHlwZW9mIEJ1biAhPT0gXCJ1bmRlZmluZWRcIikge1xuICAgICAgY29uc3QgeyBjcmVhdGVCdW5TcWxpdGVEaWFsZWN0IH0gPSBhd2FpdCBpbXBvcnQoXCIuL3NxbGl0ZS5idW4udHNcIilcbiAgICAgIHJldHVybiBhd2FpdCBjcmVhdGVCdW5TcWxpdGVEaWFsZWN0KHBhdGgpXG4gICAgfVxuXG4gICAgY29uc3QgeyBjcmVhdGVOb2RlU3FsaXRlRGlhbGVjdCB9ID0gYXdhaXQgaW1wb3J0KFwiLi9zcWxpdGUubm9kZS50c1wiKVxuICAgIHJldHVybiBhd2FpdCBjcmVhdGVOb2RlU3FsaXRlRGlhbGVjdChwYXRoKVxuICB9XG5cbiAgbm9ybWFsaXplVHlwZShkYXRhYmFzZVR5cGU6IERhdGFiYXNlVHlwZSk6IEZpZWxkVHlwZSB7XG4gICAgc3dpdGNoIChkYXRhYmFzZVR5cGUudG9Mb3dlckNhc2UoKSkge1xuICAgICAgY2FzZSBcImJsb2JcIjpcbiAgICAgICAgcmV0dXJuIFwic3RyaW5nXCJcbiAgICAgIGNhc2UgXCJ0ZXh0XCI6XG4gICAgICAgIHJldHVybiBcInN0cmluZ1wiXG4gICAgICBjYXNlIFwiaW50ZWdlclwiOlxuICAgICAgICByZXR1cm4gXCJpbnRlZ2VyXCJcbiAgICAgIGNhc2UgXCJudW1lcmljXCI6XG4gICAgICBjYXNlIFwicmVhbFwiOlxuICAgICAgICByZXR1cm4gXCJudW1iZXJcIlxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIFwic3RyaW5nXCJcbiAgICB9XG4gIH1cblxuICBkZW5vcm1hbGl6ZVR5cGUoZmllbGRUeXBlOiBGaWVsZFR5cGUpOiBEYXRhYmFzZVR5cGUge1xuICAgIHN3aXRjaCAoZmllbGRUeXBlKSB7XG4gICAgICBjYXNlIFwiYm9vbGVhblwiOlxuICAgICAgICByZXR1cm4gXCJpbnRlZ2VyXCJcbiAgICAgIGNhc2UgXCJpbnRlZ2VyXCI6XG4gICAgICAgIHJldHVybiBcImludGVnZXJcIlxuICAgICAgY2FzZSBcIm51bWJlclwiOlxuICAgICAgICByZXR1cm4gXCJyZWFsXCJcbiAgICAgIGNhc2UgXCJzdHJpbmdcIjpcbiAgICAgICAgcmV0dXJuIFwidGV4dFwiXG4gICAgICBjYXNlIFwieWVhclwiOlxuICAgICAgICByZXR1cm4gXCJpbnRlZ2VyXCJcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiBcInRleHRcIlxuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
// @ts-nocheck
|
|
2
|
+
import { buildQueryFn, parseBigInt } from "kysely-generic-sqlite";
|
|
3
|
+
import { GenericSqliteDialect } from "kysely-generic-sqlite";
|
|
4
|
+
export async function createNodeSqliteDialect(path) {
|
|
5
|
+
const { DatabaseSync } = await import("node:sqlite");
|
|
6
|
+
return new GenericSqliteDialect(() => createSqliteExecutor(new DatabaseSync(path)));
|
|
7
|
+
}
|
|
8
|
+
function createSqliteExecutor(db) {
|
|
9
|
+
const getStmt = (sql) => {
|
|
10
|
+
const stmt = db.prepare(sql);
|
|
11
|
+
// We change it from original to use plain numbers
|
|
12
|
+
//stmt.setReadBigInts(true)
|
|
13
|
+
return stmt;
|
|
14
|
+
};
|
|
15
|
+
return {
|
|
16
|
+
db,
|
|
17
|
+
query: buildQueryFn({
|
|
18
|
+
all: (sql, parameters = []) => getStmt(sql)
|
|
19
|
+
.all(...parameters)
|
|
20
|
+
// We change it from original to make it work
|
|
21
|
+
// (by default it returns object with null prototype which breaks polars)
|
|
22
|
+
.map(row => ({ ...row })),
|
|
23
|
+
run: (sql, parameters = []) => {
|
|
24
|
+
const { changes, lastInsertRowid } = getStmt(sql).run(...parameters);
|
|
25
|
+
return {
|
|
26
|
+
insertId: parseBigInt(lastInsertRowid),
|
|
27
|
+
numAffectedRows: parseBigInt(changes),
|
|
28
|
+
};
|
|
29
|
+
},
|
|
30
|
+
}),
|
|
31
|
+
close: () => db.close(),
|
|
32
|
+
iterator: (isSelect, sql, parameters = []) => {
|
|
33
|
+
if (!isSelect) {
|
|
34
|
+
throw new Error("Only support select in stream()");
|
|
35
|
+
}
|
|
36
|
+
return getStmt(sql)
|
|
37
|
+
.iterate(...parameters) // We change it from original to make it work
|
|
38
|
+
// (by default it returns object with null prototype which breaks polars)
|
|
39
|
+
.map(row => ({ ...row }));
|
|
40
|
+
},
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3FsaXRlLm5vZGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9hZGFwdGVycy9zcWxpdGUubm9kZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjO0FBQ2QsT0FBTyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQTtBQUNqRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQTtBQUc1RCxNQUFNLENBQUMsS0FBSyxVQUFVLHVCQUF1QixDQUFDLElBQVk7SUFDeEQsTUFBTSxFQUFFLFlBQVksRUFBRSxHQUFHLE1BQU0sTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBRXBELE9BQU8sSUFBSSxvQkFBb0IsQ0FBQyxHQUFHLEVBQUUsQ0FDbkMsb0JBQW9CLENBQUMsSUFBSSxZQUFZLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FDN0MsQ0FBQTtBQUNILENBQUM7QUFFRCxTQUFTLG9CQUFvQixDQUFDLEVBQWdCO0lBQzVDLE1BQU0sT0FBTyxHQUFHLENBQUMsR0FBVyxFQUFFLEVBQUU7UUFDOUIsTUFBTSxJQUFJLEdBQUcsRUFBRSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQTtRQUM1QixrREFBa0Q7UUFDbEQsMkJBQTJCO1FBQzNCLE9BQU8sSUFBSSxDQUFBO0lBQ2IsQ0FBQyxDQUFBO0lBRUQsT0FBTztRQUNMLEVBQUU7UUFDRixLQUFLLEVBQUUsWUFBWSxDQUFDO1lBQ2xCLEdBQUcsRUFBRSxDQUFDLEdBQUcsRUFBRSxVQUFVLEdBQUcsRUFBRSxFQUFFLEVBQUUsQ0FDNUIsT0FBTyxDQUFDLEdBQUcsQ0FBQztpQkFDVCxHQUFHLENBQUMsR0FBRyxVQUFVLENBQUM7Z0JBQ25CLDZDQUE2QztnQkFDN0MseUVBQXlFO2lCQUN4RSxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQUUsR0FBRyxHQUFHLEVBQUUsQ0FBQyxDQUFDO1lBRTdCLEdBQUcsRUFBRSxDQUFDLEdBQUcsRUFBRSxVQUFVLEdBQUcsRUFBRSxFQUFFLEVBQUU7Z0JBQzVCLE1BQU0sRUFBRSxPQUFPLEVBQUUsZUFBZSxFQUFFLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxHQUFHLFVBQVUsQ0FBQyxDQUFBO2dCQUNwRSxPQUFPO29CQUNMLFFBQVEsRUFBRSxXQUFXLENBQUMsZUFBZSxDQUFDO29CQUN0QyxlQUFlLEVBQUUsV0FBVyxDQUFDLE9BQU8sQ0FBQztpQkFDdEMsQ0FBQTtZQUNILENBQUM7U0FDRixDQUFDO1FBQ0YsS0FBSyxFQUFFLEdBQUcsRUFBRSxDQUFDLEVBQUUsQ0FBQyxLQUFLLEVBQUU7UUFDdkIsUUFBUSxFQUFFLENBQUMsUUFBUSxFQUFFLEdBQUcsRUFBRSxVQUFVLEdBQUcsRUFBRSxFQUFFLEVBQUU7WUFDM0MsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO2dCQUNkLE1BQU0sSUFBSSxLQUFLLENBQUMsaUNBQWlDLENBQUMsQ0FBQTtZQUNwRCxDQUFDO1lBQ0QsT0FDRSxPQUFPLENBQUMsR0FBRyxDQUFDO2lCQUNULE9BQU8sQ0FBQyxHQUFHLFVBQVUsQ0FBQyxDQUFDLDZDQUE2QztnQkFDckUseUVBQXlFO2lCQUN4RSxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQUUsR0FBRyxHQUFHLEVBQUUsQ0FBQyxDQUMzQixDQUFBO1FBQ0gsQ0FBQztLQUNGLENBQUE7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQHRzLW5vY2hlY2tcbmltcG9ydCB7IGJ1aWxkUXVlcnlGbiwgcGFyc2VCaWdJbnQgfSBmcm9tIFwia3lzZWx5LWdlbmVyaWMtc3FsaXRlXCJcbmltcG9ydCB7IEdlbmVyaWNTcWxpdGVEaWFsZWN0IH0gZnJvbSBcImt5c2VseS1nZW5lcmljLXNxbGl0ZVwiXG5pbXBvcnQgdHlwZSB7IElHZW5lcmljU3FsaXRlIH0gZnJvbSBcImt5c2VseS1nZW5lcmljLXNxbGl0ZVwiXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjcmVhdGVOb2RlU3FsaXRlRGlhbGVjdChwYXRoOiBzdHJpbmcpIHtcbiAgY29uc3QgeyBEYXRhYmFzZVN5bmMgfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6c3FsaXRlXCIpXG5cbiAgcmV0dXJuIG5ldyBHZW5lcmljU3FsaXRlRGlhbGVjdCgoKSA9PlxuICAgIGNyZWF0ZVNxbGl0ZUV4ZWN1dG9yKG5ldyBEYXRhYmFzZVN5bmMocGF0aCkpLFxuICApXG59XG5cbmZ1bmN0aW9uIGNyZWF0ZVNxbGl0ZUV4ZWN1dG9yKGRiOiBEYXRhYmFzZVN5bmMpOiBJR2VuZXJpY1NxbGl0ZTxEYXRhYmFzZVN5bmM+IHtcbiAgY29uc3QgZ2V0U3RtdCA9IChzcWw6IHN0cmluZykgPT4ge1xuICAgIGNvbnN0IHN0bXQgPSBkYi5wcmVwYXJlKHNxbClcbiAgICAvLyBXZSBjaGFuZ2UgaXQgZnJvbSBvcmlnaW5hbCB0byB1c2UgcGxhaW4gbnVtYmVyc1xuICAgIC8vc3RtdC5zZXRSZWFkQmlnSW50cyh0cnVlKVxuICAgIHJldHVybiBzdG10XG4gIH1cblxuICByZXR1cm4ge1xuICAgIGRiLFxuICAgIHF1ZXJ5OiBidWlsZFF1ZXJ5Rm4oe1xuICAgICAgYWxsOiAoc3FsLCBwYXJhbWV0ZXJzID0gW10pID0+XG4gICAgICAgIGdldFN0bXQoc3FsKVxuICAgICAgICAgIC5hbGwoLi4ucGFyYW1ldGVycylcbiAgICAgICAgICAvLyBXZSBjaGFuZ2UgaXQgZnJvbSBvcmlnaW5hbCB0byBtYWtlIGl0IHdvcmtcbiAgICAgICAgICAvLyAoYnkgZGVmYXVsdCBpdCByZXR1cm5zIG9iamVjdCB3aXRoIG51bGwgcHJvdG90eXBlIHdoaWNoIGJyZWFrcyBwb2xhcnMpXG4gICAgICAgICAgLm1hcChyb3cgPT4gKHsgLi4ucm93IH0pKSxcblxuICAgICAgcnVuOiAoc3FsLCBwYXJhbWV0ZXJzID0gW10pID0+IHtcbiAgICAgICAgY29uc3QgeyBjaGFuZ2VzLCBsYXN0SW5zZXJ0Um93aWQgfSA9IGdldFN0bXQoc3FsKS5ydW4oLi4ucGFyYW1ldGVycylcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBpbnNlcnRJZDogcGFyc2VCaWdJbnQobGFzdEluc2VydFJvd2lkKSxcbiAgICAgICAgICBudW1BZmZlY3RlZFJvd3M6IHBhcnNlQmlnSW50KGNoYW5nZXMpLFxuICAgICAgICB9XG4gICAgICB9LFxuICAgIH0pLFxuICAgIGNsb3NlOiAoKSA9PiBkYi5jbG9zZSgpLFxuICAgIGl0ZXJhdG9yOiAoaXNTZWxlY3QsIHNxbCwgcGFyYW1ldGVycyA9IFtdKSA9PiB7XG4gICAgICBpZiAoIWlzU2VsZWN0KSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcIk9ubHkgc3VwcG9ydCBzZWxlY3QgaW4gc3RyZWFtKClcIilcbiAgICAgIH1cbiAgICAgIHJldHVybiAoXG4gICAgICAgIGdldFN0bXQoc3FsKVxuICAgICAgICAgIC5pdGVyYXRlKC4uLnBhcmFtZXRlcnMpIC8vIFdlIGNoYW5nZSBpdCBmcm9tIG9yaWdpbmFsIHRvIG1ha2UgaXQgd29ya1xuICAgICAgICAgIC8vIChieSBkZWZhdWx0IGl0IHJldHVybnMgb2JqZWN0IHdpdGggbnVsbCBwcm90b3R5cGUgd2hpY2ggYnJlYWtzIHBvbGFycylcbiAgICAgICAgICAubWFwKHJvdyA9PiAoeyAuLi5yb3cgfSkpIGFzIGFueVxuICAgICAgKVxuICAgIH0sXG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|