@duckcodeailabs/dql-cli 1.4.1 → 1.4.3
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/package.json +21 -28
- package/LICENSE +0 -123
- package/README.md +0 -71
- package/dist/apps-api.d.ts +0 -16
- package/dist/apps-api.d.ts.map +0 -1
- package/dist/apps-api.js +0 -249
- package/dist/apps-api.js.map +0 -1
- package/dist/args.d.ts +0 -30
- package/dist/args.d.ts.map +0 -1
- package/dist/args.js +0 -105
- package/dist/args.js.map +0 -1
- package/dist/args.test.d.ts +0 -2
- package/dist/args.test.d.ts.map +0 -1
- package/dist/args.test.js +0 -33
- package/dist/args.test.js.map +0 -1
- package/dist/assets/dql-notebook/assets/codemirror-DJYUkPr1.js +0 -11
- package/dist/assets/dql-notebook/assets/index-BJ7MV8Gv.js +0 -847
- package/dist/assets/dql-notebook/assets/index-DrhoZmtv.css +0 -1
- package/dist/assets/dql-notebook/assets/react-CRB3T2We.js +0 -32
- package/dist/assets/dql-notebook/index.html +0 -18
- package/dist/assets/notebook-browser/app.js +0 -548
- package/dist/assets/notebook-browser/index.html +0 -83
- package/dist/assets/notebook-browser/styles.css +0 -336
- package/dist/block-templates.d.ts +0 -8
- package/dist/block-templates.d.ts.map +0 -1
- package/dist/block-templates.js +0 -60
- package/dist/block-templates.js.map +0 -1
- package/dist/commands/agent.d.ts +0 -19
- package/dist/commands/agent.d.ts.map +0 -1
- package/dist/commands/agent.js +0 -117
- package/dist/commands/agent.js.map +0 -1
- package/dist/commands/app.d.ts +0 -32
- package/dist/commands/app.d.ts.map +0 -1
- package/dist/commands/app.js +0 -307
- package/dist/commands/app.js.map +0 -1
- package/dist/commands/build.d.ts +0 -3
- package/dist/commands/build.d.ts.map +0 -1
- package/dist/commands/build.js +0 -69
- package/dist/commands/build.js.map +0 -1
- package/dist/commands/build.test.d.ts +0 -2
- package/dist/commands/build.test.d.ts.map +0 -1
- package/dist/commands/build.test.js +0 -44
- package/dist/commands/build.test.js.map +0 -1
- package/dist/commands/certify.d.ts +0 -3
- package/dist/commands/certify.d.ts.map +0 -1
- package/dist/commands/certify.js +0 -228
- package/dist/commands/certify.js.map +0 -1
- package/dist/commands/compile.d.ts +0 -21
- package/dist/commands/compile.d.ts.map +0 -1
- package/dist/commands/compile.js +0 -198
- package/dist/commands/compile.js.map +0 -1
- package/dist/commands/compile.test.d.ts +0 -2
- package/dist/commands/compile.test.d.ts.map +0 -1
- package/dist/commands/compile.test.js +0 -115
- package/dist/commands/compile.test.js.map +0 -1
- package/dist/commands/diff.d.ts +0 -3
- package/dist/commands/diff.d.ts.map +0 -1
- package/dist/commands/diff.js +0 -52
- package/dist/commands/diff.js.map +0 -1
- package/dist/commands/doctor.d.ts +0 -3
- package/dist/commands/doctor.d.ts.map +0 -1
- package/dist/commands/doctor.js +0 -191
- package/dist/commands/doctor.js.map +0 -1
- package/dist/commands/doctor.test.d.ts +0 -2
- package/dist/commands/doctor.test.d.ts.map +0 -1
- package/dist/commands/doctor.test.js +0 -43
- package/dist/commands/doctor.test.js.map +0 -1
- package/dist/commands/fmt.d.ts +0 -3
- package/dist/commands/fmt.d.ts.map +0 -1
- package/dist/commands/fmt.js +0 -53
- package/dist/commands/fmt.js.map +0 -1
- package/dist/commands/info.d.ts +0 -3
- package/dist/commands/info.d.ts.map +0 -1
- package/dist/commands/info.js +0 -56
- package/dist/commands/info.js.map +0 -1
- package/dist/commands/init.d.ts +0 -3
- package/dist/commands/init.d.ts.map +0 -1
- package/dist/commands/init.js +0 -250
- package/dist/commands/init.js.map +0 -1
- package/dist/commands/init.test.d.ts +0 -2
- package/dist/commands/init.test.d.ts.map +0 -1
- package/dist/commands/init.test.js +0 -118
- package/dist/commands/init.test.js.map +0 -1
- package/dist/commands/lineage.d.ts +0 -24
- package/dist/commands/lineage.d.ts.map +0 -1
- package/dist/commands/lineage.js +0 -634
- package/dist/commands/lineage.js.map +0 -1
- package/dist/commands/mcp.d.ts +0 -7
- package/dist/commands/mcp.d.ts.map +0 -1
- package/dist/commands/mcp.js +0 -16
- package/dist/commands/mcp.js.map +0 -1
- package/dist/commands/migrate.d.ts +0 -12
- package/dist/commands/migrate.d.ts.map +0 -1
- package/dist/commands/migrate.js +0 -192
- package/dist/commands/migrate.js.map +0 -1
- package/dist/commands/new.d.ts +0 -3
- package/dist/commands/new.d.ts.map +0 -1
- package/dist/commands/new.js +0 -490
- package/dist/commands/new.js.map +0 -1
- package/dist/commands/new.test.d.ts +0 -2
- package/dist/commands/new.test.d.ts.map +0 -1
- package/dist/commands/new.test.js +0 -191
- package/dist/commands/new.test.js.map +0 -1
- package/dist/commands/notebook.d.ts +0 -3
- package/dist/commands/notebook.d.ts.map +0 -1
- package/dist/commands/notebook.js +0 -46
- package/dist/commands/notebook.js.map +0 -1
- package/dist/commands/parse.d.ts +0 -3
- package/dist/commands/parse.d.ts.map +0 -1
- package/dist/commands/parse.js +0 -63
- package/dist/commands/parse.js.map +0 -1
- package/dist/commands/preview.d.ts +0 -3
- package/dist/commands/preview.d.ts.map +0 -1
- package/dist/commands/preview.js +0 -42
- package/dist/commands/preview.js.map +0 -1
- package/dist/commands/schedule.d.ts +0 -3
- package/dist/commands/schedule.d.ts.map +0 -1
- package/dist/commands/schedule.js +0 -215
- package/dist/commands/schedule.js.map +0 -1
- package/dist/commands/semantic.d.ts +0 -12
- package/dist/commands/semantic.d.ts.map +0 -1
- package/dist/commands/semantic.js +0 -356
- package/dist/commands/semantic.js.map +0 -1
- package/dist/commands/serve.d.ts +0 -3
- package/dist/commands/serve.d.ts.map +0 -1
- package/dist/commands/serve.js +0 -30
- package/dist/commands/serve.js.map +0 -1
- package/dist/commands/slack.d.ts +0 -13
- package/dist/commands/slack.d.ts.map +0 -1
- package/dist/commands/slack.js +0 -53
- package/dist/commands/slack.js.map +0 -1
- package/dist/commands/sync.d.ts +0 -3
- package/dist/commands/sync.d.ts.map +0 -1
- package/dist/commands/sync.js +0 -192
- package/dist/commands/sync.js.map +0 -1
- package/dist/commands/sync.test.d.ts +0 -2
- package/dist/commands/sync.test.d.ts.map +0 -1
- package/dist/commands/sync.test.js +0 -147
- package/dist/commands/sync.test.js.map +0 -1
- package/dist/commands/test.d.ts +0 -3
- package/dist/commands/test.d.ts.map +0 -1
- package/dist/commands/test.js +0 -167
- package/dist/commands/test.js.map +0 -1
- package/dist/commands/validate.d.ts +0 -3
- package/dist/commands/validate.d.ts.map +0 -1
- package/dist/commands/validate.js +0 -116
- package/dist/commands/validate.js.map +0 -1
- package/dist/commands/verify.d.ts +0 -11
- package/dist/commands/verify.d.ts.map +0 -1
- package/dist/commands/verify.js +0 -74
- package/dist/commands/verify.js.map +0 -1
- package/dist/digest.d.ts +0 -10
- package/dist/digest.d.ts.map +0 -1
- package/dist/digest.js +0 -83
- package/dist/digest.js.map +0 -1
- package/dist/git-service.d.ts +0 -17
- package/dist/git-service.d.ts.map +0 -1
- package/dist/git-service.js +0 -54
- package/dist/git-service.js.map +0 -1
- package/dist/index.d.ts +0 -3
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/llm/index.d.ts +0 -4
- package/dist/llm/index.d.ts.map +0 -1
- package/dist/llm/index.js +0 -16
- package/dist/llm/index.js.map +0 -1
- package/dist/llm/providers/claude-agent-sdk.d.ts +0 -3
- package/dist/llm/providers/claude-agent-sdk.d.ts.map +0 -1
- package/dist/llm/providers/claude-agent-sdk.js +0 -174
- package/dist/llm/providers/claude-agent-sdk.js.map +0 -1
- package/dist/llm/providers/claude-code.d.ts +0 -8
- package/dist/llm/providers/claude-code.d.ts.map +0 -1
- package/dist/llm/providers/claude-code.js +0 -171
- package/dist/llm/providers/claude-code.js.map +0 -1
- package/dist/llm/tools.d.ts +0 -9
- package/dist/llm/tools.d.ts.map +0 -1
- package/dist/llm/tools.js +0 -112
- package/dist/llm/tools.js.map +0 -1
- package/dist/llm/types.d.ts +0 -70
- package/dist/llm/types.d.ts.map +0 -1
- package/dist/llm/types.js +0 -2
- package/dist/llm/types.js.map +0 -1
- package/dist/local-runtime.d.ts +0 -142
- package/dist/local-runtime.d.ts.map +0 -1
- package/dist/local-runtime.js +0 -3836
- package/dist/local-runtime.js.map +0 -1
- package/dist/local-runtime.test.d.ts +0 -2
- package/dist/local-runtime.test.d.ts.map +0 -1
- package/dist/local-runtime.test.js +0 -241
- package/dist/local-runtime.test.js.map +0 -1
- package/dist/open-browser.d.ts +0 -2
- package/dist/open-browser.d.ts.map +0 -1
- package/dist/open-browser.js +0 -29
- package/dist/open-browser.js.map +0 -1
- package/dist/schedule/alerts.d.ts +0 -5
- package/dist/schedule/alerts.d.ts.map +0 -1
- package/dist/schedule/alerts.js +0 -54
- package/dist/schedule/alerts.js.map +0 -1
- package/dist/schedule/discovery.d.ts +0 -4
- package/dist/schedule/discovery.d.ts.map +0 -1
- package/dist/schedule/discovery.js +0 -36
- package/dist/schedule/discovery.js.map +0 -1
- package/dist/schedule/notifiers/email.d.ts +0 -3
- package/dist/schedule/notifiers/email.d.ts.map +0 -1
- package/dist/schedule/notifiers/email.js +0 -76
- package/dist/schedule/notifiers/email.js.map +0 -1
- package/dist/schedule/notifiers/file.d.ts +0 -3
- package/dist/schedule/notifiers/file.d.ts.map +0 -1
- package/dist/schedule/notifiers/file.js +0 -50
- package/dist/schedule/notifiers/file.js.map +0 -1
- package/dist/schedule/notifiers/index.d.ts +0 -10
- package/dist/schedule/notifiers/index.d.ts.map +0 -1
- package/dist/schedule/notifiers/index.js +0 -33
- package/dist/schedule/notifiers/index.js.map +0 -1
- package/dist/schedule/notifiers/slack.d.ts +0 -3
- package/dist/schedule/notifiers/slack.d.ts.map +0 -1
- package/dist/schedule/notifiers/slack.js +0 -58
- package/dist/schedule/notifiers/slack.js.map +0 -1
- package/dist/schedule/runner.d.ts +0 -11
- package/dist/schedule/runner.d.ts.map +0 -1
- package/dist/schedule/runner.js +0 -109
- package/dist/schedule/runner.js.map +0 -1
- package/dist/schedule/runs.d.ts +0 -5
- package/dist/schedule/runs.d.ts.map +0 -1
- package/dist/schedule/runs.js +0 -41
- package/dist/schedule/runs.js.map +0 -1
- package/dist/schedule/service.d.ts +0 -12
- package/dist/schedule/service.d.ts.map +0 -1
- package/dist/schedule/service.js +0 -49
- package/dist/schedule/service.js.map +0 -1
- package/dist/schedule/types.d.ts +0 -64
- package/dist/schedule/types.d.ts.map +0 -1
- package/dist/schedule/types.js +0 -2
- package/dist/schedule/types.js.map +0 -1
- package/dist/semantic-import.d.ts +0 -127
- package/dist/semantic-import.d.ts.map +0 -1
- package/dist/semantic-import.js +0 -713
- package/dist/semantic-import.js.map +0 -1
- package/dist/semantic-import.test.d.ts +0 -2
- package/dist/semantic-import.test.d.ts.map +0 -1
- package/dist/semantic-import.test.js +0 -95
- package/dist/semantic-import.test.js.map +0 -1
- /package/{dist/index.js → index.js} +0 -0
|
@@ -1,241 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it } from 'vitest';
|
|
2
|
-
import { createBlockArtifacts, createSemanticBuilderBlock, formatLocalQueryRuntimeError, normalizeProjectConnection, prepareLocalExecution, resolveProjectRelativeSqlPaths, serializeJSON, validateBlockStudioSource, } from './local-runtime.js';
|
|
3
|
-
import { afterEach } from 'vitest';
|
|
4
|
-
import { mkdtempSync, readFileSync, rmSync, writeFileSync } from 'node:fs';
|
|
5
|
-
import { join } from 'node:path';
|
|
6
|
-
import { tmpdir } from 'node:os';
|
|
7
|
-
import { SemanticLayer } from '@duckcodeailabs/dql-core';
|
|
8
|
-
const tempDirs = [];
|
|
9
|
-
afterEach(() => {
|
|
10
|
-
while (tempDirs.length > 0) {
|
|
11
|
-
const dir = tempDirs.pop();
|
|
12
|
-
if (dir)
|
|
13
|
-
rmSync(dir, { recursive: true, force: true });
|
|
14
|
-
}
|
|
15
|
-
});
|
|
16
|
-
describe('formatLocalQueryRuntimeError', () => {
|
|
17
|
-
it('explains missing DuckDB native bindings with actionable guidance', () => {
|
|
18
|
-
const message = formatLocalQueryRuntimeError({ driver: 'file', filepath: ':memory:' }, new Error("Cannot find module '/tmp/duckdb/lib/binding/duckdb.node'"));
|
|
19
|
-
expect(message).toContain('DuckDB native bindings could not be loaded');
|
|
20
|
-
expect(message).toContain(`Current Node.js runtime: ${process.versions.node}`);
|
|
21
|
-
expect(message).toContain('Node 18, 20, or 22');
|
|
22
|
-
expect(message).toContain('pnpm install');
|
|
23
|
-
});
|
|
24
|
-
});
|
|
25
|
-
describe('serializeJSON', () => {
|
|
26
|
-
it('serializes safe bigint values as numbers', () => {
|
|
27
|
-
expect(serializeJSON({ revenue: 42n })).toBe('{"revenue":42}');
|
|
28
|
-
});
|
|
29
|
-
it('serializes unsafe bigint values as strings', () => {
|
|
30
|
-
const value = BigInt(Number.MAX_SAFE_INTEGER) + 1n;
|
|
31
|
-
expect(serializeJSON({ revenue: value })).toBe(`{"revenue":"${value.toString()}"}`);
|
|
32
|
-
});
|
|
33
|
-
});
|
|
34
|
-
describe('resolveProjectRelativeSqlPaths', () => {
|
|
35
|
-
it('rewrites notebook sample file paths relative to the selected project', () => {
|
|
36
|
-
const sql = "SELECT * FROM read_csv_auto('./data/revenue.csv')";
|
|
37
|
-
const resolved = resolveProjectRelativeSqlPaths(sql, '/tmp/demo-project');
|
|
38
|
-
expect(resolved).toBe("SELECT * FROM read_csv_auto('/tmp/demo-project/data/revenue.csv')");
|
|
39
|
-
});
|
|
40
|
-
it('leaves unrelated string literals untouched', () => {
|
|
41
|
-
const sql = "SELECT './data/revenue.csv' AS label";
|
|
42
|
-
expect(resolveProjectRelativeSqlPaths(sql, '/tmp/demo-project')).toBe(sql);
|
|
43
|
-
});
|
|
44
|
-
});
|
|
45
|
-
describe('normalizeProjectConnection', () => {
|
|
46
|
-
it('resolves relative local database paths against the project root', () => {
|
|
47
|
-
expect(normalizeProjectConnection({ driver: 'duckdb', filepath: './local/dev.duckdb' }, '/tmp/demo-project')).toEqual({ driver: 'duckdb', filepath: '/tmp/demo-project/local/dev.duckdb' });
|
|
48
|
-
});
|
|
49
|
-
});
|
|
50
|
-
describe('prepareLocalExecution', () => {
|
|
51
|
-
it('rewrites SQL paths for file-backed notebook queries', () => {
|
|
52
|
-
const prepared = prepareLocalExecution("SELECT * FROM read_csv_auto('./data/revenue.csv')", { driver: 'file', filepath: ':memory:' }, '/tmp/demo-project', { dataDir: './data' });
|
|
53
|
-
expect(prepared.connection).toEqual({ driver: 'file', filepath: ':memory:' });
|
|
54
|
-
expect(prepared.sql).toBe("SELECT * FROM read_csv_auto('/tmp/demo-project/data/revenue.csv')");
|
|
55
|
-
});
|
|
56
|
-
});
|
|
57
|
-
describe('semantic block save artifacts', () => {
|
|
58
|
-
it('writes both the block file and semantic companion metadata for save-from-cell flows', () => {
|
|
59
|
-
const projectRoot = mkdtempSync(join(tmpdir(), 'dql-block-artifacts-'));
|
|
60
|
-
tempDirs.push(projectRoot);
|
|
61
|
-
writeFileSync(join(projectRoot, 'dql.config.json'), '{}\n');
|
|
62
|
-
const created = createBlockArtifacts(projectRoot, {
|
|
63
|
-
name: 'Revenue Summary',
|
|
64
|
-
domain: 'finance',
|
|
65
|
-
content: 'SELECT @metric(total_revenue), @dim(order_date);',
|
|
66
|
-
description: 'Finance summary block',
|
|
67
|
-
tags: ['finance', 'exec'],
|
|
68
|
-
});
|
|
69
|
-
expect(created.path).toBe('blocks/finance/revenue-summary.dql');
|
|
70
|
-
expect(created.companionPath).toBe('semantic-layer/blocks/finance/revenue-summary.yaml');
|
|
71
|
-
expect(readFileSync(join(projectRoot, created.path), 'utf-8')).toContain('@metric(total_revenue)');
|
|
72
|
-
const companion = readFileSync(join(projectRoot, created.companionPath), 'utf-8');
|
|
73
|
-
expect(companion).toContain('provider: dql');
|
|
74
|
-
expect(companion).toContain('semanticMetrics:');
|
|
75
|
-
expect(companion).toContain(' - total_revenue');
|
|
76
|
-
expect(companion).toContain('semanticDimensions:');
|
|
77
|
-
expect(companion).toContain(' - order_date');
|
|
78
|
-
expect(companion).toContain('reviewStatus: draft');
|
|
79
|
-
});
|
|
80
|
-
it('writes semantic builder blocks with lineage companion metadata', () => {
|
|
81
|
-
const projectRoot = mkdtempSync(join(tmpdir(), 'dql-builder-artifacts-'));
|
|
82
|
-
tempDirs.push(projectRoot);
|
|
83
|
-
writeFileSync(join(projectRoot, 'dql.config.json'), '{}\n');
|
|
84
|
-
const created = createSemanticBuilderBlock(projectRoot, {
|
|
85
|
-
name: 'Executive Revenue',
|
|
86
|
-
domain: 'finance',
|
|
87
|
-
description: 'Executive revenue cut',
|
|
88
|
-
owner: 'finance-analytics',
|
|
89
|
-
tags: ['finance'],
|
|
90
|
-
metrics: ['total_revenue'],
|
|
91
|
-
dimensions: ['sales_channel'],
|
|
92
|
-
timeDimension: { name: 'order_date', granularity: 'month' },
|
|
93
|
-
chart: 'line',
|
|
94
|
-
blockType: 'semantic',
|
|
95
|
-
sql: 'SELECT 1',
|
|
96
|
-
tables: ['analytics.orders'],
|
|
97
|
-
provider: 'dbt',
|
|
98
|
-
});
|
|
99
|
-
expect(created.path).toBe('blocks/finance/executive-revenue.dql');
|
|
100
|
-
expect(created.content).toContain('type = "semantic"');
|
|
101
|
-
expect(created.content).toContain('metric = "total_revenue"');
|
|
102
|
-
const companion = readFileSync(join(projectRoot, created.companionPath), 'utf-8');
|
|
103
|
-
expect(companion).toContain('provider: dbt');
|
|
104
|
-
expect(companion).toContain('lineage:');
|
|
105
|
-
expect(companion).toContain('analytics.orders');
|
|
106
|
-
expect(companion).toContain('semanticMetrics:');
|
|
107
|
-
expect(companion).toContain(' - total_revenue');
|
|
108
|
-
expect(companion).toContain('semanticDimensions:');
|
|
109
|
-
expect(companion).toContain(' - sales_channel');
|
|
110
|
-
expect(companion).toContain(' - order_date');
|
|
111
|
-
});
|
|
112
|
-
});
|
|
113
|
-
describe('validateBlockStudioSource', () => {
|
|
114
|
-
const semanticLayer = new SemanticLayer({
|
|
115
|
-
metrics: [
|
|
116
|
-
{
|
|
117
|
-
name: 'total_revenue',
|
|
118
|
-
label: 'Total Revenue',
|
|
119
|
-
description: 'Revenue metric',
|
|
120
|
-
domain: 'finance',
|
|
121
|
-
sql: 'SUM(revenue)',
|
|
122
|
-
type: 'sum',
|
|
123
|
-
table: 'orders',
|
|
124
|
-
tags: [],
|
|
125
|
-
},
|
|
126
|
-
],
|
|
127
|
-
dimensions: [
|
|
128
|
-
{
|
|
129
|
-
name: 'customer_type',
|
|
130
|
-
label: 'Customer Type',
|
|
131
|
-
description: 'Customer type dimension',
|
|
132
|
-
domain: 'finance',
|
|
133
|
-
sql: 'customer_type',
|
|
134
|
-
type: 'string',
|
|
135
|
-
table: 'orders',
|
|
136
|
-
tags: [],
|
|
137
|
-
},
|
|
138
|
-
],
|
|
139
|
-
hierarchies: [],
|
|
140
|
-
});
|
|
141
|
-
it('composes executable SQL for semantic blocks with metric and dimensions', () => {
|
|
142
|
-
const source = `block "Revenue by Type" {
|
|
143
|
-
domain = "finance"
|
|
144
|
-
type = "semantic"
|
|
145
|
-
description = ""
|
|
146
|
-
owner = ""
|
|
147
|
-
tags = []
|
|
148
|
-
metric = "total_revenue"
|
|
149
|
-
dimensions = ["customer_type"]
|
|
150
|
-
}`;
|
|
151
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
152
|
-
expect(validation.valid).toBe(true);
|
|
153
|
-
expect(validation.executableSql).toContain('SUM(revenue) AS total_revenue');
|
|
154
|
-
expect(validation.executableSql).toContain('customer_type AS customer_type');
|
|
155
|
-
expect(validation.executableSql).toContain('GROUP BY customer_type');
|
|
156
|
-
});
|
|
157
|
-
it('returns an actionable diagnostic when a semantic block is missing a metric', () => {
|
|
158
|
-
const source = `block "Revenue by Type" {
|
|
159
|
-
domain = "finance"
|
|
160
|
-
type = "semantic"
|
|
161
|
-
description = ""
|
|
162
|
-
owner = ""
|
|
163
|
-
tags = []
|
|
164
|
-
dimensions = ["customer_type"]
|
|
165
|
-
}`;
|
|
166
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
167
|
-
expect(validation.valid).toBe(false);
|
|
168
|
-
expect(validation.executableSql).toBeNull();
|
|
169
|
-
expect(validation.diagnostics.some((item) => item.code === 'semantic_metric_missing')).toBe(true);
|
|
170
|
-
});
|
|
171
|
-
it('returns a semantic validation error for unknown dimensions', () => {
|
|
172
|
-
const source = `block "Revenue by Type" {
|
|
173
|
-
domain = "finance"
|
|
174
|
-
type = "semantic"
|
|
175
|
-
description = ""
|
|
176
|
-
owner = ""
|
|
177
|
-
tags = []
|
|
178
|
-
metric = "total_revenue"
|
|
179
|
-
dimensions = ["missing_dimension"]
|
|
180
|
-
}`;
|
|
181
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
182
|
-
expect(validation.valid).toBe(false);
|
|
183
|
-
expect(validation.diagnostics.some((item) => item.code === 'semantic_ref' && item.message.includes('missing_dimension'))).toBe(true);
|
|
184
|
-
});
|
|
185
|
-
it('keeps custom block validation behavior unchanged', () => {
|
|
186
|
-
const source = `block "Custom Revenue" {
|
|
187
|
-
domain = "finance"
|
|
188
|
-
type = "custom"
|
|
189
|
-
description = ""
|
|
190
|
-
owner = ""
|
|
191
|
-
tags = []
|
|
192
|
-
|
|
193
|
-
query = """
|
|
194
|
-
SELECT revenue
|
|
195
|
-
FROM orders
|
|
196
|
-
"""
|
|
197
|
-
}`;
|
|
198
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
199
|
-
expect(validation.valid).toBe(true);
|
|
200
|
-
expect(validation.executableSql).toContain('SELECT revenue');
|
|
201
|
-
});
|
|
202
|
-
it('resolves semantic refs inside custom block SQL before execution', () => {
|
|
203
|
-
const source = `block "Revenue Query" {
|
|
204
|
-
domain = "finance"
|
|
205
|
-
type = "custom"
|
|
206
|
-
description = ""
|
|
207
|
-
owner = ""
|
|
208
|
-
tags = []
|
|
209
|
-
|
|
210
|
-
query = """
|
|
211
|
-
SELECT
|
|
212
|
-
@metric(total_revenue),
|
|
213
|
-
@dim(customer_type)
|
|
214
|
-
FROM orders
|
|
215
|
-
GROUP BY @dim(customer_type)
|
|
216
|
-
"""
|
|
217
|
-
}`;
|
|
218
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
219
|
-
expect(validation.valid).toBe(true);
|
|
220
|
-
expect(validation.executableSql).toContain('SUM(revenue) AS total_revenue');
|
|
221
|
-
expect(validation.executableSql).toContain('customer_type AS customer_type');
|
|
222
|
-
expect(validation.executableSql).toContain('GROUP BY customer_type');
|
|
223
|
-
});
|
|
224
|
-
it('returns a semantic validation error for unresolved refs in custom SQL', () => {
|
|
225
|
-
const source = `block "Broken Revenue Query" {
|
|
226
|
-
domain = "finance"
|
|
227
|
-
type = "custom"
|
|
228
|
-
description = ""
|
|
229
|
-
owner = ""
|
|
230
|
-
tags = []
|
|
231
|
-
|
|
232
|
-
query = """
|
|
233
|
-
SELECT @metric(missing_metric)
|
|
234
|
-
"""
|
|
235
|
-
}`;
|
|
236
|
-
const validation = validateBlockStudioSource(source, semanticLayer);
|
|
237
|
-
expect(validation.valid).toBe(false);
|
|
238
|
-
expect(validation.diagnostics.some((item) => item.code === 'semantic_ref' && item.message.includes('missing_metric'))).toBe(true);
|
|
239
|
-
});
|
|
240
|
-
});
|
|
241
|
-
//# sourceMappingURL=local-runtime.test.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"local-runtime.test.js","sourceRoot":"","sources":["../src/local-runtime.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EACL,oBAAoB,EACpB,0BAA0B,EAC1B,4BAA4B,EAC5B,0BAA0B,EAC1B,qBAAqB,EACrB,8BAA8B,EAC9B,aAAa,EACb,yBAAyB,GAC1B,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACnC,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAC3E,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AAEzD,MAAM,QAAQ,GAAa,EAAE,CAAC;AAE9B,SAAS,CAAC,GAAG,EAAE;IACb,OAAO,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC3B,MAAM,GAAG,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC;QAC3B,IAAI,GAAG;YAAE,MAAM,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACzD,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;IAC5C,EAAE,CAAC,kEAAkE,EAAE,GAAG,EAAE;QAC1E,MAAM,OAAO,GAAG,4BAA4B,CAC1C,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,EACxC,IAAI,KAAK,CAAC,0DAA0D,CAAC,CACtE,CAAC;QAEF,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,4CAA4C,CAAC,CAAC;QACxE,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,4BAA4B,OAAO,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;QAC/E,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;QAChD,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;IAC5C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;IACjE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC;QACnD,MAAM,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;IACtF,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,gCAAgC,EAAE,GAAG,EAAE;IAC9C,EAAE,CAAC,sEAAsE,EAAE,GAAG,EAAE;QAC9E,MAAM,GAAG,GAAG,mDAAmD,CAAC;QAChE,MAAM,QAAQ,GAAG,8BAA8B,CAAC,GAAG,EAAE,mBAAmB,CAAC,CAAC;QAE1E,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,mEAAmE,CAAC,CAAC;IAC7F,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,GAAG,GAAG,sCAAsC,CAAC;QACnD,MAAM,CAAC,8BAA8B,CAAC,GAAG,EAAE,mBAAmB,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC7E,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;IAC1C,EAAE,CAAC,iEAAiE,EAAE,GAAG,EAAE;QACzE,MAAM,CAAC,0BAA0B,CAC/B,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,oBAAoB,EAAE,EACpD,mBAAmB,CACpB,CAAC,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,oCAAoC,EAAE,CAAC,CAAC;IACnF,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;IACrC,EAAE,CAAC,qDAAqD,EAAE,GAAG,EAAE;QAC7D,MAAM,QAAQ,GAAG,qBAAqB,CACpC,mDAAmD,EACnD,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,EACxC,mBAAmB,EACnB,EAAE,OAAO,EAAE,QAAQ,EAAE,CACtB,CAAC;QAEF,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAC;QAC9E,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,mEAAmE,CAAC,CAAC;IACjG,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,+BAA+B,EAAE,GAAG,EAAE;IAC7C,EAAE,CAAC,qFAAqF,EAAE,GAAG,EAAE;QAC7F,MAAM,WAAW,GAAG,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,sBAAsB,CAAC,CAAC,CAAC;QACxE,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC3B,aAAa,CAAC,IAAI,CAAC,WAAW,EAAE,iBAAiB,CAAC,EAAE,MAAM,CAAC,CAAC;QAE5D,MAAM,OAAO,GAAG,oBAAoB,CAAC,WAAW,EAAE;YAChD,IAAI,EAAE,iBAAiB;YACvB,MAAM,EAAE,SAAS;YACjB,OAAO,EAAE,kDAAkD;YAC3D,WAAW,EAAE,uBAAuB;YACpC,IAAI,EAAE,CAAC,SAAS,EAAE,MAAM,CAAC;SAC1B,CAAC,CAAC;QAEH,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;QAChE,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,oDAAoD,CAAC,CAAC;QACzF,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,OAAO,CAAC,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;QAEnG,MAAM,SAAS,GAAG,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,OAAO,CAAC,aAAa,CAAC,EAAE,OAAO,CAAC,CAAC;QAClF,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QAC7C,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAChD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;QACjD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QACnD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;QAC9C,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gEAAgE,EAAE,GAAG,EAAE;QACxE,MAAM,WAAW,GAAG,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,wBAAwB,CAAC,CAAC,CAAC;QAC1E,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC3B,aAAa,CAAC,IAAI,CAAC,WAAW,EAAE,iBAAiB,CAAC,EAAE,MAAM,CAAC,CAAC;QAE5D,MAAM,OAAO,GAAG,0BAA0B,CAAC,WAAW,EAAE;YACtD,IAAI,EAAE,mBAAmB;YACzB,MAAM,EAAE,SAAS;YACjB,WAAW,EAAE,uBAAuB;YACpC,KAAK,EAAE,mBAAmB;YAC1B,IAAI,EAAE,CAAC,SAAS,CAAC;YACjB,OAAO,EAAE,CAAC,eAAe,CAAC;YAC1B,UAAU,EAAE,CAAC,eAAe,CAAC;YAC7B,aAAa,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,WAAW,EAAE,OAAO,EAAE;YAC3D,KAAK,EAAE,MAAM;YACb,SAAS,EAAE,UAAU;YACrB,GAAG,EAAE,UAAU;YACf,MAAM,EAAE,CAAC,kBAAkB,CAAC;YAC5B,QAAQ,EAAE,KAAK;SAChB,CAAC,CAAC;QAEH,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;QAClE,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;QACvD,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,0BAA0B,CAAC,CAAC;QAE9D,MAAM,SAAS,GAAG,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,OAAO,CAAC,aAAa,CAAC,EAAE,OAAO,CAAC,CAAC;QAClF,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QAC7C,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAChD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAChD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;QACjD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QACnD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;QACjD,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;IACzC,MAAM,aAAa,GAAG,IAAI,aAAa,CAAC;QACtC,OAAO,EAAE;YACP;gBACE,IAAI,EAAE,eAAe;gBACrB,KAAK,EAAE,eAAe;gBACtB,WAAW,EAAE,gBAAgB;gBAC7B,MAAM,EAAE,SAAS;gBACjB,GAAG,EAAE,cAAc;gBACnB,IAAI,EAAE,KAAK;gBACX,KAAK,EAAE,QAAQ;gBACf,IAAI,EAAE,EAAE;aACT;SACF;QACD,UAAU,EAAE;YACV;gBACE,IAAI,EAAE,eAAe;gBACrB,KAAK,EAAE,eAAe;gBACtB,WAAW,EAAE,yBAAyB;gBACtC,MAAM,EAAE,SAAS;gBACjB,GAAG,EAAE,eAAe;gBACpB,IAAI,EAAE,QAAQ;gBACd,KAAK,EAAE,QAAQ;gBACf,IAAI,EAAE,EAAE;aACT;SACF;QACD,WAAW,EAAE,EAAE;KAChB,CAAC,CAAC;IAEH,EAAE,CAAC,wEAAwE,EAAE,GAAG,EAAE;QAChF,MAAM,MAAM,GAAG;;;;;;;;EAQjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,+BAA+B,CAAC,CAAC;QAC5E,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,gCAAgC,CAAC,CAAC;QAC7E,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;IACvE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4EAA4E,EAAE,GAAG,EAAE;QACpF,MAAM,MAAM,GAAG;;;;;;;EAOjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,QAAQ,EAAE,CAAC;QAC5C,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,KAAK,yBAAyB,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpG,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4DAA4D,EAAE,GAAG,EAAE;QACpE,MAAM,MAAM,GAAG;;;;;;;;EAQjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,KAAK,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACvI,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kDAAkD,EAAE,GAAG,EAAE;QAC1D,MAAM,MAAM,GAAG;;;;;;;;;;;EAWjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iEAAiE,EAAE,GAAG,EAAE;QACzE,MAAM,MAAM,GAAG;;;;;;;;;;;;;;EAcjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,+BAA+B,CAAC,CAAC;QAC5E,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,gCAAgC,CAAC,CAAC;QAC7E,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;IACvE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uEAAuE,EAAE,GAAG,EAAE;QAC/E,MAAM,MAAM,GAAG;;;;;;;;;;EAUjB,CAAC;QAEC,MAAM,UAAU,GAAG,yBAAyB,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAEpE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,KAAK,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpI,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
package/dist/open-browser.d.ts
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"open-browser.d.ts","sourceRoot":"","sources":["../src/open-browser.ts"],"names":[],"mappings":"AAEA,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,MAAM,EAAE,UAAU,EAAE,OAAO,GAAG,IAAI,CAevE"}
|
package/dist/open-browser.js
DELETED
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
import { spawn } from 'node:child_process';
|
|
2
|
-
export function maybeOpenBrowser(url, shouldOpen) {
|
|
3
|
-
if (!shouldOpen)
|
|
4
|
-
return;
|
|
5
|
-
const command = browserCommand();
|
|
6
|
-
if (!command)
|
|
7
|
-
return;
|
|
8
|
-
try {
|
|
9
|
-
const child = spawn(command.bin, [...command.args, url], {
|
|
10
|
-
detached: true,
|
|
11
|
-
stdio: 'ignore',
|
|
12
|
-
});
|
|
13
|
-
child.unref();
|
|
14
|
-
}
|
|
15
|
-
catch {
|
|
16
|
-
// best-effort only; keep CLI flow quiet if no browser opener exists
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
function browserCommand() {
|
|
20
|
-
switch (process.platform) {
|
|
21
|
-
case 'darwin':
|
|
22
|
-
return { bin: 'open', args: [] };
|
|
23
|
-
case 'win32':
|
|
24
|
-
return { bin: 'cmd', args: ['/c', 'start', ''] };
|
|
25
|
-
default:
|
|
26
|
-
return { bin: 'xdg-open', args: [] };
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
//# sourceMappingURL=open-browser.js.map
|
package/dist/open-browser.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"open-browser.js","sourceRoot":"","sources":["../src/open-browser.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAE3C,MAAM,UAAU,gBAAgB,CAAC,GAAW,EAAE,UAAmB;IAC/D,IAAI,CAAC,UAAU;QAAE,OAAO;IAExB,MAAM,OAAO,GAAG,cAAc,EAAE,CAAC;IACjC,IAAI,CAAC,OAAO;QAAE,OAAO;IAErB,IAAI,CAAC;QACH,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,GAAG,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE;YACvD,QAAQ,EAAE,IAAI;YACd,KAAK,EAAE,QAAQ;SAChB,CAAC,CAAC;QACH,KAAK,CAAC,KAAK,EAAE,CAAC;IAChB,CAAC;IAAC,MAAM,CAAC;QACP,oEAAoE;IACtE,CAAC;AACH,CAAC;AAED,SAAS,cAAc;IACrB,QAAQ,OAAO,CAAC,QAAQ,EAAE,CAAC;QACzB,KAAK,QAAQ;YACX,OAAO,EAAE,GAAG,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;QACnC,KAAK,OAAO;YACV,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,IAAI,EAAE,OAAO,EAAE,EAAE,CAAC,EAAE,CAAC;QACnD;YACE,OAAO,EAAE,GAAG,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;IACzC,CAAC;AACH,CAAC"}
|
|
@@ -1,5 +0,0 @@
|
|
|
1
|
-
import type { AlertIR } from '@duckcodeailabs/dql-compiler';
|
|
2
|
-
import type { QueryExecutor, ConnectionConfig } from '@duckcodeailabs/dql-connectors';
|
|
3
|
-
import type { AlertEvaluation } from './types.js';
|
|
4
|
-
export declare function evaluateAlerts(alerts: AlertIR[], executor: QueryExecutor, connection: ConnectionConfig): Promise<AlertEvaluation[]>;
|
|
5
|
-
//# sourceMappingURL=alerts.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"alerts.d.ts","sourceRoot":"","sources":["../../src/schedule/alerts.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,KAAK,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,gCAAgC,CAAC;AACtF,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAElD,wBAAsB,cAAc,CAClC,MAAM,EAAE,OAAO,EAAE,EACjB,QAAQ,EAAE,aAAa,EACvB,UAAU,EAAE,gBAAgB,GAC3B,OAAO,CAAC,eAAe,EAAE,CAAC,CA+B5B"}
|
package/dist/schedule/alerts.js
DELETED
|
@@ -1,54 +0,0 @@
|
|
|
1
|
-
export async function evaluateAlerts(alerts, executor, connection) {
|
|
2
|
-
const results = [];
|
|
3
|
-
for (const alert of alerts) {
|
|
4
|
-
try {
|
|
5
|
-
const result = await executor.executeQuery(alert.conditionSQL, [], {}, connection);
|
|
6
|
-
const firstRow = result.rows[0] ?? {};
|
|
7
|
-
const firstCol = Object.values(firstRow)[0];
|
|
8
|
-
const observedValue = toNumber(firstCol);
|
|
9
|
-
if (observedValue === null) {
|
|
10
|
-
results.push({
|
|
11
|
-
alert,
|
|
12
|
-
breached: false,
|
|
13
|
-
reason: 'condition SQL did not return a numeric first column',
|
|
14
|
-
});
|
|
15
|
-
continue;
|
|
16
|
-
}
|
|
17
|
-
const breached = compareThreshold(observedValue, alert.operator ?? '>', alert.threshold ?? 0);
|
|
18
|
-
results.push({ alert, breached, observedValue });
|
|
19
|
-
}
|
|
20
|
-
catch (err) {
|
|
21
|
-
results.push({
|
|
22
|
-
alert,
|
|
23
|
-
breached: false,
|
|
24
|
-
error: err instanceof Error ? err.message : String(err),
|
|
25
|
-
});
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
return results;
|
|
29
|
-
}
|
|
30
|
-
function compareThreshold(value, op, threshold) {
|
|
31
|
-
switch (op) {
|
|
32
|
-
case '>': return value > threshold;
|
|
33
|
-
case '<': return value < threshold;
|
|
34
|
-
case '>=': return value >= threshold;
|
|
35
|
-
case '<=': return value <= threshold;
|
|
36
|
-
case '==': return value === threshold;
|
|
37
|
-
case '!=': return value !== threshold;
|
|
38
|
-
default: return value > threshold;
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
function toNumber(v) {
|
|
42
|
-
if (typeof v === 'number' && Number.isFinite(v))
|
|
43
|
-
return v;
|
|
44
|
-
if (typeof v === 'bigint')
|
|
45
|
-
return Number(v);
|
|
46
|
-
if (typeof v === 'string') {
|
|
47
|
-
const n = Number(v);
|
|
48
|
-
return Number.isFinite(n) ? n : null;
|
|
49
|
-
}
|
|
50
|
-
if (typeof v === 'boolean')
|
|
51
|
-
return v ? 1 : 0;
|
|
52
|
-
return null;
|
|
53
|
-
}
|
|
54
|
-
//# sourceMappingURL=alerts.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"alerts.js","sourceRoot":"","sources":["../../src/schedule/alerts.ts"],"names":[],"mappings":"AAIA,MAAM,CAAC,KAAK,UAAU,cAAc,CAClC,MAAiB,EACjB,QAAuB,EACvB,UAA4B;IAE5B,MAAM,OAAO,GAAsB,EAAE,CAAC;IAEtC,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,YAAY,CAAC,KAAK,CAAC,YAAY,EAAE,EAAE,EAAE,EAAE,EAAE,UAAU,CAAC,CAAC;YACnF,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YACtC,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YAC5C,MAAM,aAAa,GAAG,QAAQ,CAAC,QAAQ,CAAC,CAAC;YAEzC,IAAI,aAAa,KAAK,IAAI,EAAE,CAAC;gBAC3B,OAAO,CAAC,IAAI,CAAC;oBACX,KAAK;oBACL,QAAQ,EAAE,KAAK;oBACf,MAAM,EAAE,qDAAqD;iBAC9D,CAAC,CAAC;gBACH,SAAS;YACX,CAAC;YAED,MAAM,QAAQ,GAAG,gBAAgB,CAAC,aAAa,EAAE,KAAK,CAAC,QAAQ,IAAI,GAAG,EAAE,KAAK,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC;YAC9F,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC,CAAC;QACnD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,IAAI,CAAC;gBACX,KAAK;gBACL,QAAQ,EAAE,KAAK;gBACf,KAAK,EAAE,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC;aACxD,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC;AAED,SAAS,gBAAgB,CAAC,KAAa,EAAE,EAAuB,EAAE,SAAiB;IACjF,QAAQ,EAAE,EAAE,CAAC;QACX,KAAK,GAAG,CAAC,CAAC,OAAO,KAAK,GAAG,SAAS,CAAC;QACnC,KAAK,GAAG,CAAC,CAAC,OAAO,KAAK,GAAG,SAAS,CAAC;QACnC,KAAK,IAAI,CAAC,CAAC,OAAO,KAAK,IAAI,SAAS,CAAC;QACrC,KAAK,IAAI,CAAC,CAAC,OAAO,KAAK,IAAI,SAAS,CAAC;QACrC,KAAK,IAAI,CAAC,CAAC,OAAO,KAAK,KAAK,SAAS,CAAC;QACtC,KAAK,IAAI,CAAC,CAAC,OAAO,KAAK,KAAK,SAAS,CAAC;QACtC,OAAO,CAAC,CAAC,OAAO,KAAK,GAAG,SAAS,CAAC;IACpC,CAAC;AACH,CAAC;AAED,SAAS,QAAQ,CAAC,CAAU;IAC1B,IAAI,OAAO,CAAC,KAAK,QAAQ,IAAI,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC;QAAE,OAAO,CAAC,CAAC;IAC1D,IAAI,OAAO,CAAC,KAAK,QAAQ;QAAE,OAAO,MAAM,CAAC,CAAC,CAAC,CAAC;IAC5C,IAAI,OAAO,CAAC,KAAK,QAAQ,EAAE,CAAC;QAC1B,MAAM,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACpB,OAAO,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IACvC,CAAC;IACD,IAAI,OAAO,CAAC,KAAK,SAAS;QAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7C,OAAO,IAAI,CAAC;AACd,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"discovery.d.ts","sourceRoot":"","sources":["../../src/schedule/discovery.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAEjD,wBAAgB,uBAAuB,CAAC,WAAW,EAAE,MAAM,GAAG,cAAc,EAAE,CA2B7E;AAED,wBAAgB,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAG5E"}
|
|
@@ -1,36 +0,0 @@
|
|
|
1
|
-
import { readFileSync } from 'node:fs';
|
|
2
|
-
import { relative } from 'node:path';
|
|
3
|
-
import { compile } from '@duckcodeailabs/dql-compiler';
|
|
4
|
-
import { collectInputFiles } from '@duckcodeailabs/dql-core';
|
|
5
|
-
export function discoverScheduledBlocks(projectRoot) {
|
|
6
|
-
const files = collectInputFiles({ projectRoot }).filter((f) => f.endsWith('.dql'));
|
|
7
|
-
const out = [];
|
|
8
|
-
for (const path of files) {
|
|
9
|
-
let source;
|
|
10
|
-
try {
|
|
11
|
-
source = readFileSync(path, 'utf-8');
|
|
12
|
-
}
|
|
13
|
-
catch {
|
|
14
|
-
continue;
|
|
15
|
-
}
|
|
16
|
-
const result = compile(source, { file: path });
|
|
17
|
-
for (const dashboard of result.dashboards) {
|
|
18
|
-
const meta = dashboard.metadata;
|
|
19
|
-
if (!meta.schedule?.cron)
|
|
20
|
-
continue;
|
|
21
|
-
out.push({
|
|
22
|
-
path,
|
|
23
|
-
name: deriveBlockName(projectRoot, path),
|
|
24
|
-
schedule: meta.schedule,
|
|
25
|
-
notifications: meta.notifications ?? [],
|
|
26
|
-
alerts: meta.alerts ?? [],
|
|
27
|
-
});
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
return out;
|
|
31
|
-
}
|
|
32
|
-
export function deriveBlockName(projectRoot, absPath) {
|
|
33
|
-
const rel = relative(projectRoot, absPath);
|
|
34
|
-
return rel.replace(/\.dql$/, '').replace(/\\/g, '/');
|
|
35
|
-
}
|
|
36
|
-
//# sourceMappingURL=discovery.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"discovery.js","sourceRoot":"","sources":["../../src/schedule/discovery.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,8BAA8B,CAAC;AACvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAG7D,MAAM,UAAU,uBAAuB,CAAC,WAAmB;IACzD,MAAM,KAAK,GAAG,iBAAiB,CAAC,EAAE,WAAW,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;IACnF,MAAM,GAAG,GAAqB,EAAE,CAAC;IAEjC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,IAAI,MAAc,CAAC;QACnB,IAAI,CAAC;YACH,MAAM,GAAG,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QACvC,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;QAED,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;QAC/C,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YAC1C,MAAM,IAAI,GAAG,SAAS,CAAC,QAAQ,CAAC;YAChC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI;gBAAE,SAAS;YACnC,GAAG,CAAC,IAAI,CAAC;gBACP,IAAI;gBACJ,IAAI,EAAE,eAAe,CAAC,WAAW,EAAE,IAAI,CAAC;gBACxC,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,aAAa,EAAE,IAAI,CAAC,aAAa,IAAI,EAAE;gBACvC,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,EAAE;aAC1B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,WAAmB,EAAE,OAAe;IAClE,MAAM,GAAG,GAAG,QAAQ,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IAC3C,OAAO,GAAG,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;AACvD,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"email.d.ts","sourceRoot":"","sources":["../../../src/schedule/notifiers/email.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAmB,MAAM,aAAa,CAAC;AAE7D,wBAAgB,mBAAmB,IAAI,QAAQ,CA0C9C"}
|
|
@@ -1,76 +0,0 @@
|
|
|
1
|
-
export function createEmailNotifier() {
|
|
2
|
-
return {
|
|
3
|
-
type: 'email',
|
|
4
|
-
async send(recipients, payload) {
|
|
5
|
-
if (recipients.length === 0) {
|
|
6
|
-
return { delivered: false, error: 'no email recipients provided' };
|
|
7
|
-
}
|
|
8
|
-
const smtpUrl = process.env.DQL_SMTP_URL;
|
|
9
|
-
const from = process.env.DQL_SMTP_FROM ?? 'dql@localhost';
|
|
10
|
-
const subject = buildSubject(payload);
|
|
11
|
-
// Digest runs deliver the rendered HTML + markdown sibling as the email
|
|
12
|
-
// body; non-digest runs fall back to the legacy alert/query summary.
|
|
13
|
-
const textBody = payload.markdown ?? buildBody(payload);
|
|
14
|
-
const htmlBody = payload.html;
|
|
15
|
-
if (!smtpUrl) {
|
|
16
|
-
console.log(`[dql schedule] email stub — no DQL_SMTP_URL set`);
|
|
17
|
-
console.log(` to: ${recipients.join(', ')}`);
|
|
18
|
-
console.log(` subject: ${subject}`);
|
|
19
|
-
if (htmlBody)
|
|
20
|
-
console.log(` html: ${htmlBody.length} chars`);
|
|
21
|
-
console.log(textBody);
|
|
22
|
-
return { delivered: false, error: 'DQL_SMTP_URL not set (stub mode)' };
|
|
23
|
-
}
|
|
24
|
-
try {
|
|
25
|
-
// Dynamic import so nodemailer only loads when SMTP is actually configured.
|
|
26
|
-
const nodemailer = (await import('nodemailer'));
|
|
27
|
-
const transporter = nodemailer.createTransport(smtpUrl);
|
|
28
|
-
await transporter.sendMail({
|
|
29
|
-
from,
|
|
30
|
-
to: recipients.join(', '),
|
|
31
|
-
subject,
|
|
32
|
-
text: textBody,
|
|
33
|
-
...(htmlBody ? { html: htmlBody } : {}),
|
|
34
|
-
});
|
|
35
|
-
return { delivered: true };
|
|
36
|
-
}
|
|
37
|
-
catch (err) {
|
|
38
|
-
return { delivered: false, error: err instanceof Error ? err.message : String(err) };
|
|
39
|
-
}
|
|
40
|
-
},
|
|
41
|
-
};
|
|
42
|
-
}
|
|
43
|
-
function buildSubject(payload) {
|
|
44
|
-
const breached = payload.alerts.filter((a) => a.breached).length;
|
|
45
|
-
if (payload.markdown) {
|
|
46
|
-
return `[DQL Digest] ${payload.digestTitle ?? payload.block}`;
|
|
47
|
-
}
|
|
48
|
-
const prefix = breached > 0 ? `[DQL ALERT]` : `[DQL]`;
|
|
49
|
-
return `${prefix} ${payload.block} — ${payload.trigger} run`;
|
|
50
|
-
}
|
|
51
|
-
function buildBody(payload) {
|
|
52
|
-
const lines = [
|
|
53
|
-
`Block: ${payload.block}`,
|
|
54
|
-
`Path: ${payload.path}`,
|
|
55
|
-
`Started: ${payload.startedAt}`,
|
|
56
|
-
`Trigger: ${payload.trigger}`,
|
|
57
|
-
'',
|
|
58
|
-
];
|
|
59
|
-
if (payload.alerts.length > 0) {
|
|
60
|
-
lines.push('Alerts:');
|
|
61
|
-
for (const a of payload.alerts) {
|
|
62
|
-
lines.push(` ${a.breached ? '!' : '-'} ${a.alert.conditionSQL} ${a.alert.operator ?? '>'} ${a.alert.threshold ?? 0}` +
|
|
63
|
-
(a.observedValue !== undefined ? ` (observed ${a.observedValue})` : '') +
|
|
64
|
-
(a.error ? ` [error: ${a.error}]` : ''));
|
|
65
|
-
if (a.alert.message)
|
|
66
|
-
lines.push(` message: ${a.alert.message}`);
|
|
67
|
-
}
|
|
68
|
-
lines.push('');
|
|
69
|
-
}
|
|
70
|
-
lines.push('Queries:');
|
|
71
|
-
for (const q of payload.queries) {
|
|
72
|
-
lines.push(` ${q.chartId}: ${q.rowCount} rows in ${q.durationMs}ms${q.error ? ` [error: ${q.error}]` : ''}`);
|
|
73
|
-
}
|
|
74
|
-
return lines.join('\n');
|
|
75
|
-
}
|
|
76
|
-
//# sourceMappingURL=email.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"email.js","sourceRoot":"","sources":["../../../src/schedule/notifiers/email.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"file.d.ts","sourceRoot":"","sources":["../../../src/schedule/notifiers/file.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,QAAQ,EAAmB,MAAM,aAAa,CAAC;AAE7D,wBAAgB,kBAAkB,CAAC,WAAW,EAAE,MAAM,GAAG,QAAQ,CAkBhE"}
|
|
@@ -1,50 +0,0 @@
|
|
|
1
|
-
import { appendFileSync, existsSync, mkdirSync } from 'node:fs';
|
|
2
|
-
import { dirname, isAbsolute, join } from 'node:path';
|
|
3
|
-
export function createFileNotifier(projectRoot) {
|
|
4
|
-
return {
|
|
5
|
-
type: 'file',
|
|
6
|
-
async send(recipients, payload) {
|
|
7
|
-
const targets = recipients.length > 0 ? recipients : ['.dql/runs/notifications.log'];
|
|
8
|
-
try {
|
|
9
|
-
for (const target of targets) {
|
|
10
|
-
const abs = isAbsolute(target) ? target : join(projectRoot, target);
|
|
11
|
-
const dir = dirname(abs);
|
|
12
|
-
if (!existsSync(dir))
|
|
13
|
-
mkdirSync(dir, { recursive: true });
|
|
14
|
-
appendFileSync(abs, formatLine(payload) + '\n', 'utf-8');
|
|
15
|
-
}
|
|
16
|
-
return { delivered: true };
|
|
17
|
-
}
|
|
18
|
-
catch (err) {
|
|
19
|
-
return { delivered: false, error: err instanceof Error ? err.message : String(err) };
|
|
20
|
-
}
|
|
21
|
-
},
|
|
22
|
-
};
|
|
23
|
-
}
|
|
24
|
-
function formatLine(payload) {
|
|
25
|
-
const breached = payload.alerts.filter((a) => a.breached).length;
|
|
26
|
-
return JSON.stringify({
|
|
27
|
-
time: payload.startedAt,
|
|
28
|
-
block: payload.block,
|
|
29
|
-
trigger: payload.trigger,
|
|
30
|
-
breached,
|
|
31
|
-
alerts: payload.alerts.map((a) => ({
|
|
32
|
-
condition: a.alert.conditionSQL,
|
|
33
|
-
operator: a.alert.operator,
|
|
34
|
-
threshold: a.alert.threshold,
|
|
35
|
-
observed: a.observedValue,
|
|
36
|
-
breached: a.breached,
|
|
37
|
-
message: a.alert.message,
|
|
38
|
-
})),
|
|
39
|
-
...(payload.markdown !== undefined
|
|
40
|
-
? {
|
|
41
|
-
digest: {
|
|
42
|
-
title: payload.digestTitle ?? payload.block,
|
|
43
|
-
markdown: payload.markdown,
|
|
44
|
-
diagnostics: payload.digestDiagnostics ?? [],
|
|
45
|
-
},
|
|
46
|
-
}
|
|
47
|
-
: {}),
|
|
48
|
-
});
|
|
49
|
-
}
|
|
50
|
-
//# sourceMappingURL=file.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"file.js","sourceRoot":"","sources":["../../../src/schedule/notifiers/file.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AAChE,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGtD,MAAM,UAAU,kBAAkB,CAAC,WAAmB;IACpD,OAAO;QACL,IAAI,EAAE,MAAM;QACZ,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,OAAO;YAC5B,MAAM,OAAO,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,6BAA6B,CAAC,CAAC;YACrF,IAAI,CAAC;gBACH,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;oBAC7B,MAAM,GAAG,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;oBACpE,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;oBACzB,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;wBAAE,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;oBAC1D,cAAc,CAAC,GAAG,EAAE,UAAU,CAAC,OAAO,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC;gBAC3D,CAAC;gBACD,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;YAC7B,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC;YACvF,CAAC;QACH,CAAC;KACF,CAAC;AACJ,CAAC;AAED,SAAS,UAAU,CAAC,OAAwB;IAC1C,MAAM,QAAQ,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC;IACjE,OAAO,IAAI,CAAC,SAAS,CAAC;QACpB,IAAI,EAAE,OAAO,CAAC,SAAS;QACvB,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,OAAO,EAAE,OAAO,CAAC,OAAO;QACxB,QAAQ;QACR,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACjC,SAAS,EAAE,CAAC,CAAC,KAAK,CAAC,YAAY;YAC/B,QAAQ,EAAE,CAAC,CAAC,KAAK,CAAC,QAAQ;YAC1B,SAAS,EAAE,CAAC,CAAC,KAAK,CAAC,SAAS;YAC5B,QAAQ,EAAE,CAAC,CAAC,aAAa;YACzB,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,OAAO,EAAE,CAAC,CAAC,KAAK,CAAC,OAAO;SACzB,CAAC,CAAC;QACH,GAAG,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS;YAChC,CAAC,CAAC;gBACE,MAAM,EAAE;oBACN,KAAK,EAAE,OAAO,CAAC,WAAW,IAAI,OAAO,CAAC,KAAK;oBAC3C,QAAQ,EAAE,OAAO,CAAC,QAAQ;oBAC1B,WAAW,EAAE,OAAO,CAAC,iBAAiB,IAAI,EAAE;iBAC7C;aACF;YACH,CAAC,CAAC,EAAE,CAAC;KACR,CAAC,CAAC;AACL,CAAC"}
|
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
import type { NotificationIR } from '@duckcodeailabs/dql-compiler';
|
|
2
|
-
import type { NotifierPayload } from '../types.js';
|
|
3
|
-
export interface NotificationDispatchResult {
|
|
4
|
-
type: string;
|
|
5
|
-
recipients: string[];
|
|
6
|
-
delivered: boolean;
|
|
7
|
-
error?: string;
|
|
8
|
-
}
|
|
9
|
-
export declare function dispatchNotifications(notifications: NotificationIR[], payload: NotifierPayload, projectRoot: string): Promise<NotificationDispatchResult[]>;
|
|
10
|
-
//# sourceMappingURL=index.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/schedule/notifiers/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,KAAK,EAAY,eAAe,EAAE,MAAM,aAAa,CAAC;AAK7D,MAAM,WAAW,0BAA0B;IACzC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,OAAO,CAAC;IACnB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,wBAAsB,qBAAqB,CACzC,aAAa,EAAE,cAAc,EAAE,EAC/B,OAAO,EAAE,eAAe,EACxB,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC,0BAA0B,EAAE,CAAC,CA+BvC"}
|
|
@@ -1,33 +0,0 @@
|
|
|
1
|
-
import { createEmailNotifier } from './email.js';
|
|
2
|
-
import { createFileNotifier } from './file.js';
|
|
3
|
-
import { createSlackNotifier } from './slack.js';
|
|
4
|
-
export async function dispatchNotifications(notifications, payload, projectRoot) {
|
|
5
|
-
const notifiers = {
|
|
6
|
-
email: createEmailNotifier(),
|
|
7
|
-
slack: createSlackNotifier(),
|
|
8
|
-
file: createFileNotifier(projectRoot),
|
|
9
|
-
};
|
|
10
|
-
const out = [];
|
|
11
|
-
for (const n of notifications) {
|
|
12
|
-
const notifier = notifiers[n.type];
|
|
13
|
-
if (!notifier) {
|
|
14
|
-
out.push({
|
|
15
|
-
type: n.type,
|
|
16
|
-
recipients: n.recipients,
|
|
17
|
-
delivered: false,
|
|
18
|
-
error: `no notifier registered for type "${n.type}"`,
|
|
19
|
-
});
|
|
20
|
-
continue;
|
|
21
|
-
}
|
|
22
|
-
const result = await notifier.send(n.recipients, payload);
|
|
23
|
-
out.push({ type: n.type, recipients: n.recipients, ...result });
|
|
24
|
-
}
|
|
25
|
-
// Always append to file log for audit trail, even if no explicit file target.
|
|
26
|
-
const fileLog = notifiers.file;
|
|
27
|
-
const auditResult = await fileLog.send([], payload);
|
|
28
|
-
if (auditResult.delivered || auditResult.error) {
|
|
29
|
-
out.push({ type: 'file', recipients: ['.dql/runs/notifications.log'], ...auditResult });
|
|
30
|
-
}
|
|
31
|
-
return out;
|
|
32
|
-
}
|
|
33
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/schedule/notifiers/index.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,mBAAmB,EAAE,MAAM,YAAY,CAAC;AACjD,OAAO,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AAC/C,OAAO,EAAE,mBAAmB,EAAE,MAAM,YAAY,CAAC;AASjD,MAAM,CAAC,KAAK,UAAU,qBAAqB,CACzC,aAA+B,EAC/B,OAAwB,EACxB,WAAmB;IAEnB,MAAM,SAAS,GAA6B;QAC1C,KAAK,EAAE,mBAAmB,EAAE;QAC5B,KAAK,EAAE,mBAAmB,EAAE;QAC5B,IAAI,EAAE,kBAAkB,CAAC,WAAW,CAAC;KACtC,CAAC;IAEF,MAAM,GAAG,GAAiC,EAAE,CAAC;IAC7C,KAAK,MAAM,CAAC,IAAI,aAAa,EAAE,CAAC;QAC9B,MAAM,QAAQ,GAAG,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACnC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,GAAG,CAAC,IAAI,CAAC;gBACP,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,SAAS,EAAE,KAAK;gBAChB,KAAK,EAAE,oCAAoC,CAAC,CAAC,IAAI,GAAG;aACrD,CAAC,CAAC;YACH,SAAS;QACX,CAAC;QACD,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAC1D,GAAG,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,UAAU,EAAE,CAAC,CAAC,UAAU,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC;IAClE,CAAC;IAED,8EAA8E;IAC9E,MAAM,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC;IAC/B,MAAM,WAAW,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,OAAO,CAAC,CAAC;IACpD,IAAI,WAAW,CAAC,SAAS,IAAI,WAAW,CAAC,KAAK,EAAE,CAAC;QAC/C,GAAG,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,UAAU,EAAE,CAAC,6BAA6B,CAAC,EAAE,GAAG,WAAW,EAAE,CAAC,CAAC;IAC1F,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC"}
|