@malloy-publisher/server 0.0.165 → 0.0.168

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.
Files changed (41) hide show
  1. package/.eslintrc.json +9 -1
  2. package/dist/app/api-doc.yaml +143 -1
  3. package/dist/app/assets/HomePage-D2tUw_9U.js +1 -0
  4. package/dist/app/assets/{MainPage-DAyUfYba.js → MainPage-DBQW76L7.js} +2 -2
  5. package/dist/app/assets/{ModelPage-CrMryV1s.js → ModelPage-BnfOKuhQ.js} +1 -1
  6. package/dist/app/assets/PackagePage-zPhE-rDg.js +1 -0
  7. package/dist/app/assets/ProjectPage-BpSTvuW6.js +1 -0
  8. package/dist/app/assets/RouteError-Cp9-yCK5.js +1 -0
  9. package/dist/app/assets/{WorkbookPage-DZEVYGW3.js → WorkbookPage-FD_gmxeE.js} +1 -1
  10. package/dist/app/assets/{index-BvVmB5sv.js → index-D5QBYuLK.js} +150 -150
  11. package/dist/app/assets/{index-CsC07BYd.js → index-DNCvL_5f.js} +1 -1
  12. package/dist/app/assets/{index-DWhjtyBB.js → index-x9S1fsYn.js} +1 -1
  13. package/dist/app/assets/{index.umd-DvM-lTQa.js → index.umd-CTYdFEHH.js} +1 -1
  14. package/dist/app/index.html +1 -1
  15. package/dist/instrumentation.js +85955 -88560
  16. package/dist/server.js +197441 -106276
  17. package/package.json +2 -1
  18. package/src/controller/compile.controller.ts +35 -0
  19. package/src/controller/connection.controller.ts +22 -2
  20. package/src/controller/model.controller.ts +20 -9
  21. package/src/health.ts +8 -0
  22. package/src/instrumentation.ts +123 -34
  23. package/src/server.ts +49 -3
  24. package/src/service/connection.spec.ts +1331 -0
  25. package/src/service/connection.ts +407 -29
  26. package/src/service/db_utils.ts +104 -45
  27. package/src/service/gcs_s3_utils.ts +115 -40
  28. package/src/service/model.ts +5 -5
  29. package/src/service/project.ts +140 -4
  30. package/src/service/project_compile.spec.ts +197 -0
  31. package/src/service/project_store.ts +49 -21
  32. package/src/storage/StorageManager.ts +4 -3
  33. package/src/storage/duckdb/schema.ts +6 -5
  34. package/tests/harness/e2e.ts +4 -0
  35. package/tests/harness/mcp_test_setup.ts +172 -28
  36. package/tests/unit/duckdb/attached_databases.test.ts +61 -3
  37. package/tests/unit/ducklake/ducklake.test.ts +950 -0
  38. package/dist/app/assets/HomePage-QekMXs8r.js +0 -1
  39. package/dist/app/assets/PackagePage-DDaABD2A.js +0 -1
  40. package/dist/app/assets/ProjectPage-FAYUFGhL.js +0 -1
  41. package/dist/app/assets/RouteError-BKYctANX.js +0 -1
@@ -0,0 +1,1331 @@
1
+ import { afterEach, beforeEach, describe, expect, it } from "bun:test";
2
+ import fs from "fs/promises";
3
+ import path from "path";
4
+ import sinon from "sinon";
5
+ import { DuckDBConnection } from "@malloydata/db-duckdb";
6
+ import { createProjectConnections, testConnectionConfig } from "./connection";
7
+ import { components } from "../api";
8
+
9
+ type ApiConnection = components["schemas"]["Connection"];
10
+ type AttachedDatabase = components["schemas"]["AttachedDatabase"];
11
+
12
+ const hasPostgresCredentials = () =>
13
+ !!(
14
+ process.env.POSTGRES_TEST_HOST &&
15
+ process.env.POSTGRES_TEST_USER &&
16
+ process.env.POSTGRES_TEST_PASSWORD
17
+ );
18
+
19
+ const hasBigQueryCredentials = () =>
20
+ !!(
21
+ process.env.GOOGLE_APPLICATION_CREDENTIALS &&
22
+ process.env.BIGQUERY_TEST_PROJECT_ID
23
+ );
24
+
25
+ const hasSnowflakeCredentials = () =>
26
+ !!(
27
+ process.env.SNOWFLAKE_TEST_ACCOUNT &&
28
+ process.env.SNOWFLAKE_TEST_USER &&
29
+ process.env.SNOWFLAKE_TEST_PASSWORD &&
30
+ process.env.SNOWFLAKE_TEST_WAREHOUSE
31
+ );
32
+
33
+ const hasS3Credentials = () =>
34
+ !!(
35
+ process.env.S3_TEST_ACCESS_KEY_ID && process.env.S3_TEST_SECRET_ACCESS_KEY
36
+ );
37
+
38
+ const hasGCSCredentials = () =>
39
+ !!(process.env.GCS_TEST_KEY_ID && process.env.GCS_TEST_SECRET);
40
+
41
+ const readBigQueryServiceAccountJson = async (): Promise<string> =>
42
+ fs.readFile(process.env.GOOGLE_APPLICATION_CREDENTIALS!, "utf-8");
43
+
44
+ describe("connection integration tests", () => {
45
+ const testProjectPath = path.join(process.cwd(), "test-project-connections");
46
+ let createdConnections: DuckDBConnection[] = [];
47
+
48
+ beforeEach(async () => {
49
+ await fs.mkdir(testProjectPath, { recursive: true });
50
+ });
51
+
52
+ afterEach(async () => {
53
+ sinon.restore();
54
+
55
+ for (const conn of createdConnections) {
56
+ try {
57
+ await conn.close();
58
+ } catch (error) {
59
+ console.warn("Error closing connection:", error);
60
+ }
61
+ }
62
+ createdConnections = [];
63
+
64
+ const maxRetries = 5;
65
+ const delay = 100;
66
+ let lastError: unknown;
67
+
68
+ for (let attempt = 0; attempt < maxRetries; attempt++) {
69
+ try {
70
+ await fs.rm(testProjectPath, { recursive: true, force: true });
71
+ return;
72
+ } catch (error) {
73
+ lastError = error;
74
+ const errnoError = error as NodeJS.ErrnoException;
75
+ if (errnoError.code !== "EBUSY") throw error;
76
+ if (attempt < maxRetries - 1) {
77
+ await new Promise((resolve) =>
78
+ setTimeout(resolve, delay * Math.pow(2, attempt)),
79
+ );
80
+ }
81
+ }
82
+ }
83
+
84
+ if ((lastError as NodeJS.ErrnoException).code !== "EBUSY") {
85
+ throw lastError;
86
+ }
87
+ });
88
+
89
+ describe("createProjectConnections", () => {
90
+ describe("DuckDB with PostgreSQL attachment", () => {
91
+ it(
92
+ "should create DuckDB connection with attached PostgreSQL database",
93
+ async () => {
94
+ if (!hasPostgresCredentials()) {
95
+ console.log(
96
+ "Skipping: PostgreSQL credentials not configured",
97
+ );
98
+ return;
99
+ }
100
+
101
+ const postgresAttachment: AttachedDatabase = {
102
+ name: "pg_test",
103
+ type: "postgres",
104
+ postgresConnection: {
105
+ host: process.env.POSTGRES_TEST_HOST,
106
+ port: parseInt(process.env.POSTGRES_TEST_PORT || "5432"),
107
+ userName: process.env.POSTGRES_TEST_USER!,
108
+ password: process.env.POSTGRES_TEST_PASSWORD!,
109
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
110
+ },
111
+ };
112
+
113
+ const duckdbConnection: ApiConnection = {
114
+ name: "duckdb_with_postgres",
115
+ type: "duckdb",
116
+ duckdbConnection: { attachedDatabases: [postgresAttachment] },
117
+ };
118
+
119
+ const { malloyConnections, apiConnections } =
120
+ await createProjectConnections(
121
+ [duckdbConnection],
122
+ testProjectPath,
123
+ );
124
+
125
+ expect(malloyConnections.size).toBe(1);
126
+ expect(apiConnections.length).toBe(1);
127
+
128
+ const connection = malloyConnections.get(
129
+ "duckdb_with_postgres",
130
+ ) as DuckDBConnection;
131
+ expect(connection).toBeDefined();
132
+ createdConnections.push(connection);
133
+
134
+ const databases = await connection.runSQL("SHOW DATABASES");
135
+ const dbNames = databases.rows.map(
136
+ (row) => Object.values(row)[0],
137
+ );
138
+ expect(dbNames).toContain("pg_test");
139
+
140
+ const result = await connection.runSQL(
141
+ "SELECT 1 as test_value FROM pg_test.information_schema.tables LIMIT 1",
142
+ );
143
+ expect(result.rows).toBeDefined();
144
+ },
145
+ { timeout: 30000 },
146
+ );
147
+
148
+ it(
149
+ "should list schemas from attached PostgreSQL database",
150
+ async () => {
151
+ if (!hasPostgresCredentials()) {
152
+ console.log(
153
+ "Skipping: PostgreSQL credentials not configured",
154
+ );
155
+ return;
156
+ }
157
+
158
+ const duckdbConnection: ApiConnection = {
159
+ name: "duckdb_pg_schemas",
160
+ type: "duckdb",
161
+ duckdbConnection: {
162
+ attachedDatabases: [
163
+ {
164
+ name: "pg_schemas",
165
+ type: "postgres",
166
+ postgresConnection: {
167
+ host: process.env.POSTGRES_TEST_HOST,
168
+ port: parseInt(
169
+ process.env.POSTGRES_TEST_PORT || "5432",
170
+ ),
171
+ userName: process.env.POSTGRES_TEST_USER!,
172
+ password: process.env.POSTGRES_TEST_PASSWORD!,
173
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
174
+ },
175
+ },
176
+ ],
177
+ },
178
+ };
179
+
180
+ const { malloyConnections } = await createProjectConnections(
181
+ [duckdbConnection],
182
+ testProjectPath,
183
+ );
184
+
185
+ const connection = malloyConnections.get(
186
+ "duckdb_pg_schemas",
187
+ ) as DuckDBConnection;
188
+ createdConnections.push(connection);
189
+
190
+ // listSchemas equivalent
191
+ const result = await connection.runSQL(
192
+ "SELECT schema_name FROM pg_schemas.information_schema.schemata ORDER BY schema_name",
193
+ );
194
+ expect(result.rows.length).toBeGreaterThan(0);
195
+ const schemaNames = result.rows.map(
196
+ (row) => Object.values(row)[0] as string,
197
+ );
198
+ expect(schemaNames).toContain("public");
199
+ },
200
+ { timeout: 30000 },
201
+ );
202
+
203
+ it(
204
+ "should list tables from attached PostgreSQL database",
205
+ async () => {
206
+ if (!hasPostgresCredentials()) {
207
+ console.log(
208
+ "Skipping: PostgreSQL credentials not configured",
209
+ );
210
+ return;
211
+ }
212
+
213
+ const duckdbConnection: ApiConnection = {
214
+ name: "duckdb_pg_tables",
215
+ type: "duckdb",
216
+ duckdbConnection: {
217
+ attachedDatabases: [
218
+ {
219
+ name: "pg_tables_db",
220
+ type: "postgres",
221
+ postgresConnection: {
222
+ host: process.env.POSTGRES_TEST_HOST,
223
+ port: parseInt(
224
+ process.env.POSTGRES_TEST_PORT || "5432",
225
+ ),
226
+ userName: process.env.POSTGRES_TEST_USER!,
227
+ password: process.env.POSTGRES_TEST_PASSWORD!,
228
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
229
+ },
230
+ },
231
+ ],
232
+ },
233
+ };
234
+
235
+ const { malloyConnections } = await createProjectConnections(
236
+ [duckdbConnection],
237
+ testProjectPath,
238
+ );
239
+
240
+ const connection = malloyConnections.get(
241
+ "duckdb_pg_tables",
242
+ ) as DuckDBConnection;
243
+ createdConnections.push(connection);
244
+
245
+ // listTables equivalent
246
+ const result = await connection.runSQL(`
247
+ SELECT table_schema, table_name, table_type
248
+ FROM pg_tables_db.information_schema.tables
249
+ WHERE table_schema NOT IN ('pg_catalog', 'information_schema')
250
+ ORDER BY table_schema, table_name
251
+ `);
252
+ expect(result.rows).toBeDefined();
253
+ },
254
+ { timeout: 30000 },
255
+ );
256
+
257
+ it(
258
+ "should get table columns from attached PostgreSQL database",
259
+ async () => {
260
+ if (!hasPostgresCredentials()) {
261
+ console.log(
262
+ "Skipping: PostgreSQL credentials not configured",
263
+ );
264
+ return;
265
+ }
266
+
267
+ const duckdbConnection: ApiConnection = {
268
+ name: "duckdb_pg_cols",
269
+ type: "duckdb",
270
+ duckdbConnection: {
271
+ attachedDatabases: [
272
+ {
273
+ name: "pg_cols",
274
+ type: "postgres",
275
+ postgresConnection: {
276
+ host: process.env.POSTGRES_TEST_HOST,
277
+ port: parseInt(
278
+ process.env.POSTGRES_TEST_PORT || "5432",
279
+ ),
280
+ userName: process.env.POSTGRES_TEST_USER!,
281
+ password: process.env.POSTGRES_TEST_PASSWORD!,
282
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
283
+ },
284
+ },
285
+ ],
286
+ },
287
+ };
288
+
289
+ const { malloyConnections } = await createProjectConnections(
290
+ [duckdbConnection],
291
+ testProjectPath,
292
+ );
293
+
294
+ const connection = malloyConnections.get(
295
+ "duckdb_pg_cols",
296
+ ) as DuckDBConnection;
297
+ createdConnections.push(connection);
298
+
299
+ const result = await connection.runSQL(`
300
+ SELECT column_name, data_type, is_nullable
301
+ FROM pg_cols.information_schema.columns
302
+ WHERE table_schema = 'information_schema'
303
+ AND table_name = 'tables'
304
+ ORDER BY ordinal_position
305
+ `);
306
+ expect(result.rows.length).toBeGreaterThan(0);
307
+ const columnNames = result.rows.map(
308
+ (row) => (row as Record<string, unknown>)["column_name"],
309
+ );
310
+ expect(columnNames).toContain("table_name");
311
+ expect(columnNames).toContain("table_schema");
312
+ },
313
+ { timeout: 30000 },
314
+ );
315
+
316
+ it(
317
+ "should handle PostgreSQL connection string format",
318
+ async () => {
319
+ if (!hasPostgresCredentials()) {
320
+ console.log(
321
+ "Skipping: PostgreSQL credentials not configured",
322
+ );
323
+ return;
324
+ }
325
+
326
+ const connectionString = `host=${process.env.POSTGRES_TEST_HOST} port=${process.env.POSTGRES_TEST_PORT || "5432"} dbname=${process.env.POSTGRES_TEST_DATABASE} user=${process.env.POSTGRES_TEST_USER} password=${process.env.POSTGRES_TEST_PASSWORD}`;
327
+
328
+ const duckdbConnection: ApiConnection = {
329
+ name: "duckdb_pg_string",
330
+ type: "duckdb",
331
+ duckdbConnection: {
332
+ attachedDatabases: [
333
+ {
334
+ name: "pg_conn_string",
335
+ type: "postgres",
336
+ postgresConnection: { connectionString },
337
+ },
338
+ ],
339
+ },
340
+ };
341
+
342
+ const { malloyConnections } = await createProjectConnections(
343
+ [duckdbConnection],
344
+ testProjectPath,
345
+ );
346
+
347
+ const connection = malloyConnections.get(
348
+ "duckdb_pg_string",
349
+ ) as DuckDBConnection;
350
+ createdConnections.push(connection);
351
+
352
+ const databases = await connection.runSQL("SHOW DATABASES");
353
+ const dbNames = databases.rows.map(
354
+ (row) => Object.values(row)[0],
355
+ );
356
+ expect(dbNames).toContain("pg_conn_string");
357
+ },
358
+ { timeout: 30000 },
359
+ );
360
+ });
361
+
362
+ describe("DuckDB with BigQuery attachment", () => {
363
+ it(
364
+ "should create DuckDB connection with attached BigQuery database",
365
+ async () => {
366
+ if (!hasBigQueryCredentials()) {
367
+ console.log("Skipping: BigQuery credentials not configured");
368
+ return;
369
+ }
370
+
371
+ const serviceAccountJson =
372
+ await readBigQueryServiceAccountJson();
373
+
374
+ const duckdbConnection: ApiConnection = {
375
+ name: "duckdb_with_bigquery",
376
+ type: "duckdb",
377
+ duckdbConnection: {
378
+ attachedDatabases: [
379
+ {
380
+ name: "bq_test",
381
+ type: "bigquery",
382
+ bigqueryConnection: {
383
+ defaultProjectId:
384
+ process.env.BIGQUERY_TEST_PROJECT_ID!,
385
+ serviceAccountKeyJson: serviceAccountJson,
386
+ },
387
+ },
388
+ ],
389
+ },
390
+ };
391
+
392
+ const { malloyConnections } = await createProjectConnections(
393
+ [duckdbConnection],
394
+ testProjectPath,
395
+ );
396
+
397
+ const connection = malloyConnections.get(
398
+ "duckdb_with_bigquery",
399
+ ) as DuckDBConnection;
400
+ expect(connection).toBeDefined();
401
+ createdConnections.push(connection);
402
+
403
+ const databases = await connection.runSQL("SHOW DATABASES");
404
+ const dbNames = databases.rows.map(
405
+ (row) => Object.values(row)[0],
406
+ );
407
+ expect(dbNames).toContain("bq_test");
408
+ },
409
+ { timeout: 60000 },
410
+ );
411
+
412
+ it(
413
+ "should list datasets (schemas) from attached BigQuery database",
414
+ async () => {
415
+ if (
416
+ !hasBigQueryCredentials() ||
417
+ !process.env.BIGQUERY_TEST_DATASET
418
+ ) {
419
+ console.log(
420
+ "Skipping: GOOGLE_APPLICATION_CREDENTIALS, BIGQUERY_TEST_PROJECT_ID, or BIGQUERY_TEST_DATASET not configured",
421
+ );
422
+ return;
423
+ }
424
+
425
+ const serviceAccountJson =
426
+ await readBigQueryServiceAccountJson();
427
+
428
+ const duckdbConnection: ApiConnection = {
429
+ name: "duckdb_bq_schemas",
430
+ type: "duckdb",
431
+ duckdbConnection: {
432
+ attachedDatabases: [
433
+ {
434
+ name: "bq_schemas",
435
+ type: "bigquery",
436
+ bigqueryConnection: {
437
+ defaultProjectId:
438
+ process.env.BIGQUERY_TEST_PROJECT_ID!,
439
+ serviceAccountKeyJson: serviceAccountJson,
440
+ },
441
+ },
442
+ ],
443
+ },
444
+ };
445
+
446
+ const { malloyConnections } = await createProjectConnections(
447
+ [duckdbConnection],
448
+ testProjectPath,
449
+ );
450
+
451
+ const connection = malloyConnections.get(
452
+ "duckdb_bq_schemas",
453
+ ) as DuckDBConnection;
454
+ createdConnections.push(connection);
455
+
456
+ const result = await connection.runSQL(
457
+ "SELECT schema_name FROM bq_schemas.information_schema.schemata",
458
+ );
459
+ expect(result.rows.length).toBeGreaterThan(0);
460
+ const datasetNames = result.rows.map(
461
+ (row) => Object.values(row)[0] as string,
462
+ );
463
+ expect(datasetNames).toContain(
464
+ process.env.BIGQUERY_TEST_DATASET!,
465
+ );
466
+ },
467
+ { timeout: 60000 },
468
+ );
469
+
470
+ it(
471
+ "should list tables from attached BigQuery dataset",
472
+ async () => {
473
+ if (
474
+ !hasBigQueryCredentials() ||
475
+ !process.env.BIGQUERY_TEST_DATASET
476
+ ) {
477
+ console.log(
478
+ "Skipping: GOOGLE_APPLICATION_CREDENTIALS, BIGQUERY_TEST_PROJECT_ID, or BIGQUERY_TEST_DATASET not configured",
479
+ );
480
+ return;
481
+ }
482
+
483
+ const serviceAccountJson =
484
+ await readBigQueryServiceAccountJson();
485
+
486
+ const duckdbConnection: ApiConnection = {
487
+ name: "duckdb_bq_tables",
488
+ type: "duckdb",
489
+ duckdbConnection: {
490
+ attachedDatabases: [
491
+ {
492
+ name: "bq_tables",
493
+ type: "bigquery",
494
+ bigqueryConnection: {
495
+ defaultProjectId:
496
+ process.env.BIGQUERY_TEST_PROJECT_ID!,
497
+ serviceAccountKeyJson: serviceAccountJson,
498
+ },
499
+ },
500
+ ],
501
+ },
502
+ };
503
+
504
+ const { malloyConnections } = await createProjectConnections(
505
+ [duckdbConnection],
506
+ testProjectPath,
507
+ );
508
+
509
+ const connection = malloyConnections.get(
510
+ "duckdb_bq_tables",
511
+ ) as DuckDBConnection;
512
+ createdConnections.push(connection);
513
+
514
+ const result = await connection.runSQL(`
515
+ SELECT table_name, table_type
516
+ FROM bq_tables.${process.env.BIGQUERY_TEST_DATASET!}.INFORMATION_SCHEMA.TABLES
517
+ ORDER BY table_name
518
+ `);
519
+ expect(result.rows.length).toBeGreaterThan(0);
520
+ },
521
+ { timeout: 60000 },
522
+ );
523
+
524
+ it(
525
+ "should get table columns from attached BigQuery table",
526
+ async () => {
527
+ if (
528
+ !hasBigQueryCredentials() ||
529
+ !process.env.BIGQUERY_TEST_TABLE
530
+ ) {
531
+ console.log(
532
+ "Skipping: GOOGLE_APPLICATION_CREDENTIALS, BIGQUERY_TEST_PROJECT_ID, or BIGQUERY_TEST_TABLE not configured",
533
+ );
534
+ return;
535
+ }
536
+
537
+ const serviceAccountJson =
538
+ await readBigQueryServiceAccountJson();
539
+ const [dataset, table] =
540
+ process.env.BIGQUERY_TEST_TABLE!.split(".");
541
+
542
+ const duckdbConnection: ApiConnection = {
543
+ name: "duckdb_bq_cols",
544
+ type: "duckdb",
545
+ duckdbConnection: {
546
+ attachedDatabases: [
547
+ {
548
+ name: "bq_cols",
549
+ type: "bigquery",
550
+ bigqueryConnection: {
551
+ defaultProjectId:
552
+ process.env.BIGQUERY_TEST_PROJECT_ID!,
553
+ serviceAccountKeyJson: serviceAccountJson,
554
+ },
555
+ },
556
+ ],
557
+ },
558
+ };
559
+
560
+ const { malloyConnections } = await createProjectConnections(
561
+ [duckdbConnection],
562
+ testProjectPath,
563
+ );
564
+
565
+ const connection = malloyConnections.get(
566
+ "duckdb_bq_cols",
567
+ ) as DuckDBConnection;
568
+ createdConnections.push(connection);
569
+
570
+ // getTable equivalent — fetch column metadata
571
+ const result = await connection.runSQL(`
572
+ SELECT column_name, data_type, is_nullable
573
+ FROM bq_cols.${dataset}.INFORMATION_SCHEMA.COLUMNS
574
+ WHERE table_name = '${table}'
575
+ ORDER BY ordinal_position
576
+ `);
577
+ expect(result.rows.length).toBeGreaterThan(0);
578
+ },
579
+ { timeout: 60000 },
580
+ );
581
+
582
+ it(
583
+ "should query data from attached BigQuery table",
584
+ async () => {
585
+ if (
586
+ !hasBigQueryCredentials() ||
587
+ !process.env.BIGQUERY_TEST_TABLE
588
+ ) {
589
+ console.log(
590
+ "Skipping: GOOGLE_APPLICATION_CREDENTIALS, BIGQUERY_TEST_PROJECT_ID, or BIGQUERY_TEST_TABLE not configured",
591
+ );
592
+ return;
593
+ }
594
+
595
+ const serviceAccountJson =
596
+ await readBigQueryServiceAccountJson();
597
+
598
+ const duckdbConnection: ApiConnection = {
599
+ name: "duckdb_bq_query",
600
+ type: "duckdb",
601
+ duckdbConnection: {
602
+ attachedDatabases: [
603
+ {
604
+ name: "bq_query",
605
+ type: "bigquery",
606
+ bigqueryConnection: {
607
+ defaultProjectId:
608
+ process.env.BIGQUERY_TEST_PROJECT_ID!,
609
+ serviceAccountKeyJson: serviceAccountJson,
610
+ },
611
+ },
612
+ ],
613
+ },
614
+ };
615
+
616
+ const { malloyConnections } = await createProjectConnections(
617
+ [duckdbConnection],
618
+ testProjectPath,
619
+ );
620
+
621
+ const connection = malloyConnections.get(
622
+ "duckdb_bq_query",
623
+ ) as DuckDBConnection;
624
+ createdConnections.push(connection);
625
+
626
+ const result = await connection.runSQL(
627
+ `SELECT * FROM bq_query.${process.env.BIGQUERY_TEST_TABLE!} LIMIT 1`,
628
+ );
629
+ expect(result.rows).toBeDefined();
630
+ expect(result.rows.length).toBeGreaterThan(0);
631
+ },
632
+ { timeout: 60000 },
633
+ );
634
+
635
+ it(
636
+ "should validate BigQuery service account key format",
637
+ async () => {
638
+ await expect(
639
+ createProjectConnections(
640
+ [
641
+ {
642
+ name: "duckdb_bq_invalid",
643
+ type: "duckdb",
644
+ duckdbConnection: {
645
+ attachedDatabases: [
646
+ {
647
+ name: "bq_invalid",
648
+ type: "bigquery",
649
+ bigqueryConnection: {
650
+ defaultProjectId: "test-project",
651
+ serviceAccountKeyJson: JSON.stringify({
652
+ invalid: "key",
653
+ }),
654
+ },
655
+ },
656
+ ],
657
+ },
658
+ },
659
+ ],
660
+ testProjectPath,
661
+ ),
662
+ ).rejects.toThrow(/Invalid service account key/);
663
+ },
664
+ { timeout: 30000 },
665
+ );
666
+ });
667
+
668
+ describe("DuckDB with Snowflake attachment", () => {
669
+ it(
670
+ "should create DuckDB connection with attached Snowflake database",
671
+ async () => {
672
+ if (!hasSnowflakeCredentials()) {
673
+ console.log("Skipping: Snowflake credentials not configured");
674
+ return;
675
+ }
676
+
677
+ const duckdbConnection: ApiConnection = {
678
+ name: "duckdb_with_snowflake",
679
+ type: "duckdb",
680
+ duckdbConnection: {
681
+ attachedDatabases: [
682
+ {
683
+ name: "sf_test",
684
+ type: "snowflake",
685
+ snowflakeConnection: {
686
+ account: process.env.SNOWFLAKE_TEST_ACCOUNT!,
687
+ username: process.env.SNOWFLAKE_TEST_USER!,
688
+ password: process.env.SNOWFLAKE_TEST_PASSWORD!,
689
+ warehouse: process.env.SNOWFLAKE_TEST_WAREHOUSE!,
690
+ database: process.env.SNOWFLAKE_TEST_DATABASE,
691
+ schema: process.env.SNOWFLAKE_TEST_SCHEMA,
692
+ },
693
+ },
694
+ ],
695
+ },
696
+ };
697
+
698
+ const { malloyConnections } = await createProjectConnections(
699
+ [duckdbConnection],
700
+ testProjectPath,
701
+ );
702
+
703
+ const connection = malloyConnections.get(
704
+ "duckdb_with_snowflake",
705
+ ) as DuckDBConnection;
706
+ expect(connection).toBeDefined();
707
+ createdConnections.push(connection);
708
+
709
+ const databases = await connection.runSQL("SHOW DATABASES");
710
+ const dbNames = databases.rows.map(
711
+ (row) => Object.values(row)[0],
712
+ );
713
+ expect(dbNames).toContain("sf_test");
714
+
715
+ const result = await connection.runSQL(
716
+ "SELECT * FROM snowflake_query('SELECT 1 as test', 'sf_test_secret')",
717
+ );
718
+ expect(result.rows).toBeDefined();
719
+ },
720
+ { timeout: 30000 },
721
+ );
722
+
723
+ it("should validate required Snowflake fields", async () => {
724
+ await expect(
725
+ createProjectConnections(
726
+ [
727
+ {
728
+ name: "duckdb_sf_incomplete",
729
+ type: "duckdb",
730
+ duckdbConnection: {
731
+ attachedDatabases: [
732
+ {
733
+ name: "sf_incomplete",
734
+ type: "snowflake",
735
+ snowflakeConnection: {
736
+ account: "test-account",
737
+ username: "test-user",
738
+ },
739
+ },
740
+ ],
741
+ },
742
+ },
743
+ ],
744
+ testProjectPath,
745
+ ),
746
+ ).rejects.toThrow(/required/);
747
+ });
748
+ });
749
+
750
+ describe("DuckDB with S3 attachment", () => {
751
+ it(
752
+ "should create DuckDB connection with S3 configuration",
753
+ async () => {
754
+ if (!hasS3Credentials()) {
755
+ console.log("Skipping: S3 credentials not configured");
756
+ return;
757
+ }
758
+
759
+ const duckdbConnection: ApiConnection = {
760
+ name: "duckdb_with_s3",
761
+ type: "duckdb",
762
+ duckdbConnection: {
763
+ attachedDatabases: [
764
+ {
765
+ name: "s3_test",
766
+ type: "s3",
767
+ s3Connection: {
768
+ accessKeyId: process.env.S3_TEST_ACCESS_KEY_ID!,
769
+ secretAccessKey:
770
+ process.env.S3_TEST_SECRET_ACCESS_KEY!,
771
+ region: process.env.S3_TEST_REGION || "us-east-1",
772
+ },
773
+ },
774
+ ],
775
+ },
776
+ };
777
+
778
+ const { malloyConnections } = await createProjectConnections(
779
+ [duckdbConnection],
780
+ testProjectPath,
781
+ );
782
+
783
+ const connection = malloyConnections.get(
784
+ "duckdb_with_s3",
785
+ ) as DuckDBConnection;
786
+ expect(connection).toBeDefined();
787
+ createdConnections.push(connection);
788
+
789
+ const secrets = await connection.runSQL(
790
+ "SELECT name FROM duckdb_secrets()",
791
+ );
792
+ const secretNames = secrets.rows.map(
793
+ (row) => Object.values(row)[0],
794
+ );
795
+ expect(
796
+ secretNames.some((name) => String(name).includes("s3")),
797
+ ).toBe(true);
798
+ },
799
+ { timeout: 30000 },
800
+ );
801
+
802
+ it(
803
+ "should handle S3 with custom endpoint",
804
+ async () => {
805
+ if (!hasS3Credentials()) {
806
+ console.log("Skipping: S3 credentials not configured");
807
+ return;
808
+ }
809
+
810
+ const duckdbConnection: ApiConnection = {
811
+ name: "duckdb_s3_custom",
812
+ type: "duckdb",
813
+ duckdbConnection: {
814
+ attachedDatabases: [
815
+ {
816
+ name: "s3_custom",
817
+ type: "s3",
818
+ s3Connection: {
819
+ accessKeyId: process.env.S3_TEST_ACCESS_KEY_ID!,
820
+ secretAccessKey:
821
+ process.env.S3_TEST_SECRET_ACCESS_KEY!,
822
+ region: "us-east-1",
823
+ endpoint: "https://s3.custom-endpoint.com",
824
+ },
825
+ },
826
+ ],
827
+ },
828
+ };
829
+
830
+ const { malloyConnections } = await createProjectConnections(
831
+ [duckdbConnection],
832
+ testProjectPath,
833
+ );
834
+
835
+ const connection = malloyConnections.get(
836
+ "duckdb_s3_custom",
837
+ ) as DuckDBConnection;
838
+ createdConnections.push(connection);
839
+ expect(connection).toBeDefined();
840
+ },
841
+ { timeout: 30000 },
842
+ );
843
+ });
844
+
845
+ describe("DuckDB with GCS attachment", () => {
846
+ it(
847
+ "should create DuckDB connection with GCS configuration",
848
+ async () => {
849
+ if (!hasGCSCredentials()) {
850
+ console.log("Skipping: GCS credentials not configured");
851
+ return;
852
+ }
853
+
854
+ const duckdbConnection: ApiConnection = {
855
+ name: "duckdb_with_gcs",
856
+ type: "duckdb",
857
+ duckdbConnection: {
858
+ attachedDatabases: [
859
+ {
860
+ name: "gcs_test",
861
+ type: "gcs",
862
+ gcsConnection: {
863
+ keyId: process.env.GCS_TEST_KEY_ID!,
864
+ secret: process.env.GCS_TEST_SECRET!,
865
+ },
866
+ },
867
+ ],
868
+ },
869
+ };
870
+
871
+ const { malloyConnections } = await createProjectConnections(
872
+ [duckdbConnection],
873
+ testProjectPath,
874
+ );
875
+
876
+ const connection = malloyConnections.get(
877
+ "duckdb_with_gcs",
878
+ ) as DuckDBConnection;
879
+ expect(connection).toBeDefined();
880
+ createdConnections.push(connection);
881
+
882
+ const secrets = await connection.runSQL(
883
+ "SELECT name FROM duckdb_secrets()",
884
+ );
885
+ const secretNames = secrets.rows.map(
886
+ (row) => Object.values(row)[0],
887
+ );
888
+ expect(
889
+ secretNames.some((name) => String(name).includes("gcs")),
890
+ ).toBe(true);
891
+ },
892
+ { timeout: 30000 },
893
+ );
894
+ });
895
+
896
+ describe("DuckDB with multiple attachments", () => {
897
+ it(
898
+ "should attach multiple databases to single DuckDB connection",
899
+ async () => {
900
+ const attachments: AttachedDatabase[] = [];
901
+
902
+ if (hasPostgresCredentials()) {
903
+ attachments.push({
904
+ name: "pg_multi",
905
+ type: "postgres",
906
+ postgresConnection: {
907
+ host: process.env.POSTGRES_TEST_HOST,
908
+ port: parseInt(
909
+ process.env.POSTGRES_TEST_PORT || "5432",
910
+ ),
911
+ userName: process.env.POSTGRES_TEST_USER!,
912
+ password: process.env.POSTGRES_TEST_PASSWORD!,
913
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
914
+ },
915
+ });
916
+ }
917
+
918
+ if (hasBigQueryCredentials()) {
919
+ const serviceAccountJson =
920
+ await readBigQueryServiceAccountJson();
921
+ attachments.push({
922
+ name: "bq_multi",
923
+ type: "bigquery",
924
+ bigqueryConnection: {
925
+ defaultProjectId: process.env.BIGQUERY_TEST_PROJECT_ID!,
926
+ serviceAccountKeyJson: serviceAccountJson,
927
+ },
928
+ });
929
+ }
930
+
931
+ if (hasS3Credentials()) {
932
+ attachments.push({
933
+ name: "s3_multi",
934
+ type: "s3",
935
+ s3Connection: {
936
+ accessKeyId: process.env.S3_TEST_ACCESS_KEY_ID!,
937
+ secretAccessKey: process.env.S3_TEST_SECRET_ACCESS_KEY!,
938
+ region: process.env.S3_TEST_REGION || "us-east-1",
939
+ },
940
+ });
941
+ }
942
+
943
+ if (attachments.length === 0) {
944
+ console.log("Skipping: No database credentials configured");
945
+ return;
946
+ }
947
+
948
+ const { malloyConnections } = await createProjectConnections(
949
+ [
950
+ {
951
+ name: "duckdb_multi",
952
+ type: "duckdb",
953
+ duckdbConnection: { attachedDatabases: attachments },
954
+ },
955
+ ],
956
+ testProjectPath,
957
+ );
958
+
959
+ const connection = malloyConnections.get(
960
+ "duckdb_multi",
961
+ ) as DuckDBConnection;
962
+ expect(connection).toBeDefined();
963
+ createdConnections.push(connection);
964
+
965
+ const databases = await connection.runSQL("SHOW DATABASES");
966
+ const dbNames = databases.rows.map(
967
+ (row) => Object.values(row)[0],
968
+ );
969
+ attachments.forEach((attachment) => {
970
+ expect(dbNames).toContain(attachment.name!);
971
+ });
972
+ },
973
+ { timeout: 60000 },
974
+ );
975
+ });
976
+
977
+ describe("error handling", () => {
978
+ describe("DuckLake connection type", () => {
979
+ it(
980
+ "should create DuckLake connection",
981
+ async () => {
982
+ if (!hasPostgresCredentials() || !hasS3Credentials()) {
983
+ console.log(
984
+ "Skipping: PostgreSQL and S3 credentials not configured",
985
+ );
986
+ return;
987
+ }
988
+
989
+ const { malloyConnections } = await createProjectConnections(
990
+ [
991
+ {
992
+ name: "ducklake_test",
993
+ type: "ducklake",
994
+ ducklakeConnection: {
995
+ catalog: {
996
+ postgresConnection: {
997
+ host: process.env.POSTGRES_TEST_HOST,
998
+ port: parseInt(
999
+ process.env.POSTGRES_TEST_PORT || "5432",
1000
+ ),
1001
+ userName: process.env.POSTGRES_TEST_USER!,
1002
+ password:
1003
+ process.env.POSTGRES_TEST_PASSWORD!,
1004
+ databaseName:
1005
+ process.env.POSTGRES_TEST_DATABASE,
1006
+ },
1007
+ },
1008
+ storage: {
1009
+ bucketUrl:
1010
+ process.env.S3_TEST_BUCKET_URL ||
1011
+ "s3://test-bucket",
1012
+ s3Connection: {
1013
+ accessKeyId:
1014
+ process.env.S3_TEST_ACCESS_KEY_ID!,
1015
+ secretAccessKey:
1016
+ process.env.S3_TEST_SECRET_ACCESS_KEY!,
1017
+ },
1018
+ },
1019
+ },
1020
+ },
1021
+ ],
1022
+ testProjectPath,
1023
+ );
1024
+
1025
+ const connection = malloyConnections.get(
1026
+ "ducklake_test",
1027
+ ) as DuckDBConnection;
1028
+ createdConnections.push(connection);
1029
+ expect(connection).toBeDefined();
1030
+
1031
+ // Verify DuckLake database is attached
1032
+ const databases = await connection.runSQL("SHOW DATABASES");
1033
+ const dbNames = databases.rows.map(
1034
+ (row) => Object.values(row)[0],
1035
+ );
1036
+ expect(dbNames).toContain("ducklake_test");
1037
+ },
1038
+ { timeout: 30000 },
1039
+ );
1040
+
1041
+ it("should throw error if DuckLake catalog connection is missing", async () => {
1042
+ await expect(
1043
+ createProjectConnections(
1044
+ [
1045
+ {
1046
+ name: "ducklake_no_catalog",
1047
+ type: "ducklake",
1048
+ ducklakeConnection: {
1049
+ storage: {
1050
+ bucketUrl: "s3://test-bucket",
1051
+ s3Connection: {
1052
+ accessKeyId: "test",
1053
+ secretAccessKey: "test",
1054
+ },
1055
+ },
1056
+ },
1057
+ } as ApiConnection,
1058
+ ],
1059
+ testProjectPath,
1060
+ ),
1061
+ ).rejects.toThrow(
1062
+ /PostgreSQL connection configuration is required/,
1063
+ );
1064
+ });
1065
+
1066
+ it("should throw error if DuckLake connection config is missing", async () => {
1067
+ await expect(
1068
+ createProjectConnections(
1069
+ [
1070
+ {
1071
+ name: "ducklake_missing_config",
1072
+ type: "ducklake",
1073
+ },
1074
+ ],
1075
+ testProjectPath,
1076
+ ),
1077
+ ).rejects.toThrow(
1078
+ /DuckLake connection configuration is missing/,
1079
+ );
1080
+ });
1081
+ });
1082
+
1083
+ it("should throw error if DuckDB connection name conflicts with attached database", async () => {
1084
+ await expect(
1085
+ createProjectConnections(
1086
+ [
1087
+ {
1088
+ name: "conflict_db",
1089
+ type: "duckdb",
1090
+ duckdbConnection: {
1091
+ attachedDatabases: [
1092
+ {
1093
+ name: "conflict_db",
1094
+ type: "postgres",
1095
+ postgresConnection: {
1096
+ connectionString:
1097
+ "postgresql://localhost/test",
1098
+ },
1099
+ },
1100
+ ],
1101
+ },
1102
+ },
1103
+ ],
1104
+ testProjectPath,
1105
+ ),
1106
+ ).rejects.toThrow(/cannot conflict/);
1107
+ });
1108
+
1109
+ it("should throw error if connection name is 'duckdb'", async () => {
1110
+ await expect(
1111
+ createProjectConnections(
1112
+ [
1113
+ {
1114
+ name: "duckdb",
1115
+ type: "duckdb",
1116
+ duckdbConnection: { attachedDatabases: [] },
1117
+ },
1118
+ ],
1119
+ testProjectPath,
1120
+ ),
1121
+ ).rejects.toThrow(/cannot be 'duckdb'/);
1122
+ });
1123
+
1124
+ it("should throw error if no attached databases configured", async () => {
1125
+ await expect(
1126
+ createProjectConnections(
1127
+ [
1128
+ {
1129
+ name: "empty_duckdb",
1130
+ type: "duckdb",
1131
+ duckdbConnection: { attachedDatabases: [] },
1132
+ },
1133
+ ],
1134
+ testProjectPath,
1135
+ ),
1136
+ ).rejects.toThrow(/at least one attached database/);
1137
+ });
1138
+
1139
+ it("should handle already attached database gracefully", async () => {
1140
+ if (!hasPostgresCredentials()) {
1141
+ console.log("Skipping: PostgreSQL credentials not configured");
1142
+ return;
1143
+ }
1144
+
1145
+ const postgresAttachment: AttachedDatabase = {
1146
+ name: "pg_duplicate",
1147
+ type: "postgres",
1148
+ postgresConnection: {
1149
+ host: process.env.POSTGRES_TEST_HOST,
1150
+ port: parseInt(process.env.POSTGRES_TEST_PORT || "5432"),
1151
+ userName: process.env.POSTGRES_TEST_USER!,
1152
+ password: process.env.POSTGRES_TEST_PASSWORD!,
1153
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
1154
+ },
1155
+ };
1156
+
1157
+ const { malloyConnections } = await createProjectConnections(
1158
+ [
1159
+ {
1160
+ name: "duckdb_duplicate_test",
1161
+ type: "duckdb",
1162
+ duckdbConnection: {
1163
+ attachedDatabases: [
1164
+ postgresAttachment,
1165
+ postgresAttachment,
1166
+ ],
1167
+ },
1168
+ },
1169
+ ],
1170
+ testProjectPath,
1171
+ );
1172
+
1173
+ const connection = malloyConnections.get(
1174
+ "duckdb_duplicate_test",
1175
+ ) as DuckDBConnection;
1176
+ createdConnections.push(connection);
1177
+ expect(connection).toBeDefined();
1178
+ });
1179
+ });
1180
+ });
1181
+
1182
+ describe("testConnectionConfig", () => {
1183
+ it(
1184
+ "should successfully test valid PostgreSQL connection",
1185
+ async () => {
1186
+ if (!hasPostgresCredentials()) {
1187
+ console.log("Skipping: PostgreSQL credentials not configured");
1188
+ return;
1189
+ }
1190
+
1191
+ const result = await testConnectionConfig({
1192
+ name: "test_postgres",
1193
+ type: "postgres",
1194
+ postgresConnection: {
1195
+ host: process.env.POSTGRES_TEST_HOST,
1196
+ port: parseInt(process.env.POSTGRES_TEST_PORT || "5432"),
1197
+ userName: process.env.POSTGRES_TEST_USER!,
1198
+ password: process.env.POSTGRES_TEST_PASSWORD!,
1199
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
1200
+ },
1201
+ });
1202
+
1203
+ expect(result.status).toBe("ok");
1204
+ expect(result.errorMessage).toBe("");
1205
+ },
1206
+ { timeout: 30000 },
1207
+ );
1208
+
1209
+ it(
1210
+ "should fail for invalid PostgreSQL credentials",
1211
+ async () => {
1212
+ const result = await testConnectionConfig({
1213
+ name: "test_postgres_invalid",
1214
+ type: "postgres",
1215
+ postgresConnection: {
1216
+ host: process.env.POSTGRES_TEST_HOST,
1217
+ port: parseInt(process.env.POSTGRES_TEST_PORT || "5432"),
1218
+ userName: "invalid_user",
1219
+ password: "invalid_password",
1220
+ databaseName: "nonexistent",
1221
+ },
1222
+ });
1223
+
1224
+ expect(result.status).toBe("failed");
1225
+ expect(result.errorMessage).toBeDefined();
1226
+ },
1227
+ { timeout: 30000 },
1228
+ );
1229
+
1230
+ it("should fail for missing connection name", async () => {
1231
+ const result = await testConnectionConfig({
1232
+ name: "",
1233
+ type: "postgres",
1234
+ postgresConnection: {},
1235
+ });
1236
+
1237
+ expect(result.status).toBe("failed");
1238
+ expect(result.errorMessage).toContain("name is required");
1239
+ });
1240
+ });
1241
+
1242
+ describe("SQL injection prevention", () => {
1243
+ it("should properly escape special characters in credentials", async () => {
1244
+ if (!hasPostgresCredentials()) {
1245
+ console.log("Skipping: PostgreSQL credentials not configured");
1246
+ return;
1247
+ }
1248
+
1249
+ const { malloyConnections } = await createProjectConnections(
1250
+ [
1251
+ {
1252
+ name: "duckdb_special_chars",
1253
+ type: "duckdb",
1254
+ duckdbConnection: {
1255
+ attachedDatabases: [
1256
+ {
1257
+ name: "pg_special",
1258
+ type: "postgres",
1259
+ postgresConnection: {
1260
+ host: process.env.POSTGRES_TEST_HOST,
1261
+ port: parseInt(
1262
+ process.env.POSTGRES_TEST_PORT || "5432",
1263
+ ),
1264
+ userName: process.env.POSTGRES_TEST_USER!,
1265
+ password: process.env.POSTGRES_TEST_PASSWORD!,
1266
+ databaseName: process.env.POSTGRES_TEST_DATABASE,
1267
+ },
1268
+ },
1269
+ ],
1270
+ },
1271
+ },
1272
+ ],
1273
+ testProjectPath,
1274
+ );
1275
+
1276
+ const connection = malloyConnections.get(
1277
+ "duckdb_special_chars",
1278
+ ) as DuckDBConnection;
1279
+ createdConnections.push(connection);
1280
+ expect(connection).toBeDefined();
1281
+ });
1282
+ });
1283
+
1284
+ describe("connection attributes", () => {
1285
+ it(
1286
+ "should return correct attributes for DuckDB connection",
1287
+ async () => {
1288
+ if (!hasPostgresCredentials()) {
1289
+ console.log("Skipping: PostgreSQL credentials not configured");
1290
+ return;
1291
+ }
1292
+
1293
+ const { apiConnections } = await createProjectConnections(
1294
+ [
1295
+ {
1296
+ name: "duckdb_attrs",
1297
+ type: "duckdb",
1298
+ duckdbConnection: {
1299
+ attachedDatabases: [
1300
+ {
1301
+ name: "pg_attrs",
1302
+ type: "postgres",
1303
+ postgresConnection: {
1304
+ host: process.env.POSTGRES_TEST_HOST,
1305
+ port: parseInt(
1306
+ process.env.POSTGRES_TEST_PORT || "5432",
1307
+ ),
1308
+ userName: process.env.POSTGRES_TEST_USER!,
1309
+ password: process.env.POSTGRES_TEST_PASSWORD!,
1310
+ databaseName:
1311
+ process.env.POSTGRES_TEST_DATABASE,
1312
+ },
1313
+ },
1314
+ ],
1315
+ },
1316
+ },
1317
+ ],
1318
+ testProjectPath,
1319
+ );
1320
+
1321
+ const connection = apiConnections[0];
1322
+ expect(connection.attributes).toBeDefined();
1323
+ expect(connection.attributes?.dialectName).toBe("duckdb");
1324
+ expect(typeof connection.attributes?.canPersist).toBe("boolean");
1325
+ expect(typeof connection.attributes?.canStream).toBe("boolean");
1326
+ expect(typeof connection.attributes?.isPool).toBe("boolean");
1327
+ },
1328
+ { timeout: 30000 },
1329
+ );
1330
+ });
1331
+ });