@zenstackhq/testtools 3.3.0-beta.2 → 3.3.0-beta.4
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/dist/index.cjs +61 -24
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +13 -5
- package/dist/index.d.ts +13 -5
- package/dist/index.js +60 -25
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +504 -0
- package/dist/index.mjs.map +1 -0
- package/package.json +10 -9
package/dist/index.cjs
CHANGED
|
@@ -31,6 +31,8 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
|
|
|
31
31
|
// src/index.ts
|
|
32
32
|
var src_exports = {};
|
|
33
33
|
__export(src_exports, {
|
|
34
|
+
TEST_MYSQL_CONFIG: () => TEST_MYSQL_CONFIG,
|
|
35
|
+
TEST_MYSQL_URL: () => TEST_MYSQL_URL,
|
|
34
36
|
TEST_PG_CONFIG: () => TEST_PG_CONFIG,
|
|
35
37
|
TEST_PG_URL: () => TEST_PG_URL,
|
|
36
38
|
createPolicyTestClient: () => createPolicyTestClient,
|
|
@@ -54,11 +56,13 @@ var import_sdk2 = require("@zenstackhq/sdk");
|
|
|
54
56
|
var import_better_sqlite3 = __toESM(require("better-sqlite3"), 1);
|
|
55
57
|
var import_glob = require("glob");
|
|
56
58
|
var import_kysely = require("kysely");
|
|
59
|
+
var import_mysql2 = require("mysql2");
|
|
57
60
|
var import_node_child_process2 = require("child_process");
|
|
58
61
|
var import_node_crypto2 = require("crypto");
|
|
59
62
|
var import_node_fs3 = __toESM(require("fs"), 1);
|
|
60
63
|
var import_node_path3 = __toESM(require("path"), 1);
|
|
61
64
|
var import_pg = require("pg");
|
|
65
|
+
var import_ts_pattern2 = require("ts-pattern");
|
|
62
66
|
var import_vitest2 = require("vitest");
|
|
63
67
|
|
|
64
68
|
// src/project.ts
|
|
@@ -148,12 +152,19 @@ datasource db {
|
|
|
148
152
|
provider = 'postgresql'
|
|
149
153
|
url = '${dbUrl ?? "postgres://postgres:postgres@localhost:5432/db"}'
|
|
150
154
|
}
|
|
155
|
+
`;
|
|
156
|
+
}).with("mysql", () => {
|
|
157
|
+
return `
|
|
158
|
+
datasource db {
|
|
159
|
+
provider = 'mysql'
|
|
160
|
+
url = '${dbUrl ?? "mysql://root:mysql@localhost:3306/db"}'
|
|
161
|
+
}
|
|
151
162
|
`;
|
|
152
163
|
}).exhaustive();
|
|
153
164
|
}
|
|
154
165
|
__name(makePrelude, "makePrelude");
|
|
155
166
|
function replacePlaceholders(schemaText, provider, dbUrl) {
|
|
156
|
-
const url = dbUrl ?? (provider === "sqlite" ? "file:./test.db" : "postgres://postgres:postgres@localhost:5432/db");
|
|
167
|
+
const url = dbUrl ?? (provider === "sqlite" ? "file:./test.db" : provider === "mysql" ? "mysql://root:mysql@localhost:3306/db" : "postgres://postgres:postgres@localhost:5432/db");
|
|
157
168
|
return schemaText.replace(/\$DB_URL/g, url).replace(/\$PROVIDER/g, provider);
|
|
158
169
|
}
|
|
159
170
|
__name(replacePlaceholders, "replacePlaceholders");
|
|
@@ -273,7 +284,8 @@ function getTestDbProvider() {
|
|
|
273
284
|
const val = process.env["TEST_DB_PROVIDER"] ?? "sqlite";
|
|
274
285
|
if (![
|
|
275
286
|
"sqlite",
|
|
276
|
-
"postgresql"
|
|
287
|
+
"postgresql",
|
|
288
|
+
"mysql"
|
|
277
289
|
].includes(val)) {
|
|
278
290
|
throw new Error(`Invalid TEST_DB_PROVIDER value: ${val}`);
|
|
279
291
|
}
|
|
@@ -287,12 +299,20 @@ var TEST_PG_CONFIG = {
|
|
|
287
299
|
password: process.env["TEST_PG_PASSWORD"] ?? "postgres"
|
|
288
300
|
};
|
|
289
301
|
var TEST_PG_URL = `postgres://${TEST_PG_CONFIG.user}:${TEST_PG_CONFIG.password}@${TEST_PG_CONFIG.host}:${TEST_PG_CONFIG.port}`;
|
|
302
|
+
var TEST_MYSQL_CONFIG = {
|
|
303
|
+
host: process.env["TEST_MYSQL_HOST"] ?? "localhost",
|
|
304
|
+
port: process.env["TEST_MYSQL_PORT"] ? parseInt(process.env["TEST_MYSQL_PORT"]) : 3306,
|
|
305
|
+
user: process.env["TEST_MYSQL_USER"] ?? "root",
|
|
306
|
+
password: process.env["TEST_MYSQL_PASSWORD"] ?? "mysql",
|
|
307
|
+
timezone: "Z"
|
|
308
|
+
};
|
|
309
|
+
var TEST_MYSQL_URL = `mysql://${TEST_MYSQL_CONFIG.user}:${TEST_MYSQL_CONFIG.password}@${TEST_MYSQL_CONFIG.host}:${TEST_MYSQL_CONFIG.port}`;
|
|
290
310
|
async function createTestClient(schema, options) {
|
|
291
311
|
let workDir = options?.workDir;
|
|
292
312
|
let _schema;
|
|
293
313
|
const provider = options?.provider ?? getTestDbProvider() ?? "sqlite";
|
|
294
314
|
const dbName = options?.dbName ?? getTestDbName(provider);
|
|
295
|
-
const dbUrl = provider
|
|
315
|
+
const dbUrl = (0, import_ts_pattern2.match)(provider).with("sqlite", () => `file:${dbName}`).with("mysql", () => `${TEST_MYSQL_URL}/${dbName}`).with("postgresql", () => `${TEST_PG_URL}/${dbName}`).exhaustive();
|
|
296
316
|
let model;
|
|
297
317
|
if (typeof schema === "string") {
|
|
298
318
|
const generated = await generateTsSchema(schema, provider, dbUrl, options?.extraSourceFiles, void 0);
|
|
@@ -333,7 +353,7 @@ async function createTestClient(schema, options) {
|
|
|
333
353
|
if (options?.debug) {
|
|
334
354
|
console.log(`Work directory: ${workDir}`);
|
|
335
355
|
console.log(`Database name: ${dbName}`);
|
|
336
|
-
_options.log
|
|
356
|
+
_options.log ??= testLogger;
|
|
337
357
|
}
|
|
338
358
|
if (options?.dbFile) {
|
|
339
359
|
if (provider !== "sqlite") {
|
|
@@ -379,28 +399,10 @@ async function createTestClient(schema, options) {
|
|
|
379
399
|
stdio: options.debug ? "inherit" : "ignore"
|
|
380
400
|
});
|
|
381
401
|
} else {
|
|
382
|
-
|
|
383
|
-
(0, import_common_helpers2.invariant)(dbName, "dbName is required");
|
|
384
|
-
const pgClient = new import_pg.Client(TEST_PG_CONFIG);
|
|
385
|
-
await pgClient.connect();
|
|
386
|
-
await pgClient.query(`DROP DATABASE IF EXISTS "${dbName}"`);
|
|
387
|
-
await pgClient.query(`CREATE DATABASE "${dbName}"`);
|
|
388
|
-
await pgClient.end();
|
|
389
|
-
}
|
|
402
|
+
await prepareDatabase(provider, dbName);
|
|
390
403
|
}
|
|
391
404
|
}
|
|
392
|
-
|
|
393
|
-
_options.dialect = new import_kysely.PostgresDialect({
|
|
394
|
-
pool: new import_pg.Pool({
|
|
395
|
-
...TEST_PG_CONFIG,
|
|
396
|
-
database: dbName
|
|
397
|
-
})
|
|
398
|
-
});
|
|
399
|
-
} else {
|
|
400
|
-
_options.dialect = new import_kysely.SqliteDialect({
|
|
401
|
-
database: new import_better_sqlite3.default(import_node_path3.default.join(workDir, dbName))
|
|
402
|
-
});
|
|
403
|
-
}
|
|
405
|
+
_options.dialect = createDialect(provider, dbName, workDir);
|
|
404
406
|
let client = new import_orm.ZenStackClient(_schema, _options);
|
|
405
407
|
if (!options?.usePrismaPush && !options?.dbFile) {
|
|
406
408
|
await client.$pushSchema();
|
|
@@ -413,6 +415,39 @@ async function createTestClient(schema, options) {
|
|
|
413
415
|
return client;
|
|
414
416
|
}
|
|
415
417
|
__name(createTestClient, "createTestClient");
|
|
418
|
+
function createDialect(provider, dbName, workDir) {
|
|
419
|
+
return (0, import_ts_pattern2.match)(provider).with("postgresql", () => new import_kysely.PostgresDialect({
|
|
420
|
+
pool: new import_pg.Pool({
|
|
421
|
+
...TEST_PG_CONFIG,
|
|
422
|
+
database: dbName
|
|
423
|
+
})
|
|
424
|
+
})).with("mysql", () => new import_kysely.MysqlDialect({
|
|
425
|
+
pool: (0, import_mysql2.createPool)({
|
|
426
|
+
...TEST_MYSQL_CONFIG,
|
|
427
|
+
database: dbName
|
|
428
|
+
})
|
|
429
|
+
})).with("sqlite", () => new import_kysely.SqliteDialect({
|
|
430
|
+
database: new import_better_sqlite3.default(import_node_path3.default.join(workDir, dbName))
|
|
431
|
+
})).exhaustive();
|
|
432
|
+
}
|
|
433
|
+
__name(createDialect, "createDialect");
|
|
434
|
+
async function prepareDatabase(provider, dbName) {
|
|
435
|
+
if (provider === "postgresql") {
|
|
436
|
+
(0, import_common_helpers2.invariant)(dbName, "dbName is required");
|
|
437
|
+
const pgClient = new import_pg.Client(TEST_PG_CONFIG);
|
|
438
|
+
await pgClient.connect();
|
|
439
|
+
await pgClient.query(`DROP DATABASE IF EXISTS "${dbName}"`);
|
|
440
|
+
await pgClient.query(`CREATE DATABASE "${dbName}"`);
|
|
441
|
+
await pgClient.end();
|
|
442
|
+
} else if (provider === "mysql") {
|
|
443
|
+
(0, import_common_helpers2.invariant)(dbName, "dbName is required");
|
|
444
|
+
const mysqlPool = (0, import_mysql2.createPool)(TEST_MYSQL_CONFIG);
|
|
445
|
+
await mysqlPool.promise().query(`DROP DATABASE IF EXISTS \`${dbName}\``);
|
|
446
|
+
await mysqlPool.promise().query(`CREATE DATABASE \`${dbName}\``);
|
|
447
|
+
await mysqlPool.promise().end();
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
__name(prepareDatabase, "prepareDatabase");
|
|
416
451
|
async function createPolicyTestClient(schema, options) {
|
|
417
452
|
return createTestClient(schema, {
|
|
418
453
|
...options,
|
|
@@ -584,6 +619,8 @@ import_vitest3.expect.extend({
|
|
|
584
619
|
});
|
|
585
620
|
// Annotate the CommonJS export names for ESM import in node:
|
|
586
621
|
0 && (module.exports = {
|
|
622
|
+
TEST_MYSQL_CONFIG,
|
|
623
|
+
TEST_MYSQL_URL,
|
|
587
624
|
TEST_PG_CONFIG,
|
|
588
625
|
TEST_PG_URL,
|
|
589
626
|
createPolicyTestClient,
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/client.ts","../src/project.ts","../src/schema.ts","../src/utils.ts","../src/vitest-ext.ts"],"sourcesContent":["export * from './client';\nexport * from './project';\nexport * from './schema';\nexport * from './vitest-ext';\n","import { invariant } from '@zenstackhq/common-helpers';\nimport type { Model } from '@zenstackhq/language/ast';\nimport { ZenStackClient, type ClientContract, type ClientOptions } from '@zenstackhq/orm';\nimport type { SchemaDef } from '@zenstackhq/orm/schema';\nimport { PolicyPlugin } from '@zenstackhq/plugin-policy';\nimport { PrismaSchemaGenerator } from '@zenstackhq/sdk';\nimport SQLite from 'better-sqlite3';\nimport { glob } from 'glob';\nimport { PostgresDialect, SqliteDialect, type LogEvent } from 'kysely';\nimport { execSync } from 'node:child_process';\nimport { createHash } from 'node:crypto';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport { Client as PGClient, Pool } from 'pg';\nimport { expect } from 'vitest';\nimport { createTestProject } from './project';\nimport { generateTsSchema } from './schema';\nimport { loadDocumentWithPlugins } from './utils';\n\nexport function getTestDbProvider() {\n const val = process.env['TEST_DB_PROVIDER'] ?? 'sqlite';\n if (!['sqlite', 'postgresql'].includes(val!)) {\n throw new Error(`Invalid TEST_DB_PROVIDER value: ${val}`);\n }\n return val as 'sqlite' | 'postgresql';\n}\n\nexport const TEST_PG_CONFIG = {\n host: process.env['TEST_PG_HOST'] ?? 'localhost',\n port: process.env['TEST_PG_PORT'] ? parseInt(process.env['TEST_PG_PORT']) : 5432,\n user: process.env['TEST_PG_USER'] ?? 'postgres',\n password: process.env['TEST_PG_PASSWORD'] ?? 'postgres',\n};\n\nexport const TEST_PG_URL = `postgres://${TEST_PG_CONFIG.user}:${TEST_PG_CONFIG.password}@${TEST_PG_CONFIG.host}:${TEST_PG_CONFIG.port}`;\n\ntype ExtraTestClientOptions = {\n /**\n * Database provider\n */\n provider?: 'sqlite' | 'postgresql';\n\n /**\n * The main ZModel file. Only used when `usePrismaPush` is true and `schema` is an object.\n */\n schemaFile?: string;\n\n /**\n * Database name. If not provided, a name will be generated based on the test name.\n */\n dbName?: string;\n\n /**\n * Use `prisma db push` instead of ZenStack's `$pushSchema` for database initialization.\n */\n usePrismaPush?: boolean;\n\n /**\n * Extra source files to create and compile.\n */\n extraSourceFiles?: Record<string, string>;\n\n /**\n * Working directory for the test client. If not provided, a temporary directory will be created.\n */\n workDir?: string;\n\n /**\n * Debug mode.\n */\n debug?: boolean;\n\n /**\n * A sqlite database file to be used for the test. Only supported for sqlite provider.\n */\n dbFile?: string;\n\n /**\n * PostgreSQL extensions to be added to the datasource. Only supported for postgresql provider.\n */\n dataSourceExtensions?: string[];\n\n /**\n * Additional files to be copied to the working directory. The glob pattern is relative to the test file.\n */\n copyFiles?: {\n globPattern: string;\n destination: string;\n }[];\n};\n\nexport type CreateTestClientOptions<Schema extends SchemaDef> = Omit<ClientOptions<Schema>, 'dialect'> &\n ExtraTestClientOptions;\n\nexport async function createTestClient<\n Schema extends SchemaDef,\n Options extends ClientOptions<Schema>,\n CreateOptions = Omit<Options, 'dialect'>,\n>(schema: Schema, options?: CreateOptions): Promise<ClientContract<Schema, Options>>;\nexport async function createTestClient(schema: string, options?: CreateTestClientOptions<SchemaDef>): Promise<any>;\nexport async function createTestClient(\n schema: SchemaDef | string,\n options?: CreateTestClientOptions<SchemaDef>,\n): Promise<any> {\n let workDir = options?.workDir;\n let _schema: SchemaDef;\n const provider = options?.provider ?? getTestDbProvider() ?? 'sqlite';\n const dbName = options?.dbName ?? getTestDbName(provider);\n const dbUrl = provider === 'sqlite' ? `file:${dbName}` : `${TEST_PG_URL}/${dbName}`;\n\n let model: Model | undefined;\n\n if (typeof schema === 'string') {\n const generated = await generateTsSchema(schema, provider, dbUrl, options?.extraSourceFiles, undefined);\n workDir = generated.workDir;\n model = generated.model;\n // replace schema's provider\n _schema = {\n ...generated.schema,\n provider: {\n ...generated.schema.provider,\n type: provider,\n },\n } as SchemaDef;\n } else {\n // replace schema's provider\n _schema = {\n ...schema,\n provider: {\n type: provider,\n },\n };\n workDir ??= createTestProject();\n if (options?.schemaFile) {\n let schemaContent = fs.readFileSync(options.schemaFile, 'utf-8');\n if (dbUrl) {\n // replace `datasource db { }` section\n schemaContent = schemaContent.replace(\n /datasource\\s+db\\s*{[^}]*}/m,\n `datasource db {\n provider = '${provider}'\n url = '${dbUrl}'\n ${options.dataSourceExtensions ? `extensions = [${options.dataSourceExtensions.join(', ')}]` : ''}\n}`,\n );\n }\n fs.writeFileSync(path.join(workDir!, 'schema.zmodel'), schemaContent);\n }\n }\n\n invariant(workDir);\n\n const { plugins, ...rest } = options ?? {};\n const _options = {\n ...rest,\n } as ClientOptions<SchemaDef>;\n\n if (options?.debug) {\n console.log(`Work directory: ${workDir}`);\n console.log(`Database name: ${dbName}`);\n _options.log = testLogger;\n }\n\n // copy db file to workDir if specified\n if (options?.dbFile) {\n if (provider !== 'sqlite') {\n throw new Error('dbFile option is only supported for sqlite provider');\n }\n fs.copyFileSync(options.dbFile, path.join(workDir, dbName));\n }\n\n // copy additional files if specified\n if (options?.copyFiles) {\n const state = expect.getState();\n const currentTestPath = state.testPath;\n if (!currentTestPath) {\n throw new Error('Unable to determine current test file path');\n }\n for (const { globPattern, destination } of options.copyFiles) {\n const files = glob.sync(globPattern, { cwd: path.dirname(currentTestPath) });\n for (const file of files) {\n const src = path.resolve(path.dirname(currentTestPath), file);\n const dest = path.resolve(workDir, destination, path.basename(file));\n fs.mkdirSync(path.dirname(dest), { recursive: true });\n fs.copyFileSync(src, dest);\n }\n }\n }\n\n if (!options?.dbFile) {\n if (options?.usePrismaPush) {\n invariant(\n typeof schema === 'string' || options?.schemaFile,\n 'a schema file must be provided when using prisma db push',\n );\n if (!model) {\n const r = await loadDocumentWithPlugins(path.join(workDir, 'schema.zmodel'));\n if (!r.success) {\n throw new Error(r.errors.join('\\n'));\n }\n model = r.model;\n }\n const prismaSchema = new PrismaSchemaGenerator(model);\n const prismaSchemaText = await prismaSchema.generate();\n fs.writeFileSync(path.resolve(workDir!, 'schema.prisma'), prismaSchemaText);\n execSync('npx prisma db push --schema ./schema.prisma --skip-generate --force-reset', {\n cwd: workDir,\n stdio: options.debug ? 'inherit' : 'ignore',\n });\n } else {\n if (provider === 'postgresql') {\n invariant(dbName, 'dbName is required');\n const pgClient = new PGClient(TEST_PG_CONFIG);\n await pgClient.connect();\n await pgClient.query(`DROP DATABASE IF EXISTS \"${dbName}\"`);\n await pgClient.query(`CREATE DATABASE \"${dbName}\"`);\n await pgClient.end();\n }\n }\n }\n\n if (provider === 'postgresql') {\n _options.dialect = new PostgresDialect({\n pool: new Pool({\n ...TEST_PG_CONFIG,\n database: dbName,\n }),\n });\n } else {\n _options.dialect = new SqliteDialect({\n database: new SQLite(path.join(workDir!, dbName)),\n });\n }\n\n let client = new ZenStackClient(_schema, _options);\n\n if (!options?.usePrismaPush && !options?.dbFile) {\n await client.$pushSchema();\n }\n\n if (plugins) {\n for (const plugin of plugins) {\n client = client.$use(plugin);\n }\n }\n\n return client;\n}\n\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: Schema,\n options?: CreateTestClientOptions<Schema>,\n): Promise<ClientContract<Schema>>;\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: string,\n options?: CreateTestClientOptions<Schema>,\n): Promise<any>;\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: Schema | string,\n options?: CreateTestClientOptions<Schema>,\n): Promise<any> {\n return createTestClient(\n schema as any,\n {\n ...options,\n plugins: [...(options?.plugins ?? []), new PolicyPlugin()],\n } as any,\n );\n}\n\nexport function testLogger(e: LogEvent) {\n console.log(e.query.sql, e.query.parameters);\n}\n\nfunction getTestDbName(provider: string) {\n if (provider === 'sqlite') {\n return './test.db';\n }\n const testName = expect.getState().currentTestName ?? 'unnamed';\n const testPath = expect.getState().testPath ?? '';\n // digest test name\n const digest = createHash('md5')\n .update(testName + testPath)\n .digest('hex');\n // compute a database name based on test name\n return (\n 'test_' +\n testName\n .toLowerCase()\n .replace(/[^a-z0-9_]/g, '_')\n .replace(/_+/g, '_')\n .substring(0, 30) +\n digest.slice(0, 6)\n );\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\nimport tmp from 'tmp';\n\nexport function createTestProject(zmodelContent?: string) {\n const { name: workDir } = tmp.dirSync({ unsafeCleanup: true });\n\n fs.mkdirSync(path.join(workDir, 'node_modules'));\n\n // symlink all entries from \"node_modules\"\n const nodeModules = fs.readdirSync(path.join(__dirname, '../node_modules'));\n for (const entry of nodeModules) {\n if (entry.startsWith('@zenstackhq')) {\n continue;\n }\n fs.symlinkSync(\n path.join(__dirname, '../node_modules', entry),\n path.join(workDir, 'node_modules', entry),\n 'dir',\n );\n }\n\n // in addition, symlink zenstack packages\n const zenstackPackages = ['language', 'sdk', 'orm', 'cli'];\n fs.mkdirSync(path.join(workDir, 'node_modules/@zenstackhq'));\n for (const pkg of zenstackPackages) {\n fs.symlinkSync(\n path.join(__dirname, `../../${pkg}`),\n path.join(workDir, `node_modules/@zenstackhq/${pkg}`),\n 'dir',\n );\n }\n\n fs.writeFileSync(\n path.join(workDir, 'package.json'),\n JSON.stringify(\n {\n name: 'test',\n version: '1.0.0',\n type: 'module',\n },\n null,\n 4,\n ),\n );\n\n fs.writeFileSync(\n path.join(workDir, 'tsconfig.json'),\n JSON.stringify(\n {\n compilerOptions: {\n module: 'ESNext',\n target: 'ESNext',\n moduleResolution: 'Bundler',\n esModuleInterop: true,\n skipLibCheck: true,\n strict: true,\n },\n include: ['**/*.ts'],\n },\n null,\n 4,\n ),\n );\n\n if (zmodelContent) {\n fs.writeFileSync(path.join(workDir, 'schema.zmodel'), zmodelContent);\n }\n\n return workDir;\n}\n","import { invariant } from '@zenstackhq/common-helpers';\nimport type { SchemaDef } from '@zenstackhq/schema';\nimport { TsSchemaGenerator } from '@zenstackhq/sdk';\nimport { execSync } from 'node:child_process';\nimport crypto from 'node:crypto';\nimport fs from 'node:fs';\nimport os from 'node:os';\nimport path from 'node:path';\nimport { match } from 'ts-pattern';\nimport { expect } from 'vitest';\nimport { createTestProject } from './project';\nimport { loadDocumentWithPlugins } from './utils';\n\nfunction makePrelude(provider: 'sqlite' | 'postgresql', dbUrl?: string) {\n return match(provider)\n .with('sqlite', () => {\n return `\ndatasource db {\n provider = 'sqlite'\n url = '${dbUrl ?? 'file:./test.db'}'\n}\n`;\n })\n .with('postgresql', () => {\n return `\ndatasource db {\n provider = 'postgresql'\n url = '${dbUrl ?? 'postgres://postgres:postgres@localhost:5432/db'}'\n}\n`;\n })\n .exhaustive();\n}\n\nfunction replacePlaceholders(schemaText: string, provider: 'sqlite' | 'postgresql', dbUrl: string | undefined) {\n const url = dbUrl ?? (provider === 'sqlite' ? 'file:./test.db' : 'postgres://postgres:postgres@localhost:5432/db');\n return schemaText.replace(/\\$DB_URL/g, url).replace(/\\$PROVIDER/g, provider);\n}\n\nexport async function generateTsSchema(\n schemaText: string,\n provider: 'sqlite' | 'postgresql' = 'sqlite',\n dbUrl?: string,\n extraSourceFiles?: Record<string, string>,\n withLiteSchema?: boolean,\n) {\n const workDir = createTestProject();\n\n const zmodelPath = path.join(workDir, 'schema.zmodel');\n const noPrelude = schemaText.includes('datasource ');\n fs.writeFileSync(\n zmodelPath,\n `${noPrelude ? '' : makePrelude(provider, dbUrl)}\\n\\n${replacePlaceholders(schemaText, provider, dbUrl)}`,\n );\n\n const result = await loadDocumentWithPlugins(zmodelPath);\n if (!result.success) {\n throw new Error(`Failed to load schema from ${zmodelPath}: ${result.errors}`);\n }\n\n const generator = new TsSchemaGenerator();\n await generator.generate(result.model, { outDir: workDir, lite: withLiteSchema });\n\n if (extraSourceFiles) {\n for (const [fileName, content] of Object.entries(extraSourceFiles)) {\n const filePath = path.resolve(workDir, !fileName.endsWith('.ts') ? `${fileName}.ts` : fileName);\n fs.mkdirSync(path.dirname(filePath), { recursive: true });\n fs.writeFileSync(filePath, content);\n }\n }\n\n // compile the generated TS schema\n return { ...(await compileAndLoad(workDir)), model: result.model };\n}\n\nasync function compileAndLoad(workDir: string) {\n execSync('npx tsc', {\n cwd: workDir,\n stdio: 'inherit',\n });\n\n // load the schema module\n const module = await import(path.join(workDir, 'schema.js'));\n\n let moduleLite: any;\n try {\n moduleLite = await import(path.join(workDir, 'schema-lite.js'));\n } catch {\n // ignore\n }\n return { workDir, schema: module.schema as SchemaDef, schemaLite: moduleLite?.schema as SchemaDef | undefined };\n}\n\nexport function generateTsSchemaFromFile(filePath: string) {\n const schemaText = fs.readFileSync(filePath, 'utf8');\n return generateTsSchema(schemaText);\n}\n\nexport async function generateTsSchemaInPlace(schemaPath: string) {\n const workDir = path.dirname(schemaPath);\n const result = await loadDocumentWithPlugins(schemaPath);\n if (!result.success) {\n throw new Error(`Failed to load schema from ${schemaPath}: ${result.errors}`);\n }\n const generator = new TsSchemaGenerator();\n await generator.generate(result.model, { outDir: workDir });\n return compileAndLoad(workDir);\n}\n\nexport async function loadSchema(schema: string, additionalSchemas?: Record<string, string>) {\n if (!schema.includes('datasource ')) {\n schema = `${makePrelude('sqlite')}\\n\\n${schema}`;\n }\n\n // create a temp folder\n const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'zenstack-schema'));\n\n // create a temp file\n const tempFile = path.join(tempDir, `schema.zmodel`);\n fs.writeFileSync(tempFile, schema);\n\n if (additionalSchemas) {\n for (const [fileName, content] of Object.entries(additionalSchemas)) {\n let name = fileName;\n if (!name.endsWith('.zmodel')) {\n name += '.zmodel';\n }\n const filePath = path.join(tempDir, name);\n fs.writeFileSync(filePath, content);\n }\n }\n\n const r = await loadDocumentWithPlugins(tempFile);\n expect(r).toSatisfy(\n (r) => r.success,\n `Failed to load schema: ${(r as any).errors?.map((e: any) => e.toString()).join(', ')}`,\n );\n invariant(r.success);\n return r.model;\n}\n\nexport async function loadSchemaWithError(schema: string, error: string | RegExp) {\n if (!schema.includes('datasource ')) {\n schema = `${makePrelude('sqlite')}\\n\\n${schema}`;\n }\n\n // create a temp file\n const tempFile = path.join(os.tmpdir(), `zenstack-schema-${crypto.randomUUID()}.zmodel`);\n fs.writeFileSync(tempFile, schema);\n const r = await loadDocumentWithPlugins(tempFile);\n expect(r.success).toBe(false);\n invariant(!r.success);\n if (typeof error === 'string') {\n expect(r).toSatisfy(\n (r) => r.errors.some((e: any) => e.toString().toLowerCase().includes(error.toLowerCase())),\n `Expected error message to include \"${error}\" but got: ${r.errors.map((e: any) => e.toString()).join(', ')}`,\n );\n } else {\n expect(r).toSatisfy(\n (r) => r.errors.some((e: any) => error.test(e)),\n `Expected error message to match \"${error}\" but got: ${r.errors.map((e: any) => e.toString()).join(', ')}`,\n );\n }\n}\n","import { loadDocument } from '@zenstackhq/language';\n\nexport function loadDocumentWithPlugins(filePath: string) {\n const pluginModelFiles = [require.resolve('@zenstackhq/plugin-policy/plugin.zmodel')];\n return loadDocument(filePath, pluginModelFiles);\n}\n","import { ORMError, ORMErrorReason } from '@zenstackhq/orm';\nimport { expect } from 'vitest';\n\nfunction isPromise(value: any) {\n return typeof value.then === 'function' && typeof value.catch === 'function';\n}\n\nfunction expectErrorReason(err: any, errorReason: ORMErrorReason) {\n if (err instanceof ORMError && err.reason === errorReason) {\n return {\n message: () => '',\n pass: true,\n };\n } else {\n return {\n message: () => `expected ORMError of reason ${errorReason}, got ${err}`,\n pass: false,\n };\n }\n}\n\nfunction expectErrorMessages(expectedMessages: string[], message: string) {\n for (const m of expectedMessages) {\n if (!message.toLowerCase().includes(m.toLowerCase())) {\n return {\n message: () => `expected message not found in error: ${m}, got message: ${message}`,\n pass: false,\n };\n }\n }\n return undefined;\n}\n\nexpect.extend({\n async toResolveTruthy(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: !!r,\n message: () => `Expected promise to resolve to a truthy value, but got ${r}`,\n };\n },\n\n async toResolveFalsy(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: !r,\n message: () => `Expected promise to resolve to a falsy value, but got ${r}`,\n };\n },\n\n async toResolveNull(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: r === null,\n message: () => `Expected promise to resolve to a null value, but got ${r}`,\n };\n },\n\n async toResolveWithLength(received: Promise<unknown>, length: number) {\n const r = await received;\n return {\n pass: Array.isArray(r) && r.length === length,\n message: () => `Expected promise to resolve with an array with length ${length}, but got ${r}`,\n };\n },\n\n async toBeRejectedNotFound(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n return expectErrorReason(err, ORMErrorReason.NOT_FOUND);\n }\n return {\n message: () => `expected NotFoundError, got no error`,\n pass: false,\n };\n },\n\n async toBeRejectedByPolicy(received: Promise<unknown>, expectedMessages?: string[]) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n if (expectedMessages && err instanceof ORMError && err.reason === ORMErrorReason.REJECTED_BY_POLICY) {\n const r = expectErrorMessages(expectedMessages, err.message || '');\n if (r) {\n return r;\n }\n }\n return expectErrorReason(err, ORMErrorReason.REJECTED_BY_POLICY);\n }\n return {\n message: () => `expected PolicyError, got no error`,\n pass: false,\n };\n },\n\n async toBeRejectedByValidation(received: Promise<unknown>, expectedMessages?: string[]) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n if (expectedMessages && err instanceof ORMError && err.reason === ORMErrorReason.INVALID_INPUT) {\n const r = expectErrorMessages(expectedMessages, err.message || '');\n if (r) {\n return r;\n }\n }\n return expectErrorReason(err, ORMErrorReason.INVALID_INPUT);\n }\n return {\n message: () => `expected InputValidationError, got no error`,\n pass: false,\n };\n },\n});\n"],"mappings":"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","names":["import_common_helpers","import_sdk","import_node_child_process","import_node_crypto","import_node_fs","import_node_path","import_vitest","createTestProject","zmodelContent","name","workDir","tmp","dirSync","unsafeCleanup","fs","mkdirSync","path","join","nodeModules","readdirSync","__dirname","entry","startsWith","symlinkSync","zenstackPackages","pkg","writeFileSync","JSON","stringify","version","type","compilerOptions","module","target","moduleResolution","esModuleInterop","skipLibCheck","strict","include","import_node_fs","import_node_path","loadDocumentWithPlugins","filePath","pluginModelFiles","require","loadDocument","makePrelude","provider","dbUrl","match","with","exhaustive","replacePlaceholders","schemaText","url","replace","generateTsSchema","extraSourceFiles","withLiteSchema","workDir","createTestProject","zmodelPath","path","join","noPrelude","includes","fs","writeFileSync","result","loadDocumentWithPlugins","success","Error","errors","generator","TsSchemaGenerator","generate","model","outDir","lite","fileName","content","Object","entries","filePath","resolve","endsWith","mkdirSync","dirname","recursive","compileAndLoad","execSync","cwd","stdio","module","moduleLite","schema","schemaLite","generateTsSchemaFromFile","readFileSync","generateTsSchemaInPlace","schemaPath","loadSchema","additionalSchemas","tempDir","mkdtempSync","os","tmpdir","tempFile","name","r","expect","toSatisfy","map","e","toString","invariant","loadSchemaWithError","error","crypto","randomUUID","toBe","some","toLowerCase","test","getTestDbProvider","val","process","env","includes","Error","TEST_PG_CONFIG","host","port","parseInt","user","password","TEST_PG_URL","createTestClient","schema","options","workDir","_schema","provider","dbName","getTestDbName","dbUrl","model","generated","generateTsSchema","extraSourceFiles","undefined","type","createTestProject","schemaFile","schemaContent","fs","readFileSync","replace","dataSourceExtensions","join","writeFileSync","path","invariant","plugins","rest","_options","debug","console","log","testLogger","dbFile","copyFileSync","copyFiles","state","expect","getState","currentTestPath","testPath","globPattern","destination","files","glob","sync","cwd","dirname","file","src","resolve","dest","basename","mkdirSync","recursive","usePrismaPush","r","loadDocumentWithPlugins","success","errors","prismaSchema","PrismaSchemaGenerator","prismaSchemaText","generate","execSync","stdio","pgClient","PGClient","connect","query","end","dialect","PostgresDialect","pool","Pool","database","SqliteDialect","SQLite","client","ZenStackClient","$pushSchema","plugin","$use","createPolicyTestClient","PolicyPlugin","e","sql","parameters","testName","currentTestName","digest","createHash","update","toLowerCase","substring","slice","import_orm","import_vitest","isPromise","value","then","catch","expectErrorReason","err","errorReason","ORMError","reason","message","pass","expectErrorMessages","expectedMessages","m","toLowerCase","includes","undefined","expect","extend","toResolveTruthy","received","r","toResolveFalsy","toResolveNull","toResolveWithLength","length","Array","isArray","toBeRejectedNotFound","ORMErrorReason","NOT_FOUND","toBeRejectedByPolicy","REJECTED_BY_POLICY","toBeRejectedByValidation","INVALID_INPUT"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/client.ts","../src/project.ts","../src/schema.ts","../src/utils.ts","../src/vitest-ext.ts"],"sourcesContent":["export * from './client';\nexport * from './project';\nexport * from './schema';\nexport * from './vitest-ext';\n","import { invariant } from '@zenstackhq/common-helpers';\nimport type { Model } from '@zenstackhq/language/ast';\nimport { ZenStackClient, type ClientContract, type ClientOptions } from '@zenstackhq/orm';\nimport type { DataSourceProviderType, SchemaDef } from '@zenstackhq/orm/schema';\nimport { PolicyPlugin } from '@zenstackhq/plugin-policy';\nimport { PrismaSchemaGenerator } from '@zenstackhq/sdk';\nimport SQLite from 'better-sqlite3';\nimport { glob } from 'glob';\nimport { MysqlDialect, PostgresDialect, SqliteDialect, type LogEvent } from 'kysely';\nimport { createPool as createMysqlPool } from 'mysql2';\nimport { execSync } from 'node:child_process';\nimport { createHash } from 'node:crypto';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport { Client as PGClient, Pool } from 'pg';\nimport { match } from 'ts-pattern';\nimport { expect } from 'vitest';\nimport { createTestProject } from './project';\nimport { generateTsSchema } from './schema';\nimport { loadDocumentWithPlugins } from './utils';\n\nexport function getTestDbProvider() {\n const val = process.env['TEST_DB_PROVIDER'] ?? 'sqlite';\n if (!['sqlite', 'postgresql', 'mysql'].includes(val!)) {\n throw new Error(`Invalid TEST_DB_PROVIDER value: ${val}`);\n }\n return val as 'sqlite' | 'postgresql' | 'mysql';\n}\n\nexport const TEST_PG_CONFIG = {\n host: process.env['TEST_PG_HOST'] ?? 'localhost',\n port: process.env['TEST_PG_PORT'] ? parseInt(process.env['TEST_PG_PORT']) : 5432,\n user: process.env['TEST_PG_USER'] ?? 'postgres',\n password: process.env['TEST_PG_PASSWORD'] ?? 'postgres',\n};\n\nexport const TEST_PG_URL = `postgres://${TEST_PG_CONFIG.user}:${TEST_PG_CONFIG.password}@${TEST_PG_CONFIG.host}:${TEST_PG_CONFIG.port}`;\n\nexport const TEST_MYSQL_CONFIG = {\n host: process.env['TEST_MYSQL_HOST'] ?? 'localhost',\n port: process.env['TEST_MYSQL_PORT'] ? parseInt(process.env['TEST_MYSQL_PORT']) : 3306,\n user: process.env['TEST_MYSQL_USER'] ?? 'root',\n password: process.env['TEST_MYSQL_PASSWORD'] ?? 'mysql',\n timezone: 'Z',\n};\n\nexport const TEST_MYSQL_URL = `mysql://${TEST_MYSQL_CONFIG.user}:${TEST_MYSQL_CONFIG.password}@${TEST_MYSQL_CONFIG.host}:${TEST_MYSQL_CONFIG.port}`;\n\ntype ExtraTestClientOptions = {\n /**\n * Database provider\n */\n provider?: 'sqlite' | 'postgresql' | 'mysql';\n\n /**\n * The main ZModel file. Only used when `usePrismaPush` is true and `schema` is an object.\n */\n schemaFile?: string;\n\n /**\n * Database name. If not provided, a name will be generated based on the test name.\n */\n dbName?: string;\n\n /**\n * Use `prisma db push` instead of ZenStack's `$pushSchema` for database initialization.\n */\n usePrismaPush?: boolean;\n\n /**\n * Extra source files to create and compile.\n */\n extraSourceFiles?: Record<string, string>;\n\n /**\n * Working directory for the test client. If not provided, a temporary directory will be created.\n */\n workDir?: string;\n\n /**\n * Debug mode.\n */\n debug?: boolean;\n\n /**\n * A sqlite database file to be used for the test. Only supported for sqlite provider.\n */\n dbFile?: string;\n\n /**\n * PostgreSQL extensions to be added to the datasource. Only supported for postgresql provider.\n */\n dataSourceExtensions?: string[];\n\n /**\n * Additional files to be copied to the working directory. The glob pattern is relative to the test file.\n */\n copyFiles?: {\n globPattern: string;\n destination: string;\n }[];\n};\n\nexport type CreateTestClientOptions<Schema extends SchemaDef> = Omit<ClientOptions<Schema>, 'dialect'> &\n ExtraTestClientOptions;\n\nexport async function createTestClient<\n Schema extends SchemaDef,\n Options extends ClientOptions<Schema>,\n CreateOptions = Omit<Options, 'dialect'>,\n>(schema: Schema, options?: CreateOptions): Promise<ClientContract<Schema, Options>>;\nexport async function createTestClient(schema: string, options?: CreateTestClientOptions<SchemaDef>): Promise<any>;\nexport async function createTestClient(\n schema: SchemaDef | string,\n options?: CreateTestClientOptions<SchemaDef>,\n): Promise<any> {\n let workDir = options?.workDir;\n let _schema: SchemaDef;\n const provider = options?.provider ?? getTestDbProvider() ?? 'sqlite';\n const dbName = options?.dbName ?? getTestDbName(provider);\n const dbUrl = match(provider)\n .with('sqlite', () => `file:${dbName}`)\n .with('mysql', () => `${TEST_MYSQL_URL}/${dbName}`)\n .with('postgresql', () => `${TEST_PG_URL}/${dbName}`)\n .exhaustive();\n let model: Model | undefined;\n\n if (typeof schema === 'string') {\n const generated = await generateTsSchema(schema, provider, dbUrl, options?.extraSourceFiles, undefined);\n workDir = generated.workDir;\n model = generated.model;\n // replace schema's provider\n _schema = {\n ...generated.schema,\n provider: {\n ...generated.schema.provider,\n type: provider,\n },\n } as SchemaDef;\n } else {\n // replace schema's provider\n _schema = {\n ...schema,\n provider: {\n type: provider,\n },\n };\n workDir ??= createTestProject();\n if (options?.schemaFile) {\n let schemaContent = fs.readFileSync(options.schemaFile, 'utf-8');\n if (dbUrl) {\n // replace `datasource db { }` section\n schemaContent = schemaContent.replace(\n /datasource\\s+db\\s*{[^}]*}/m,\n `datasource db {\n provider = '${provider}'\n url = '${dbUrl}'\n ${options.dataSourceExtensions ? `extensions = [${options.dataSourceExtensions.join(', ')}]` : ''}\n}`,\n );\n }\n fs.writeFileSync(path.join(workDir!, 'schema.zmodel'), schemaContent);\n }\n }\n\n invariant(workDir);\n\n const { plugins, ...rest } = options ?? {};\n const _options = {\n ...rest,\n } as ClientOptions<SchemaDef>;\n\n if (options?.debug) {\n console.log(`Work directory: ${workDir}`);\n console.log(`Database name: ${dbName}`);\n _options.log ??= testLogger;\n }\n\n // copy db file to workDir if specified\n if (options?.dbFile) {\n if (provider !== 'sqlite') {\n throw new Error('dbFile option is only supported for sqlite provider');\n }\n fs.copyFileSync(options.dbFile, path.join(workDir, dbName));\n }\n\n // copy additional files if specified\n if (options?.copyFiles) {\n const state = expect.getState();\n const currentTestPath = state.testPath;\n if (!currentTestPath) {\n throw new Error('Unable to determine current test file path');\n }\n for (const { globPattern, destination } of options.copyFiles) {\n const files = glob.sync(globPattern, { cwd: path.dirname(currentTestPath) });\n for (const file of files) {\n const src = path.resolve(path.dirname(currentTestPath), file);\n const dest = path.resolve(workDir, destination, path.basename(file));\n fs.mkdirSync(path.dirname(dest), { recursive: true });\n fs.copyFileSync(src, dest);\n }\n }\n }\n\n if (!options?.dbFile) {\n if (options?.usePrismaPush) {\n invariant(\n typeof schema === 'string' || options?.schemaFile,\n 'a schema file must be provided when using prisma db push',\n );\n if (!model) {\n const r = await loadDocumentWithPlugins(path.join(workDir, 'schema.zmodel'));\n if (!r.success) {\n throw new Error(r.errors.join('\\n'));\n }\n model = r.model;\n }\n const prismaSchema = new PrismaSchemaGenerator(model);\n const prismaSchemaText = await prismaSchema.generate();\n fs.writeFileSync(path.resolve(workDir!, 'schema.prisma'), prismaSchemaText);\n execSync('npx prisma db push --schema ./schema.prisma --skip-generate --force-reset', {\n cwd: workDir,\n stdio: options.debug ? 'inherit' : 'ignore',\n });\n } else {\n await prepareDatabase(provider, dbName);\n }\n }\n\n // create Kysely dialect\n _options.dialect = createDialect(provider, dbName, workDir);\n\n let client = new ZenStackClient(_schema, _options);\n\n if (!options?.usePrismaPush && !options?.dbFile) {\n await client.$pushSchema();\n }\n\n // install plugins\n if (plugins) {\n for (const plugin of plugins) {\n client = client.$use(plugin);\n }\n }\n\n return client;\n}\n\nfunction createDialect(provider: DataSourceProviderType, dbName: string, workDir: string) {\n return match(provider)\n .with(\n 'postgresql',\n () =>\n new PostgresDialect({\n pool: new Pool({\n ...TEST_PG_CONFIG,\n database: dbName,\n }),\n }),\n )\n .with(\n 'mysql',\n () =>\n new MysqlDialect({\n pool: createMysqlPool({\n ...TEST_MYSQL_CONFIG,\n database: dbName,\n }),\n }),\n )\n .with(\n 'sqlite',\n () =>\n new SqliteDialect({\n database: new SQLite(path.join(workDir!, dbName)),\n }),\n )\n .exhaustive();\n}\n\nasync function prepareDatabase(provider: string, dbName: string) {\n if (provider === 'postgresql') {\n invariant(dbName, 'dbName is required');\n const pgClient = new PGClient(TEST_PG_CONFIG);\n await pgClient.connect();\n await pgClient.query(`DROP DATABASE IF EXISTS \"${dbName}\"`);\n await pgClient.query(`CREATE DATABASE \"${dbName}\"`);\n await pgClient.end();\n } else if (provider === 'mysql') {\n invariant(dbName, 'dbName is required');\n const mysqlPool = createMysqlPool(TEST_MYSQL_CONFIG);\n await mysqlPool.promise().query(`DROP DATABASE IF EXISTS \\`${dbName}\\``);\n await mysqlPool.promise().query(`CREATE DATABASE \\`${dbName}\\``);\n await mysqlPool.promise().end();\n }\n}\n\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: Schema,\n options?: CreateTestClientOptions<Schema>,\n): Promise<ClientContract<Schema>>;\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: string,\n options?: CreateTestClientOptions<Schema>,\n): Promise<any>;\nexport async function createPolicyTestClient<Schema extends SchemaDef>(\n schema: Schema | string,\n options?: CreateTestClientOptions<Schema>,\n): Promise<any> {\n return createTestClient(\n schema as any,\n {\n ...options,\n plugins: [...(options?.plugins ?? []), new PolicyPlugin()],\n } as any,\n );\n}\n\nexport function testLogger(e: LogEvent) {\n console.log(e.query.sql, e.query.parameters);\n}\n\nfunction getTestDbName(provider: string) {\n if (provider === 'sqlite') {\n return './test.db';\n }\n const testName = expect.getState().currentTestName ?? 'unnamed';\n const testPath = expect.getState().testPath ?? '';\n // digest test name\n const digest = createHash('md5')\n .update(testName + testPath)\n .digest('hex');\n // compute a database name based on test name\n return (\n 'test_' +\n testName\n .toLowerCase()\n .replace(/[^a-z0-9_]/g, '_')\n .replace(/_+/g, '_')\n .substring(0, 30) +\n digest.slice(0, 6)\n );\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\nimport tmp from 'tmp';\n\nexport function createTestProject(zmodelContent?: string) {\n const { name: workDir } = tmp.dirSync({ unsafeCleanup: true });\n\n fs.mkdirSync(path.join(workDir, 'node_modules'));\n\n // symlink all entries from \"node_modules\"\n const nodeModules = fs.readdirSync(path.join(__dirname, '../node_modules'));\n for (const entry of nodeModules) {\n if (entry.startsWith('@zenstackhq')) {\n continue;\n }\n fs.symlinkSync(\n path.join(__dirname, '../node_modules', entry),\n path.join(workDir, 'node_modules', entry),\n 'dir',\n );\n }\n\n // in addition, symlink zenstack packages\n const zenstackPackages = ['language', 'sdk', 'orm', 'cli'];\n fs.mkdirSync(path.join(workDir, 'node_modules/@zenstackhq'));\n for (const pkg of zenstackPackages) {\n fs.symlinkSync(\n path.join(__dirname, `../../${pkg}`),\n path.join(workDir, `node_modules/@zenstackhq/${pkg}`),\n 'dir',\n );\n }\n\n fs.writeFileSync(\n path.join(workDir, 'package.json'),\n JSON.stringify(\n {\n name: 'test',\n version: '1.0.0',\n type: 'module',\n },\n null,\n 4,\n ),\n );\n\n fs.writeFileSync(\n path.join(workDir, 'tsconfig.json'),\n JSON.stringify(\n {\n compilerOptions: {\n module: 'ESNext',\n target: 'ESNext',\n moduleResolution: 'Bundler',\n esModuleInterop: true,\n skipLibCheck: true,\n strict: true,\n },\n include: ['**/*.ts'],\n },\n null,\n 4,\n ),\n );\n\n if (zmodelContent) {\n fs.writeFileSync(path.join(workDir, 'schema.zmodel'), zmodelContent);\n }\n\n return workDir;\n}\n","import { invariant } from '@zenstackhq/common-helpers';\nimport type { DataSourceProviderType, SchemaDef } from '@zenstackhq/schema';\nimport { TsSchemaGenerator } from '@zenstackhq/sdk';\nimport { execSync } from 'node:child_process';\nimport crypto from 'node:crypto';\nimport fs from 'node:fs';\nimport os from 'node:os';\nimport path from 'node:path';\nimport { match } from 'ts-pattern';\nimport { expect } from 'vitest';\nimport { createTestProject } from './project';\nimport { loadDocumentWithPlugins } from './utils';\n\nfunction makePrelude(provider: DataSourceProviderType, dbUrl?: string) {\n return match(provider)\n .with('sqlite', () => {\n return `\ndatasource db {\n provider = 'sqlite'\n url = '${dbUrl ?? 'file:./test.db'}'\n}\n`;\n })\n .with('postgresql', () => {\n return `\ndatasource db {\n provider = 'postgresql'\n url = '${dbUrl ?? 'postgres://postgres:postgres@localhost:5432/db'}'\n}\n`;\n })\n .with('mysql', () => {\n return `\ndatasource db {\n provider = 'mysql'\n url = '${dbUrl ?? 'mysql://root:mysql@localhost:3306/db'}'\n}\n`;\n })\n .exhaustive();\n}\n\nfunction replacePlaceholders(\n schemaText: string,\n provider: 'sqlite' | 'postgresql' | 'mysql',\n dbUrl: string | undefined,\n) {\n const url =\n dbUrl ??\n (provider === 'sqlite'\n ? 'file:./test.db'\n : provider === 'mysql'\n ? 'mysql://root:mysql@localhost:3306/db'\n : 'postgres://postgres:postgres@localhost:5432/db');\n return schemaText.replace(/\\$DB_URL/g, url).replace(/\\$PROVIDER/g, provider);\n}\n\nexport async function generateTsSchema(\n schemaText: string,\n provider: DataSourceProviderType = 'sqlite',\n dbUrl?: string,\n extraSourceFiles?: Record<string, string>,\n withLiteSchema?: boolean,\n) {\n const workDir = createTestProject();\n\n const zmodelPath = path.join(workDir, 'schema.zmodel');\n const noPrelude = schemaText.includes('datasource ');\n fs.writeFileSync(\n zmodelPath,\n `${noPrelude ? '' : makePrelude(provider, dbUrl)}\\n\\n${replacePlaceholders(schemaText, provider, dbUrl)}`,\n );\n\n const result = await loadDocumentWithPlugins(zmodelPath);\n if (!result.success) {\n throw new Error(`Failed to load schema from ${zmodelPath}: ${result.errors}`);\n }\n\n const generator = new TsSchemaGenerator();\n await generator.generate(result.model, { outDir: workDir, lite: withLiteSchema });\n\n if (extraSourceFiles) {\n for (const [fileName, content] of Object.entries(extraSourceFiles)) {\n const filePath = path.resolve(workDir, !fileName.endsWith('.ts') ? `${fileName}.ts` : fileName);\n fs.mkdirSync(path.dirname(filePath), { recursive: true });\n fs.writeFileSync(filePath, content);\n }\n }\n\n // compile the generated TS schema\n return { ...(await compileAndLoad(workDir)), model: result.model };\n}\n\nasync function compileAndLoad(workDir: string) {\n execSync('npx tsc', {\n cwd: workDir,\n stdio: 'inherit',\n });\n\n // load the schema module\n const module = await import(path.join(workDir, 'schema.js'));\n\n let moduleLite: any;\n try {\n moduleLite = await import(path.join(workDir, 'schema-lite.js'));\n } catch {\n // ignore\n }\n return { workDir, schema: module.schema as SchemaDef, schemaLite: moduleLite?.schema as SchemaDef | undefined };\n}\n\nexport function generateTsSchemaFromFile(filePath: string) {\n const schemaText = fs.readFileSync(filePath, 'utf8');\n return generateTsSchema(schemaText);\n}\n\nexport async function generateTsSchemaInPlace(schemaPath: string) {\n const workDir = path.dirname(schemaPath);\n const result = await loadDocumentWithPlugins(schemaPath);\n if (!result.success) {\n throw new Error(`Failed to load schema from ${schemaPath}: ${result.errors}`);\n }\n const generator = new TsSchemaGenerator();\n await generator.generate(result.model, { outDir: workDir });\n return compileAndLoad(workDir);\n}\n\nexport async function loadSchema(schema: string, additionalSchemas?: Record<string, string>) {\n if (!schema.includes('datasource ')) {\n schema = `${makePrelude('sqlite')}\\n\\n${schema}`;\n }\n\n // create a temp folder\n const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'zenstack-schema'));\n\n // create a temp file\n const tempFile = path.join(tempDir, `schema.zmodel`);\n fs.writeFileSync(tempFile, schema);\n\n if (additionalSchemas) {\n for (const [fileName, content] of Object.entries(additionalSchemas)) {\n let name = fileName;\n if (!name.endsWith('.zmodel')) {\n name += '.zmodel';\n }\n const filePath = path.join(tempDir, name);\n fs.writeFileSync(filePath, content);\n }\n }\n\n const r = await loadDocumentWithPlugins(tempFile);\n expect(r).toSatisfy(\n (r) => r.success,\n `Failed to load schema: ${(r as any).errors?.map((e: any) => e.toString()).join(', ')}`,\n );\n invariant(r.success);\n return r.model;\n}\n\nexport async function loadSchemaWithError(schema: string, error: string | RegExp) {\n if (!schema.includes('datasource ')) {\n schema = `${makePrelude('sqlite')}\\n\\n${schema}`;\n }\n\n // create a temp file\n const tempFile = path.join(os.tmpdir(), `zenstack-schema-${crypto.randomUUID()}.zmodel`);\n fs.writeFileSync(tempFile, schema);\n const r = await loadDocumentWithPlugins(tempFile);\n expect(r.success).toBe(false);\n invariant(!r.success);\n if (typeof error === 'string') {\n expect(r).toSatisfy(\n (r) => r.errors.some((e: any) => e.toString().toLowerCase().includes(error.toLowerCase())),\n `Expected error message to include \"${error}\" but got: ${r.errors.map((e: any) => e.toString()).join(', ')}`,\n );\n } else {\n expect(r).toSatisfy(\n (r) => r.errors.some((e: any) => error.test(e)),\n `Expected error message to match \"${error}\" but got: ${r.errors.map((e: any) => e.toString()).join(', ')}`,\n );\n }\n}\n","import { loadDocument } from '@zenstackhq/language';\n\nexport function loadDocumentWithPlugins(filePath: string) {\n const pluginModelFiles = [require.resolve('@zenstackhq/plugin-policy/plugin.zmodel')];\n return loadDocument(filePath, pluginModelFiles);\n}\n","import { ORMError, ORMErrorReason } from '@zenstackhq/orm';\nimport { expect } from 'vitest';\n\nfunction isPromise(value: any) {\n return typeof value.then === 'function' && typeof value.catch === 'function';\n}\n\nfunction expectErrorReason(err: any, errorReason: ORMErrorReason) {\n if (err instanceof ORMError && err.reason === errorReason) {\n return {\n message: () => '',\n pass: true,\n };\n } else {\n return {\n message: () => `expected ORMError of reason ${errorReason}, got ${err}`,\n pass: false,\n };\n }\n}\n\nfunction expectErrorMessages(expectedMessages: string[], message: string) {\n for (const m of expectedMessages) {\n if (!message.toLowerCase().includes(m.toLowerCase())) {\n return {\n message: () => `expected message not found in error: ${m}, got message: ${message}`,\n pass: false,\n };\n }\n }\n return undefined;\n}\n\nexpect.extend({\n async toResolveTruthy(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: !!r,\n message: () => `Expected promise to resolve to a truthy value, but got ${r}`,\n };\n },\n\n async toResolveFalsy(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: !r,\n message: () => `Expected promise to resolve to a falsy value, but got ${r}`,\n };\n },\n\n async toResolveNull(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n const r = await received;\n return {\n pass: r === null,\n message: () => `Expected promise to resolve to a null value, but got ${r}`,\n };\n },\n\n async toResolveWithLength(received: Promise<unknown>, length: number) {\n const r = await received;\n return {\n pass: Array.isArray(r) && r.length === length,\n message: () => `Expected promise to resolve with an array with length ${length}, but got ${r}`,\n };\n },\n\n async toBeRejectedNotFound(received: Promise<unknown>) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n return expectErrorReason(err, ORMErrorReason.NOT_FOUND);\n }\n return {\n message: () => `expected NotFoundError, got no error`,\n pass: false,\n };\n },\n\n async toBeRejectedByPolicy(received: Promise<unknown>, expectedMessages?: string[]) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n if (expectedMessages && err instanceof ORMError && err.reason === ORMErrorReason.REJECTED_BY_POLICY) {\n const r = expectErrorMessages(expectedMessages, err.message || '');\n if (r) {\n return r;\n }\n }\n return expectErrorReason(err, ORMErrorReason.REJECTED_BY_POLICY);\n }\n return {\n message: () => `expected PolicyError, got no error`,\n pass: false,\n };\n },\n\n async toBeRejectedByValidation(received: Promise<unknown>, expectedMessages?: string[]) {\n if (!isPromise(received)) {\n return { message: () => 'a promise is expected', pass: false };\n }\n try {\n await received;\n } catch (err) {\n if (expectedMessages && err instanceof ORMError && err.reason === ORMErrorReason.INVALID_INPUT) {\n const r = expectErrorMessages(expectedMessages, err.message || '');\n if (r) {\n return r;\n }\n }\n return expectErrorReason(err, ORMErrorReason.INVALID_INPUT);\n }\n return {\n message: () => `expected InputValidationError, got no error`,\n pass: false,\n };\n },\n});\n"],"mappings":"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","names":["import_common_helpers","import_sdk","import_node_child_process","import_node_crypto","import_node_fs","import_node_path","import_ts_pattern","import_vitest","createTestProject","zmodelContent","name","workDir","tmp","dirSync","unsafeCleanup","fs","mkdirSync","path","join","nodeModules","readdirSync","__dirname","entry","startsWith","symlinkSync","zenstackPackages","pkg","writeFileSync","JSON","stringify","version","type","compilerOptions","module","target","moduleResolution","esModuleInterop","skipLibCheck","strict","include","import_node_fs","import_node_path","loadDocumentWithPlugins","filePath","pluginModelFiles","require","loadDocument","makePrelude","provider","dbUrl","match","with","exhaustive","replacePlaceholders","schemaText","url","replace","generateTsSchema","extraSourceFiles","withLiteSchema","workDir","createTestProject","zmodelPath","path","join","noPrelude","includes","fs","writeFileSync","result","loadDocumentWithPlugins","success","Error","errors","generator","TsSchemaGenerator","generate","model","outDir","lite","fileName","content","Object","entries","filePath","resolve","endsWith","mkdirSync","dirname","recursive","compileAndLoad","execSync","cwd","stdio","module","moduleLite","schema","schemaLite","generateTsSchemaFromFile","readFileSync","generateTsSchemaInPlace","schemaPath","loadSchema","additionalSchemas","tempDir","mkdtempSync","os","tmpdir","tempFile","name","r","expect","toSatisfy","map","e","toString","invariant","loadSchemaWithError","error","crypto","randomUUID","toBe","some","toLowerCase","test","getTestDbProvider","val","process","env","includes","Error","TEST_PG_CONFIG","host","port","parseInt","user","password","TEST_PG_URL","TEST_MYSQL_CONFIG","timezone","TEST_MYSQL_URL","createTestClient","schema","options","workDir","_schema","provider","dbName","getTestDbName","dbUrl","match","with","exhaustive","model","generated","generateTsSchema","extraSourceFiles","undefined","type","createTestProject","schemaFile","schemaContent","fs","readFileSync","replace","dataSourceExtensions","join","writeFileSync","path","invariant","plugins","rest","_options","debug","console","log","testLogger","dbFile","copyFileSync","copyFiles","state","expect","getState","currentTestPath","testPath","globPattern","destination","files","glob","sync","cwd","dirname","file","src","resolve","dest","basename","mkdirSync","recursive","usePrismaPush","r","loadDocumentWithPlugins","success","errors","prismaSchema","PrismaSchemaGenerator","prismaSchemaText","generate","execSync","stdio","prepareDatabase","dialect","createDialect","client","ZenStackClient","$pushSchema","plugin","$use","PostgresDialect","pool","Pool","database","MysqlDialect","createMysqlPool","SqliteDialect","SQLite","pgClient","PGClient","connect","query","end","mysqlPool","promise","createPolicyTestClient","PolicyPlugin","e","sql","parameters","testName","currentTestName","digest","createHash","update","toLowerCase","substring","slice","import_orm","import_vitest","isPromise","value","then","catch","expectErrorReason","err","errorReason","ORMError","reason","message","pass","expectErrorMessages","expectedMessages","m","toLowerCase","includes","undefined","expect","extend","toResolveTruthy","received","r","toResolveFalsy","toResolveNull","toResolveWithLength","length","Array","isArray","toBeRejectedNotFound","ORMErrorReason","NOT_FOUND","toBeRejectedByPolicy","REJECTED_BY_POLICY","toBeRejectedByValidation","INVALID_INPUT"]}
|
package/dist/index.d.cts
CHANGED
|
@@ -2,9 +2,9 @@ import { ClientOptions, ClientContract } from '@zenstackhq/orm';
|
|
|
2
2
|
import { SchemaDef } from '@zenstackhq/orm/schema';
|
|
3
3
|
import { LogEvent } from 'kysely';
|
|
4
4
|
import * as _zenstackhq_language_ast from '@zenstackhq/language/ast';
|
|
5
|
-
import { SchemaDef as SchemaDef$1 } from '@zenstackhq/schema';
|
|
5
|
+
import { DataSourceProviderType, SchemaDef as SchemaDef$1 } from '@zenstackhq/schema';
|
|
6
6
|
|
|
7
|
-
declare function getTestDbProvider(): "sqlite" | "postgresql";
|
|
7
|
+
declare function getTestDbProvider(): "sqlite" | "postgresql" | "mysql";
|
|
8
8
|
declare const TEST_PG_CONFIG: {
|
|
9
9
|
host: string;
|
|
10
10
|
port: number;
|
|
@@ -12,11 +12,19 @@ declare const TEST_PG_CONFIG: {
|
|
|
12
12
|
password: string;
|
|
13
13
|
};
|
|
14
14
|
declare const TEST_PG_URL: string;
|
|
15
|
+
declare const TEST_MYSQL_CONFIG: {
|
|
16
|
+
host: string;
|
|
17
|
+
port: number;
|
|
18
|
+
user: string;
|
|
19
|
+
password: string;
|
|
20
|
+
timezone: string;
|
|
21
|
+
};
|
|
22
|
+
declare const TEST_MYSQL_URL: string;
|
|
15
23
|
type ExtraTestClientOptions = {
|
|
16
24
|
/**
|
|
17
25
|
* Database provider
|
|
18
26
|
*/
|
|
19
|
-
provider?: 'sqlite' | 'postgresql';
|
|
27
|
+
provider?: 'sqlite' | 'postgresql' | 'mysql';
|
|
20
28
|
/**
|
|
21
29
|
* The main ZModel file. Only used when `usePrismaPush` is true and `schema` is an object.
|
|
22
30
|
*/
|
|
@@ -66,7 +74,7 @@ declare function testLogger(e: LogEvent): void;
|
|
|
66
74
|
|
|
67
75
|
declare function createTestProject(zmodelContent?: string): string;
|
|
68
76
|
|
|
69
|
-
declare function generateTsSchema(schemaText: string, provider?:
|
|
77
|
+
declare function generateTsSchema(schemaText: string, provider?: DataSourceProviderType, dbUrl?: string, extraSourceFiles?: Record<string, string>, withLiteSchema?: boolean): Promise<{
|
|
70
78
|
model: _zenstackhq_language_ast.Model;
|
|
71
79
|
workDir: string;
|
|
72
80
|
schema: SchemaDef$1;
|
|
@@ -86,4 +94,4 @@ declare function generateTsSchemaInPlace(schemaPath: string): Promise<{
|
|
|
86
94
|
declare function loadSchema(schema: string, additionalSchemas?: Record<string, string>): Promise<_zenstackhq_language_ast.Model>;
|
|
87
95
|
declare function loadSchemaWithError(schema: string, error: string | RegExp): Promise<void>;
|
|
88
96
|
|
|
89
|
-
export { type CreateTestClientOptions, TEST_PG_CONFIG, TEST_PG_URL, createPolicyTestClient, createTestClient, createTestProject, generateTsSchema, generateTsSchemaFromFile, generateTsSchemaInPlace, getTestDbProvider, loadSchema, loadSchemaWithError, testLogger };
|
|
97
|
+
export { type CreateTestClientOptions, TEST_MYSQL_CONFIG, TEST_MYSQL_URL, TEST_PG_CONFIG, TEST_PG_URL, createPolicyTestClient, createTestClient, createTestProject, generateTsSchema, generateTsSchemaFromFile, generateTsSchemaInPlace, getTestDbProvider, loadSchema, loadSchemaWithError, testLogger };
|
package/dist/index.d.ts
CHANGED
|
@@ -2,9 +2,9 @@ import { ClientOptions, ClientContract } from '@zenstackhq/orm';
|
|
|
2
2
|
import { SchemaDef } from '@zenstackhq/orm/schema';
|
|
3
3
|
import { LogEvent } from 'kysely';
|
|
4
4
|
import * as _zenstackhq_language_ast from '@zenstackhq/language/ast';
|
|
5
|
-
import { SchemaDef as SchemaDef$1 } from '@zenstackhq/schema';
|
|
5
|
+
import { DataSourceProviderType, SchemaDef as SchemaDef$1 } from '@zenstackhq/schema';
|
|
6
6
|
|
|
7
|
-
declare function getTestDbProvider(): "sqlite" | "postgresql";
|
|
7
|
+
declare function getTestDbProvider(): "sqlite" | "postgresql" | "mysql";
|
|
8
8
|
declare const TEST_PG_CONFIG: {
|
|
9
9
|
host: string;
|
|
10
10
|
port: number;
|
|
@@ -12,11 +12,19 @@ declare const TEST_PG_CONFIG: {
|
|
|
12
12
|
password: string;
|
|
13
13
|
};
|
|
14
14
|
declare const TEST_PG_URL: string;
|
|
15
|
+
declare const TEST_MYSQL_CONFIG: {
|
|
16
|
+
host: string;
|
|
17
|
+
port: number;
|
|
18
|
+
user: string;
|
|
19
|
+
password: string;
|
|
20
|
+
timezone: string;
|
|
21
|
+
};
|
|
22
|
+
declare const TEST_MYSQL_URL: string;
|
|
15
23
|
type ExtraTestClientOptions = {
|
|
16
24
|
/**
|
|
17
25
|
* Database provider
|
|
18
26
|
*/
|
|
19
|
-
provider?: 'sqlite' | 'postgresql';
|
|
27
|
+
provider?: 'sqlite' | 'postgresql' | 'mysql';
|
|
20
28
|
/**
|
|
21
29
|
* The main ZModel file. Only used when `usePrismaPush` is true and `schema` is an object.
|
|
22
30
|
*/
|
|
@@ -66,7 +74,7 @@ declare function testLogger(e: LogEvent): void;
|
|
|
66
74
|
|
|
67
75
|
declare function createTestProject(zmodelContent?: string): string;
|
|
68
76
|
|
|
69
|
-
declare function generateTsSchema(schemaText: string, provider?:
|
|
77
|
+
declare function generateTsSchema(schemaText: string, provider?: DataSourceProviderType, dbUrl?: string, extraSourceFiles?: Record<string, string>, withLiteSchema?: boolean): Promise<{
|
|
70
78
|
model: _zenstackhq_language_ast.Model;
|
|
71
79
|
workDir: string;
|
|
72
80
|
schema: SchemaDef$1;
|
|
@@ -86,4 +94,4 @@ declare function generateTsSchemaInPlace(schemaPath: string): Promise<{
|
|
|
86
94
|
declare function loadSchema(schema: string, additionalSchemas?: Record<string, string>): Promise<_zenstackhq_language_ast.Model>;
|
|
87
95
|
declare function loadSchemaWithError(schema: string, error: string | RegExp): Promise<void>;
|
|
88
96
|
|
|
89
|
-
export { type CreateTestClientOptions, TEST_PG_CONFIG, TEST_PG_URL, createPolicyTestClient, createTestClient, createTestProject, generateTsSchema, generateTsSchemaFromFile, generateTsSchemaInPlace, getTestDbProvider, loadSchema, loadSchemaWithError, testLogger };
|
|
97
|
+
export { type CreateTestClientOptions, TEST_MYSQL_CONFIG, TEST_MYSQL_URL, TEST_PG_CONFIG, TEST_PG_URL, createPolicyTestClient, createTestClient, createTestProject, generateTsSchema, generateTsSchemaFromFile, generateTsSchemaInPlace, getTestDbProvider, loadSchema, loadSchemaWithError, testLogger };
|
package/dist/index.js
CHANGED
|
@@ -14,12 +14,14 @@ import { PolicyPlugin } from "@zenstackhq/plugin-policy";
|
|
|
14
14
|
import { PrismaSchemaGenerator } from "@zenstackhq/sdk";
|
|
15
15
|
import SQLite from "better-sqlite3";
|
|
16
16
|
import { glob } from "glob";
|
|
17
|
-
import { PostgresDialect, SqliteDialect } from "kysely";
|
|
17
|
+
import { MysqlDialect, PostgresDialect, SqliteDialect } from "kysely";
|
|
18
|
+
import { createPool as createMysqlPool } from "mysql2";
|
|
18
19
|
import { execSync as execSync2 } from "child_process";
|
|
19
20
|
import { createHash } from "crypto";
|
|
20
21
|
import fs3 from "fs";
|
|
21
22
|
import path3 from "path";
|
|
22
23
|
import { Client as PGClient, Pool } from "pg";
|
|
24
|
+
import { match as match2 } from "ts-pattern";
|
|
23
25
|
import { expect as expect2 } from "vitest";
|
|
24
26
|
|
|
25
27
|
// src/project.ts
|
|
@@ -109,12 +111,19 @@ datasource db {
|
|
|
109
111
|
provider = 'postgresql'
|
|
110
112
|
url = '${dbUrl ?? "postgres://postgres:postgres@localhost:5432/db"}'
|
|
111
113
|
}
|
|
114
|
+
`;
|
|
115
|
+
}).with("mysql", () => {
|
|
116
|
+
return `
|
|
117
|
+
datasource db {
|
|
118
|
+
provider = 'mysql'
|
|
119
|
+
url = '${dbUrl ?? "mysql://root:mysql@localhost:3306/db"}'
|
|
120
|
+
}
|
|
112
121
|
`;
|
|
113
122
|
}).exhaustive();
|
|
114
123
|
}
|
|
115
124
|
__name(makePrelude, "makePrelude");
|
|
116
125
|
function replacePlaceholders(schemaText, provider, dbUrl) {
|
|
117
|
-
const url = dbUrl ?? (provider === "sqlite" ? "file:./test.db" : "postgres://postgres:postgres@localhost:5432/db");
|
|
126
|
+
const url = dbUrl ?? (provider === "sqlite" ? "file:./test.db" : provider === "mysql" ? "mysql://root:mysql@localhost:3306/db" : "postgres://postgres:postgres@localhost:5432/db");
|
|
118
127
|
return schemaText.replace(/\$DB_URL/g, url).replace(/\$PROVIDER/g, provider);
|
|
119
128
|
}
|
|
120
129
|
__name(replacePlaceholders, "replacePlaceholders");
|
|
@@ -234,7 +243,8 @@ function getTestDbProvider() {
|
|
|
234
243
|
const val = process.env["TEST_DB_PROVIDER"] ?? "sqlite";
|
|
235
244
|
if (![
|
|
236
245
|
"sqlite",
|
|
237
|
-
"postgresql"
|
|
246
|
+
"postgresql",
|
|
247
|
+
"mysql"
|
|
238
248
|
].includes(val)) {
|
|
239
249
|
throw new Error(`Invalid TEST_DB_PROVIDER value: ${val}`);
|
|
240
250
|
}
|
|
@@ -248,12 +258,20 @@ var TEST_PG_CONFIG = {
|
|
|
248
258
|
password: process.env["TEST_PG_PASSWORD"] ?? "postgres"
|
|
249
259
|
};
|
|
250
260
|
var TEST_PG_URL = `postgres://${TEST_PG_CONFIG.user}:${TEST_PG_CONFIG.password}@${TEST_PG_CONFIG.host}:${TEST_PG_CONFIG.port}`;
|
|
261
|
+
var TEST_MYSQL_CONFIG = {
|
|
262
|
+
host: process.env["TEST_MYSQL_HOST"] ?? "localhost",
|
|
263
|
+
port: process.env["TEST_MYSQL_PORT"] ? parseInt(process.env["TEST_MYSQL_PORT"]) : 3306,
|
|
264
|
+
user: process.env["TEST_MYSQL_USER"] ?? "root",
|
|
265
|
+
password: process.env["TEST_MYSQL_PASSWORD"] ?? "mysql",
|
|
266
|
+
timezone: "Z"
|
|
267
|
+
};
|
|
268
|
+
var TEST_MYSQL_URL = `mysql://${TEST_MYSQL_CONFIG.user}:${TEST_MYSQL_CONFIG.password}@${TEST_MYSQL_CONFIG.host}:${TEST_MYSQL_CONFIG.port}`;
|
|
251
269
|
async function createTestClient(schema, options) {
|
|
252
270
|
let workDir = options?.workDir;
|
|
253
271
|
let _schema;
|
|
254
272
|
const provider = options?.provider ?? getTestDbProvider() ?? "sqlite";
|
|
255
273
|
const dbName = options?.dbName ?? getTestDbName(provider);
|
|
256
|
-
const dbUrl = provider
|
|
274
|
+
const dbUrl = match2(provider).with("sqlite", () => `file:${dbName}`).with("mysql", () => `${TEST_MYSQL_URL}/${dbName}`).with("postgresql", () => `${TEST_PG_URL}/${dbName}`).exhaustive();
|
|
257
275
|
let model;
|
|
258
276
|
if (typeof schema === "string") {
|
|
259
277
|
const generated = await generateTsSchema(schema, provider, dbUrl, options?.extraSourceFiles, void 0);
|
|
@@ -294,7 +312,7 @@ async function createTestClient(schema, options) {
|
|
|
294
312
|
if (options?.debug) {
|
|
295
313
|
console.log(`Work directory: ${workDir}`);
|
|
296
314
|
console.log(`Database name: ${dbName}`);
|
|
297
|
-
_options.log
|
|
315
|
+
_options.log ??= testLogger;
|
|
298
316
|
}
|
|
299
317
|
if (options?.dbFile) {
|
|
300
318
|
if (provider !== "sqlite") {
|
|
@@ -340,28 +358,10 @@ async function createTestClient(schema, options) {
|
|
|
340
358
|
stdio: options.debug ? "inherit" : "ignore"
|
|
341
359
|
});
|
|
342
360
|
} else {
|
|
343
|
-
|
|
344
|
-
invariant2(dbName, "dbName is required");
|
|
345
|
-
const pgClient = new PGClient(TEST_PG_CONFIG);
|
|
346
|
-
await pgClient.connect();
|
|
347
|
-
await pgClient.query(`DROP DATABASE IF EXISTS "${dbName}"`);
|
|
348
|
-
await pgClient.query(`CREATE DATABASE "${dbName}"`);
|
|
349
|
-
await pgClient.end();
|
|
350
|
-
}
|
|
361
|
+
await prepareDatabase(provider, dbName);
|
|
351
362
|
}
|
|
352
363
|
}
|
|
353
|
-
|
|
354
|
-
_options.dialect = new PostgresDialect({
|
|
355
|
-
pool: new Pool({
|
|
356
|
-
...TEST_PG_CONFIG,
|
|
357
|
-
database: dbName
|
|
358
|
-
})
|
|
359
|
-
});
|
|
360
|
-
} else {
|
|
361
|
-
_options.dialect = new SqliteDialect({
|
|
362
|
-
database: new SQLite(path3.join(workDir, dbName))
|
|
363
|
-
});
|
|
364
|
-
}
|
|
364
|
+
_options.dialect = createDialect(provider, dbName, workDir);
|
|
365
365
|
let client = new ZenStackClient(_schema, _options);
|
|
366
366
|
if (!options?.usePrismaPush && !options?.dbFile) {
|
|
367
367
|
await client.$pushSchema();
|
|
@@ -374,6 +374,39 @@ async function createTestClient(schema, options) {
|
|
|
374
374
|
return client;
|
|
375
375
|
}
|
|
376
376
|
__name(createTestClient, "createTestClient");
|
|
377
|
+
function createDialect(provider, dbName, workDir) {
|
|
378
|
+
return match2(provider).with("postgresql", () => new PostgresDialect({
|
|
379
|
+
pool: new Pool({
|
|
380
|
+
...TEST_PG_CONFIG,
|
|
381
|
+
database: dbName
|
|
382
|
+
})
|
|
383
|
+
})).with("mysql", () => new MysqlDialect({
|
|
384
|
+
pool: createMysqlPool({
|
|
385
|
+
...TEST_MYSQL_CONFIG,
|
|
386
|
+
database: dbName
|
|
387
|
+
})
|
|
388
|
+
})).with("sqlite", () => new SqliteDialect({
|
|
389
|
+
database: new SQLite(path3.join(workDir, dbName))
|
|
390
|
+
})).exhaustive();
|
|
391
|
+
}
|
|
392
|
+
__name(createDialect, "createDialect");
|
|
393
|
+
async function prepareDatabase(provider, dbName) {
|
|
394
|
+
if (provider === "postgresql") {
|
|
395
|
+
invariant2(dbName, "dbName is required");
|
|
396
|
+
const pgClient = new PGClient(TEST_PG_CONFIG);
|
|
397
|
+
await pgClient.connect();
|
|
398
|
+
await pgClient.query(`DROP DATABASE IF EXISTS "${dbName}"`);
|
|
399
|
+
await pgClient.query(`CREATE DATABASE "${dbName}"`);
|
|
400
|
+
await pgClient.end();
|
|
401
|
+
} else if (provider === "mysql") {
|
|
402
|
+
invariant2(dbName, "dbName is required");
|
|
403
|
+
const mysqlPool = createMysqlPool(TEST_MYSQL_CONFIG);
|
|
404
|
+
await mysqlPool.promise().query(`DROP DATABASE IF EXISTS \`${dbName}\``);
|
|
405
|
+
await mysqlPool.promise().query(`CREATE DATABASE \`${dbName}\``);
|
|
406
|
+
await mysqlPool.promise().end();
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
__name(prepareDatabase, "prepareDatabase");
|
|
377
410
|
async function createPolicyTestClient(schema, options) {
|
|
378
411
|
return createTestClient(schema, {
|
|
379
412
|
...options,
|
|
@@ -544,6 +577,8 @@ expect3.extend({
|
|
|
544
577
|
}
|
|
545
578
|
});
|
|
546
579
|
export {
|
|
580
|
+
TEST_MYSQL_CONFIG,
|
|
581
|
+
TEST_MYSQL_URL,
|
|
547
582
|
TEST_PG_CONFIG,
|
|
548
583
|
TEST_PG_URL,
|
|
549
584
|
createPolicyTestClient,
|