@duckcodeailabs/dql-cli 1.0.3 → 1.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/args.d.ts +1 -0
- package/dist/args.d.ts.map +1 -1
- package/dist/args.js +4 -0
- package/dist/args.js.map +1 -1
- package/dist/assets/dql-notebook/assets/index-Cscl1A2H.js +628 -0
- package/dist/assets/dql-notebook/index.html +1 -1
- package/dist/commands/app.d.ts +3 -0
- package/dist/commands/app.d.ts.map +1 -0
- package/dist/commands/app.js +191 -0
- package/dist/commands/app.js.map +1 -0
- package/dist/commands/build.d.ts.map +1 -1
- package/dist/commands/build.js +30 -2
- package/dist/commands/build.js.map +1 -1
- package/dist/commands/diff.d.ts +1 -1
- package/dist/commands/diff.d.ts.map +1 -1
- package/dist/commands/diff.js +38 -9
- package/dist/commands/diff.js.map +1 -1
- package/dist/commands/fmt.d.ts.map +1 -1
- package/dist/commands/fmt.js +27 -8
- package/dist/commands/fmt.js.map +1 -1
- package/dist/commands/init.d.ts.map +1 -1
- package/dist/commands/init.js +1 -1
- package/dist/commands/init.js.map +1 -1
- package/dist/commands/mcp.d.ts +7 -0
- package/dist/commands/mcp.d.ts.map +1 -0
- package/dist/commands/mcp.js +16 -0
- package/dist/commands/mcp.js.map +1 -0
- package/dist/commands/migrate.d.ts +1 -0
- package/dist/commands/migrate.d.ts.map +1 -1
- package/dist/commands/migrate.js +86 -0
- package/dist/commands/migrate.js.map +1 -1
- package/dist/commands/schedule.d.ts +3 -0
- package/dist/commands/schedule.d.ts.map +1 -0
- package/dist/commands/schedule.js +132 -0
- package/dist/commands/schedule.js.map +1 -0
- package/dist/digest.d.ts +10 -0
- package/dist/digest.d.ts.map +1 -0
- package/dist/digest.js +83 -0
- package/dist/digest.js.map +1 -0
- package/dist/git-service.d.ts +17 -0
- package/dist/git-service.d.ts.map +1 -0
- package/dist/git-service.js +54 -0
- package/dist/git-service.js.map +1 -0
- package/dist/index.js +20 -3
- package/dist/index.js.map +1 -1
- package/dist/llm/claude-agent-sdk.d.ts +3 -0
- package/dist/llm/claude-agent-sdk.d.ts.map +1 -0
- package/dist/llm/claude-agent-sdk.js +174 -0
- package/dist/llm/claude-agent-sdk.js.map +1 -0
- package/dist/llm/claude-code.d.ts +8 -0
- package/dist/llm/claude-code.d.ts.map +1 -0
- package/dist/llm/claude-code.js +171 -0
- package/dist/llm/claude-code.js.map +1 -0
- package/dist/llm/index.d.ts +4 -0
- package/dist/llm/index.d.ts.map +1 -0
- package/dist/llm/index.js +10 -0
- package/dist/llm/index.js.map +1 -0
- package/dist/llm/tools.d.ts +9 -0
- package/dist/llm/tools.d.ts.map +1 -0
- package/dist/llm/tools.js +112 -0
- package/dist/llm/tools.js.map +1 -0
- package/dist/llm/types.d.ts +64 -0
- package/dist/llm/types.d.ts.map +1 -0
- package/dist/llm/types.js +2 -0
- package/dist/llm/types.js.map +1 -0
- package/dist/local-runtime.d.ts +6 -0
- package/dist/local-runtime.d.ts.map +1 -1
- package/dist/local-runtime.js +124 -8
- package/dist/local-runtime.js.map +1 -1
- package/dist/schedule/alerts.d.ts +5 -0
- package/dist/schedule/alerts.d.ts.map +1 -0
- package/dist/schedule/alerts.js +54 -0
- package/dist/schedule/alerts.js.map +1 -0
- package/dist/schedule/discovery.d.ts +4 -0
- package/dist/schedule/discovery.d.ts.map +1 -0
- package/dist/schedule/discovery.js +36 -0
- package/dist/schedule/discovery.js.map +1 -0
- package/dist/schedule/notifiers/email.d.ts +3 -0
- package/dist/schedule/notifiers/email.d.ts.map +1 -0
- package/dist/schedule/notifiers/email.js +76 -0
- package/dist/schedule/notifiers/email.js.map +1 -0
- package/dist/schedule/notifiers/file.d.ts +3 -0
- package/dist/schedule/notifiers/file.d.ts.map +1 -0
- package/dist/schedule/notifiers/file.js +50 -0
- package/dist/schedule/notifiers/file.js.map +1 -0
- package/dist/schedule/notifiers/index.d.ts +10 -0
- package/dist/schedule/notifiers/index.d.ts.map +1 -0
- package/dist/schedule/notifiers/index.js +33 -0
- package/dist/schedule/notifiers/index.js.map +1 -0
- package/dist/schedule/notifiers/slack.d.ts +3 -0
- package/dist/schedule/notifiers/slack.d.ts.map +1 -0
- package/dist/schedule/notifiers/slack.js +58 -0
- package/dist/schedule/notifiers/slack.js.map +1 -0
- package/dist/schedule/runner.d.ts +11 -0
- package/dist/schedule/runner.d.ts.map +1 -0
- package/dist/schedule/runner.js +109 -0
- package/dist/schedule/runner.js.map +1 -0
- package/dist/schedule/runs.d.ts +5 -0
- package/dist/schedule/runs.d.ts.map +1 -0
- package/dist/schedule/runs.js +40 -0
- package/dist/schedule/runs.js.map +1 -0
- package/dist/schedule/service.d.ts +12 -0
- package/dist/schedule/service.d.ts.map +1 -0
- package/dist/schedule/service.js +49 -0
- package/dist/schedule/service.js.map +1 -0
- package/dist/schedule/types.d.ts +64 -0
- package/dist/schedule/types.d.ts.map +1 -0
- package/dist/schedule/types.js +2 -0
- package/dist/schedule/types.js.map +1 -0
- package/package.json +17 -10
- package/dist/assets/dql-notebook/assets/index-BJbWzCsK.js +0 -622
|
@@ -0,0 +1,5 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,54 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,36 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,76 @@
|
|
|
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: true };
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"email.js","sourceRoot":"","sources":["../../../src/schedule/notifiers/email.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,50 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
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
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../../src/schedule/notifiers/slack.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAmB,MAAM,aAAa,CAAC;AAE7D,wBAAgB,mBAAmB,IAAI,QAAQ,CAyD9C"}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
export function createSlackNotifier() {
|
|
2
|
+
return {
|
|
3
|
+
type: 'slack',
|
|
4
|
+
async send(recipients, payload) {
|
|
5
|
+
const webhook = process.env.DQL_SLACK_WEBHOOK;
|
|
6
|
+
if (!webhook) {
|
|
7
|
+
return {
|
|
8
|
+
delivered: false,
|
|
9
|
+
error: 'DQL_SLACK_WEBHOOK environment variable is not set',
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
const breached = payload.alerts.filter((a) => a.breached);
|
|
13
|
+
const hasDigest = Boolean(payload.markdown);
|
|
14
|
+
// Skip noise: no digest to deliver AND no alert breached.
|
|
15
|
+
if (!hasDigest && breached.length === 0) {
|
|
16
|
+
return { delivered: true };
|
|
17
|
+
}
|
|
18
|
+
const channels = recipients.length > 0 ? recipients : ['#default'];
|
|
19
|
+
const text = hasDigest
|
|
20
|
+
? [
|
|
21
|
+
`*[DQL Digest] ${payload.digestTitle ?? payload.block}*`,
|
|
22
|
+
`Generated: ${payload.startedAt} (${payload.trigger})`,
|
|
23
|
+
'',
|
|
24
|
+
truncateMarkdown(payload.markdown, 2500),
|
|
25
|
+
].join('\n')
|
|
26
|
+
: [
|
|
27
|
+
`*[DQL] ${payload.block}* — ${breached.length} alert(s) breached`,
|
|
28
|
+
`Triggered: ${payload.trigger} at ${payload.startedAt}`,
|
|
29
|
+
...breached.map((a) => `• \`${a.alert.conditionSQL}\` ${a.alert.operator ?? '>'} ${a.alert.threshold ?? 0} — observed \`${a.observedValue}\`${a.alert.message ? ` — ${a.alert.message}` : ''}`),
|
|
30
|
+
].join('\n');
|
|
31
|
+
try {
|
|
32
|
+
for (const channel of channels) {
|
|
33
|
+
const res = await fetch(webhook, {
|
|
34
|
+
method: 'POST',
|
|
35
|
+
headers: { 'Content-Type': 'application/json' },
|
|
36
|
+
body: JSON.stringify({ channel, text }),
|
|
37
|
+
});
|
|
38
|
+
if (!res.ok) {
|
|
39
|
+
return {
|
|
40
|
+
delivered: false,
|
|
41
|
+
error: `slack webhook returned ${res.status} ${res.statusText}`,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return { delivered: true };
|
|
46
|
+
}
|
|
47
|
+
catch (err) {
|
|
48
|
+
return { delivered: false, error: err instanceof Error ? err.message : String(err) };
|
|
49
|
+
}
|
|
50
|
+
},
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
function truncateMarkdown(md, limit) {
|
|
54
|
+
if (md.length <= limit)
|
|
55
|
+
return md;
|
|
56
|
+
return md.slice(0, limit).trimEnd() + '\n…';
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=slack.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slack.js","sourceRoot":"","sources":["../../../src/schedule/notifiers/slack.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,mBAAmB;IACjC,OAAO;QACL,IAAI,EAAE,OAAO;QACb,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,OAAO;YAC5B,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC;YAC9C,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,OAAO;oBACL,SAAS,EAAE,KAAK;oBAChB,KAAK,EAAE,mDAAmD;iBAC3D,CAAC;YACJ,CAAC;YAED,MAAM,QAAQ,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YAC1D,MAAM,SAAS,GAAG,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAE5C,0DAA0D;YAC1D,IAAI,CAAC,SAAS,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACxC,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;YAC7B,CAAC;YAED,MAAM,QAAQ,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;YACnE,MAAM,IAAI,GAAG,SAAS;gBACpB,CAAC,CAAC;oBACE,iBAAiB,OAAO,CAAC,WAAW,IAAI,OAAO,CAAC,KAAK,GAAG;oBACxD,cAAc,OAAO,CAAC,SAAS,KAAK,OAAO,CAAC,OAAO,GAAG;oBACtD,EAAE;oBACF,gBAAgB,CAAC,OAAO,CAAC,QAAkB,EAAE,IAAI,CAAC;iBACnD,CAAC,IAAI,CAAC,IAAI,CAAC;gBACd,CAAC,CAAC;oBACE,UAAU,OAAO,CAAC,KAAK,OAAO,QAAQ,CAAC,MAAM,oBAAoB;oBACjE,cAAc,OAAO,CAAC,OAAO,OAAO,OAAO,CAAC,SAAS,EAAE;oBACvD,GAAG,QAAQ,CAAC,GAAG,CACb,CAAC,CAAC,EAAE,EAAE,CACJ,OAAO,CAAC,CAAC,KAAK,CAAC,YAAY,MAAM,CAAC,CAAC,KAAK,CAAC,QAAQ,IAAI,GAAG,IAAI,CAAC,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,iBAAiB,CAAC,CAAC,aAAa,KAAK,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CAC1K;iBACF,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAEjB,IAAI,CAAC;gBACH,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;oBAC/B,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE;wBAC/B,MAAM,EAAE,MAAM;wBACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;wBAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;qBACxC,CAAC,CAAC;oBACH,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;wBACZ,OAAO;4BACL,SAAS,EAAE,KAAK;4BAChB,KAAK,EAAE,0BAA0B,GAAG,CAAC,MAAM,IAAI,GAAG,CAAC,UAAU,EAAE;yBAChE,CAAC;oBACJ,CAAC;gBACH,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,gBAAgB,CAAC,EAAU,EAAE,KAAa;IACjD,IAAI,EAAE,CAAC,MAAM,IAAI,KAAK;QAAE,OAAO,EAAE,CAAC;IAClC,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,OAAO,EAAE,GAAG,KAAK,CAAC;AAC9C,CAAC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { QueryExecutor, ConnectionConfig } from '@duckcodeailabs/dql-connectors';
|
|
2
|
+
import type { RunRecord } from './types.js';
|
|
3
|
+
export interface RunOptions {
|
|
4
|
+
executor: QueryExecutor;
|
|
5
|
+
connection: ConnectionConfig;
|
|
6
|
+
projectRoot?: string;
|
|
7
|
+
trigger?: 'manual' | 'cron';
|
|
8
|
+
previewRows?: number;
|
|
9
|
+
}
|
|
10
|
+
export declare function runBlock(absPath: string, options: RunOptions): Promise<RunRecord>;
|
|
11
|
+
//# sourceMappingURL=runner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runner.d.ts","sourceRoot":"","sources":["../../src/schedule/runner.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,gCAAgC,CAAC;AAOtF,OAAO,KAAK,EAAmC,SAAS,EAAE,MAAM,YAAY,CAAC;AAE7E,MAAM,WAAW,UAAU;IACzB,QAAQ,EAAE,aAAa,CAAC;IACxB,UAAU,EAAE,gBAAgB,CAAC;IAC7B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,QAAQ,GAAG,MAAM,CAAC;IAC5B,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,wBAAsB,QAAQ,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,SAAS,CAAC,CAwHvF"}
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import { readFileSync } from 'node:fs';
|
|
2
|
+
import { compile } from '@duckcodeailabs/dql-compiler';
|
|
3
|
+
import { findProjectRoot, loadProjectConfig, prepareLocalExecution } from '../local-runtime.js';
|
|
4
|
+
import { isDigestOutput, runDigestBuild } from '../digest.js';
|
|
5
|
+
import { evaluateAlerts } from './alerts.js';
|
|
6
|
+
import { deriveBlockName } from './discovery.js';
|
|
7
|
+
import { dispatchNotifications } from './notifiers/index.js';
|
|
8
|
+
import { writeRunRecord } from './runs.js';
|
|
9
|
+
export async function runBlock(absPath, options) {
|
|
10
|
+
const projectRoot = options.projectRoot ?? findProjectRoot(process.cwd());
|
|
11
|
+
const projectConfig = loadProjectConfig(projectRoot);
|
|
12
|
+
const trigger = options.trigger ?? 'manual';
|
|
13
|
+
const previewRows = options.previewRows ?? 3;
|
|
14
|
+
const startedAt = new Date().toISOString();
|
|
15
|
+
const block = deriveBlockName(projectRoot, absPath);
|
|
16
|
+
let queries = [];
|
|
17
|
+
let alertResults = [];
|
|
18
|
+
let notifications = [];
|
|
19
|
+
let error;
|
|
20
|
+
try {
|
|
21
|
+
const source = readFileSync(absPath, 'utf-8');
|
|
22
|
+
const compiled = compile(source, { file: absPath });
|
|
23
|
+
if (compiled.errors.length > 0) {
|
|
24
|
+
error = compiled.errors.join('; ');
|
|
25
|
+
}
|
|
26
|
+
const dashboard = compiled.dashboards[0];
|
|
27
|
+
if (!dashboard) {
|
|
28
|
+
error = error ?? 'compile produced no dashboards';
|
|
29
|
+
}
|
|
30
|
+
else {
|
|
31
|
+
// Execute each chart query.
|
|
32
|
+
for (const q of dashboard.metadata.queries) {
|
|
33
|
+
const t0 = Date.now();
|
|
34
|
+
try {
|
|
35
|
+
const prepared = prepareLocalExecution(q.sql, options.connection, projectRoot, projectConfig);
|
|
36
|
+
const result = await options.executor.executeQuery(prepared.sql, [], {}, prepared.connection);
|
|
37
|
+
queries.push({
|
|
38
|
+
chartId: q.id,
|
|
39
|
+
sql: q.sql,
|
|
40
|
+
rowCount: result.rows.length,
|
|
41
|
+
durationMs: Date.now() - t0,
|
|
42
|
+
preview: result.rows.slice(0, previewRows),
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
catch (err) {
|
|
46
|
+
queries.push({
|
|
47
|
+
chartId: q.id,
|
|
48
|
+
sql: q.sql,
|
|
49
|
+
rowCount: 0,
|
|
50
|
+
durationMs: Date.now() - t0,
|
|
51
|
+
error: err instanceof Error ? err.message : String(err),
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
// Evaluate alerts.
|
|
56
|
+
alertResults = await evaluateAlerts(dashboard.metadata.alerts ?? [], options.executor, options.connection);
|
|
57
|
+
// Fire notifications only if at least one alert breached OR no alerts were
|
|
58
|
+
// declared (so scheduled runs without alerts still produce an audit entry).
|
|
59
|
+
const anyBreached = alertResults.some((a) => a.breached);
|
|
60
|
+
const hasAlerts = alertResults.length > 0;
|
|
61
|
+
const shouldNotify = anyBreached || !hasAlerts;
|
|
62
|
+
if (shouldNotify && (dashboard.metadata.notifications?.length ?? 0) > 0) {
|
|
63
|
+
const payload = {
|
|
64
|
+
block,
|
|
65
|
+
path: absPath,
|
|
66
|
+
startedAt,
|
|
67
|
+
alerts: alertResults,
|
|
68
|
+
queries,
|
|
69
|
+
trigger,
|
|
70
|
+
};
|
|
71
|
+
if (isDigestOutput(dashboard)) {
|
|
72
|
+
try {
|
|
73
|
+
const digest = await runDigestBuild(dashboard, projectRoot);
|
|
74
|
+
payload.html = digest.html;
|
|
75
|
+
payload.markdown = digest.markdown;
|
|
76
|
+
payload.digestTitle = dashboard.metadata.title ?? block;
|
|
77
|
+
payload.digestDiagnostics = digest.diagnostics;
|
|
78
|
+
}
|
|
79
|
+
catch (err) {
|
|
80
|
+
payload.digestDiagnostics = [
|
|
81
|
+
{
|
|
82
|
+
level: 'warning',
|
|
83
|
+
message: `digest build failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
84
|
+
},
|
|
85
|
+
];
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
notifications = await dispatchNotifications(dashboard.metadata.notifications ?? [], payload, projectRoot);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
catch (err) {
|
|
93
|
+
error = err instanceof Error ? err.message : String(err);
|
|
94
|
+
}
|
|
95
|
+
const record = {
|
|
96
|
+
startedAt,
|
|
97
|
+
finishedAt: new Date().toISOString(),
|
|
98
|
+
block,
|
|
99
|
+
path: absPath,
|
|
100
|
+
trigger,
|
|
101
|
+
queries,
|
|
102
|
+
alerts: alertResults,
|
|
103
|
+
notifications,
|
|
104
|
+
error,
|
|
105
|
+
};
|
|
106
|
+
writeRunRecord(projectRoot, record);
|
|
107
|
+
return record;
|
|
108
|
+
}
|
|
109
|
+
//# sourceMappingURL=runner.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runner.js","sourceRoot":"","sources":["../../src/schedule/runner.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { RunRecord } from './types.js';
|
|
2
|
+
export declare function ensureRunsDir(projectRoot: string): string;
|
|
3
|
+
export declare function writeRunRecord(projectRoot: string, record: RunRecord): string;
|
|
4
|
+
export declare function listRunRecords(projectRoot: string, limit?: number): RunRecord[];
|
|
5
|
+
//# sourceMappingURL=runs.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runs.d.ts","sourceRoot":"","sources":["../../src/schedule/runs.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAI5C,wBAAgB,aAAa,CAAC,WAAW,EAAE,MAAM,GAAG,MAAM,CAMzD;AAED,wBAAgB,cAAc,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,GAAG,MAAM,CAO7E;AAED,wBAAgB,cAAc,CAAC,WAAW,EAAE,MAAM,EAAE,KAAK,SAAK,GAAG,SAAS,EAAE,CAoB3E"}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, readdirSync, readFileSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
const RUNS_DIR = '.dql/runs';
|
|
4
|
+
export function ensureRunsDir(projectRoot) {
|
|
5
|
+
const dir = join(projectRoot, RUNS_DIR);
|
|
6
|
+
if (!existsSync(dir)) {
|
|
7
|
+
mkdirSync(dir, { recursive: true });
|
|
8
|
+
}
|
|
9
|
+
return dir;
|
|
10
|
+
}
|
|
11
|
+
export function writeRunRecord(projectRoot, record) {
|
|
12
|
+
const dir = ensureRunsDir(projectRoot);
|
|
13
|
+
const stamp = record.startedAt.replace(/[:.]/g, '-');
|
|
14
|
+
const safeName = record.block.replace(/[^a-zA-Z0-9_-]+/g, '_');
|
|
15
|
+
const file = join(dir, `${stamp}-${safeName}.json`);
|
|
16
|
+
writeFileSync(file, JSON.stringify(record, null, 2) + '\n', 'utf-8');
|
|
17
|
+
return file;
|
|
18
|
+
}
|
|
19
|
+
export function listRunRecords(projectRoot, limit = 20) {
|
|
20
|
+
const dir = join(projectRoot, RUNS_DIR);
|
|
21
|
+
if (!existsSync(dir))
|
|
22
|
+
return [];
|
|
23
|
+
const files = readdirSync(dir)
|
|
24
|
+
.filter((f) => f.endsWith('.json'))
|
|
25
|
+
.sort()
|
|
26
|
+
.reverse()
|
|
27
|
+
.slice(0, limit);
|
|
28
|
+
const records = [];
|
|
29
|
+
for (const f of files) {
|
|
30
|
+
try {
|
|
31
|
+
const raw = readFileSync(join(dir, f), 'utf-8');
|
|
32
|
+
records.push(JSON.parse(raw));
|
|
33
|
+
}
|
|
34
|
+
catch {
|
|
35
|
+
/* skip malformed */
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
return records;
|
|
39
|
+
}
|
|
40
|
+
//# sourceMappingURL=runs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runs.js","sourceRoot":"","sources":["../../src/schedule/runs.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,WAAW,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAC1F,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC,MAAM,QAAQ,GAAG,WAAW,CAAC;AAE7B,MAAM,UAAU,aAAa,CAAC,WAAmB;IAC/C,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IACxC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QACrB,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACtC,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,UAAU,cAAc,CAAC,WAAmB,EAAE,MAAiB;IACnE,MAAM,GAAG,GAAG,aAAa,CAAC,WAAW,CAAC,CAAC;IACvC,MAAM,KAAK,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;IACrD,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,kBAAkB,EAAE,GAAG,CAAC,CAAC;IAC/D,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,IAAI,QAAQ,OAAO,CAAC,CAAC;IACpD,aAAa,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC;IACrE,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,cAAc,CAAC,WAAmB,EAAE,KAAK,GAAG,EAAE;IAC5D,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IACxC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,OAAO,EAAE,CAAC;IAEhC,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,CAAC;SAC3B,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;SAClC,IAAI,EAAE;SACN,OAAO,EAAE;SACT,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;IAEnB,MAAM,OAAO,GAAgB,EAAE,CAAC;IAChC,KAAK,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;QACtB,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,YAAY,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;YAChD,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAc,CAAC,CAAC;QAC7C,CAAC;QAAC,MAAM,CAAC;YACP,oBAAoB;QACtB,CAAC;IACH,CAAC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { type ConnectionConfig } from '@duckcodeailabs/dql-connectors';
|
|
2
|
+
import type { ScheduledBlock } from './types.js';
|
|
3
|
+
export interface ServiceOptions {
|
|
4
|
+
projectRoot?: string;
|
|
5
|
+
connection?: ConnectionConfig;
|
|
6
|
+
}
|
|
7
|
+
export interface StartedService {
|
|
8
|
+
blocks: ScheduledBlock[];
|
|
9
|
+
stop: () => Promise<void>;
|
|
10
|
+
}
|
|
11
|
+
export declare function startScheduleService(options?: ServiceOptions): Promise<StartedService>;
|
|
12
|
+
//# sourceMappingURL=service.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"service.d.ts","sourceRoot":"","sources":["../../src/schedule/service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiB,KAAK,gBAAgB,EAAE,MAAM,gCAAgC,CAAC;AAItF,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAEjD,MAAM,WAAW,cAAc;IAC7B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,CAAC,EAAE,gBAAgB,CAAC;CAC/B;AAED,MAAM,WAAW,cAAc;IAC7B,MAAM,EAAE,cAAc,EAAE,CAAC;IACzB,IAAI,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC3B;AAED,wBAAsB,oBAAoB,CAAC,OAAO,GAAE,cAAmB,GAAG,OAAO,CAAC,cAAc,CAAC,CAiDhG"}
|