@duckcodeailabs/dql-cli 1.4.3 → 1.5.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.
Files changed (283) hide show
  1. package/LICENSE +123 -0
  2. package/README.md +72 -0
  3. package/dist/apps-api.d.ts +79 -0
  4. package/dist/apps-api.d.ts.map +1 -0
  5. package/dist/apps-api.js +934 -0
  6. package/dist/apps-api.js.map +1 -0
  7. package/dist/apps-api.test.d.ts +2 -0
  8. package/dist/apps-api.test.d.ts.map +1 -0
  9. package/dist/apps-api.test.js +111 -0
  10. package/dist/apps-api.test.js.map +1 -0
  11. package/dist/args.d.ts +30 -0
  12. package/dist/args.d.ts.map +1 -0
  13. package/dist/args.js +105 -0
  14. package/dist/args.js.map +1 -0
  15. package/dist/args.test.d.ts +2 -0
  16. package/dist/args.test.d.ts.map +1 -0
  17. package/dist/args.test.js +41 -0
  18. package/dist/args.test.js.map +1 -0
  19. package/dist/assets/dql-notebook/assets/codemirror-DJYUkPr1.js +11 -0
  20. package/dist/assets/dql-notebook/assets/index-R3UrqjLQ.css +1 -0
  21. package/dist/assets/dql-notebook/assets/index-mlfOQ2me.js +857 -0
  22. package/dist/assets/dql-notebook/assets/react-CRB3T2We.js +32 -0
  23. package/dist/assets/dql-notebook/index.html +18 -0
  24. package/dist/assets/notebook-browser/app.js +548 -0
  25. package/dist/assets/notebook-browser/index.html +83 -0
  26. package/dist/assets/notebook-browser/styles.css +336 -0
  27. package/dist/block-studio-import.d.ts +58 -0
  28. package/dist/block-studio-import.d.ts.map +1 -0
  29. package/dist/block-studio-import.js +390 -0
  30. package/dist/block-studio-import.js.map +1 -0
  31. package/dist/block-studio-import.test.d.ts +2 -0
  32. package/dist/block-studio-import.test.d.ts.map +1 -0
  33. package/dist/block-studio-import.test.js +106 -0
  34. package/dist/block-studio-import.test.js.map +1 -0
  35. package/dist/block-templates.d.ts +8 -0
  36. package/dist/block-templates.d.ts.map +1 -0
  37. package/dist/block-templates.js +60 -0
  38. package/dist/block-templates.js.map +1 -0
  39. package/dist/commands/agent.d.ts +19 -0
  40. package/dist/commands/agent.d.ts.map +1 -0
  41. package/dist/commands/agent.js +258 -0
  42. package/dist/commands/agent.js.map +1 -0
  43. package/dist/commands/app.d.ts +32 -0
  44. package/dist/commands/app.d.ts.map +1 -0
  45. package/dist/commands/app.js +307 -0
  46. package/dist/commands/app.js.map +1 -0
  47. package/dist/commands/build.d.ts +3 -0
  48. package/dist/commands/build.d.ts.map +1 -0
  49. package/dist/commands/build.js +69 -0
  50. package/dist/commands/build.js.map +1 -0
  51. package/dist/commands/build.test.d.ts +2 -0
  52. package/dist/commands/build.test.d.ts.map +1 -0
  53. package/dist/commands/build.test.js +44 -0
  54. package/dist/commands/build.test.js.map +1 -0
  55. package/dist/commands/certify.d.ts +3 -0
  56. package/dist/commands/certify.d.ts.map +1 -0
  57. package/dist/commands/certify.js +228 -0
  58. package/dist/commands/certify.js.map +1 -0
  59. package/dist/commands/compile.d.ts +21 -0
  60. package/dist/commands/compile.d.ts.map +1 -0
  61. package/dist/commands/compile.js +198 -0
  62. package/dist/commands/compile.js.map +1 -0
  63. package/dist/commands/compile.test.d.ts +2 -0
  64. package/dist/commands/compile.test.d.ts.map +1 -0
  65. package/dist/commands/compile.test.js +115 -0
  66. package/dist/commands/compile.test.js.map +1 -0
  67. package/dist/commands/diff.d.ts +3 -0
  68. package/dist/commands/diff.d.ts.map +1 -0
  69. package/dist/commands/diff.js +52 -0
  70. package/dist/commands/diff.js.map +1 -0
  71. package/dist/commands/doctor.d.ts +3 -0
  72. package/dist/commands/doctor.d.ts.map +1 -0
  73. package/dist/commands/doctor.js +191 -0
  74. package/dist/commands/doctor.js.map +1 -0
  75. package/dist/commands/doctor.test.d.ts +2 -0
  76. package/dist/commands/doctor.test.d.ts.map +1 -0
  77. package/dist/commands/doctor.test.js +43 -0
  78. package/dist/commands/doctor.test.js.map +1 -0
  79. package/dist/commands/fmt.d.ts +3 -0
  80. package/dist/commands/fmt.d.ts.map +1 -0
  81. package/dist/commands/fmt.js +53 -0
  82. package/dist/commands/fmt.js.map +1 -0
  83. package/dist/commands/import.d.ts +3 -0
  84. package/dist/commands/import.d.ts.map +1 -0
  85. package/dist/commands/import.js +50 -0
  86. package/dist/commands/import.js.map +1 -0
  87. package/dist/commands/info.d.ts +3 -0
  88. package/dist/commands/info.d.ts.map +1 -0
  89. package/dist/commands/info.js +56 -0
  90. package/dist/commands/info.js.map +1 -0
  91. package/dist/commands/init.d.ts +3 -0
  92. package/dist/commands/init.d.ts.map +1 -0
  93. package/dist/commands/init.js +250 -0
  94. package/dist/commands/init.js.map +1 -0
  95. package/dist/commands/init.test.d.ts +2 -0
  96. package/dist/commands/init.test.d.ts.map +1 -0
  97. package/dist/commands/init.test.js +118 -0
  98. package/dist/commands/init.test.js.map +1 -0
  99. package/dist/commands/lineage.d.ts +24 -0
  100. package/dist/commands/lineage.d.ts.map +1 -0
  101. package/dist/commands/lineage.js +634 -0
  102. package/dist/commands/lineage.js.map +1 -0
  103. package/dist/commands/mcp.d.ts +7 -0
  104. package/dist/commands/mcp.d.ts.map +1 -0
  105. package/dist/commands/mcp.js +16 -0
  106. package/dist/commands/mcp.js.map +1 -0
  107. package/dist/commands/migrate.d.ts +12 -0
  108. package/dist/commands/migrate.d.ts.map +1 -0
  109. package/dist/commands/migrate.js +197 -0
  110. package/dist/commands/migrate.js.map +1 -0
  111. package/dist/commands/new.d.ts +3 -0
  112. package/dist/commands/new.d.ts.map +1 -0
  113. package/dist/commands/new.js +490 -0
  114. package/dist/commands/new.js.map +1 -0
  115. package/dist/commands/new.test.d.ts +2 -0
  116. package/dist/commands/new.test.d.ts.map +1 -0
  117. package/dist/commands/new.test.js +191 -0
  118. package/dist/commands/new.test.js.map +1 -0
  119. package/dist/commands/notebook.d.ts +3 -0
  120. package/dist/commands/notebook.d.ts.map +1 -0
  121. package/dist/commands/notebook.js +46 -0
  122. package/dist/commands/notebook.js.map +1 -0
  123. package/dist/commands/parse.d.ts +3 -0
  124. package/dist/commands/parse.d.ts.map +1 -0
  125. package/dist/commands/parse.js +63 -0
  126. package/dist/commands/parse.js.map +1 -0
  127. package/dist/commands/preview.d.ts +3 -0
  128. package/dist/commands/preview.d.ts.map +1 -0
  129. package/dist/commands/preview.js +42 -0
  130. package/dist/commands/preview.js.map +1 -0
  131. package/dist/commands/schedule.d.ts +3 -0
  132. package/dist/commands/schedule.d.ts.map +1 -0
  133. package/dist/commands/schedule.js +215 -0
  134. package/dist/commands/schedule.js.map +1 -0
  135. package/dist/commands/semantic.d.ts +12 -0
  136. package/dist/commands/semantic.d.ts.map +1 -0
  137. package/dist/commands/semantic.js +356 -0
  138. package/dist/commands/semantic.js.map +1 -0
  139. package/dist/commands/serve.d.ts +3 -0
  140. package/dist/commands/serve.d.ts.map +1 -0
  141. package/dist/commands/serve.js +30 -0
  142. package/dist/commands/serve.js.map +1 -0
  143. package/dist/commands/slack.d.ts +13 -0
  144. package/dist/commands/slack.d.ts.map +1 -0
  145. package/dist/commands/slack.js +53 -0
  146. package/dist/commands/slack.js.map +1 -0
  147. package/dist/commands/sync.d.ts +3 -0
  148. package/dist/commands/sync.d.ts.map +1 -0
  149. package/dist/commands/sync.js +192 -0
  150. package/dist/commands/sync.js.map +1 -0
  151. package/dist/commands/sync.test.d.ts +2 -0
  152. package/dist/commands/sync.test.d.ts.map +1 -0
  153. package/dist/commands/sync.test.js +147 -0
  154. package/dist/commands/sync.test.js.map +1 -0
  155. package/dist/commands/test.d.ts +3 -0
  156. package/dist/commands/test.d.ts.map +1 -0
  157. package/dist/commands/test.js +167 -0
  158. package/dist/commands/test.js.map +1 -0
  159. package/dist/commands/validate.d.ts +3 -0
  160. package/dist/commands/validate.d.ts.map +1 -0
  161. package/dist/commands/validate.js +163 -0
  162. package/dist/commands/validate.js.map +1 -0
  163. package/dist/commands/validate.test.d.ts +2 -0
  164. package/dist/commands/validate.test.d.ts.map +1 -0
  165. package/dist/commands/validate.test.js +55 -0
  166. package/dist/commands/validate.test.js.map +1 -0
  167. package/dist/commands/verify.d.ts +11 -0
  168. package/dist/commands/verify.d.ts.map +1 -0
  169. package/dist/commands/verify.js +74 -0
  170. package/dist/commands/verify.js.map +1 -0
  171. package/dist/digest.d.ts +10 -0
  172. package/dist/digest.d.ts.map +1 -0
  173. package/dist/digest.js +83 -0
  174. package/dist/digest.js.map +1 -0
  175. package/dist/git-service.d.ts +17 -0
  176. package/dist/git-service.d.ts.map +1 -0
  177. package/dist/git-service.js +54 -0
  178. package/dist/git-service.js.map +1 -0
  179. package/dist/governance-runtime.d.ts +15 -0
  180. package/dist/governance-runtime.d.ts.map +1 -0
  181. package/dist/governance-runtime.js +50 -0
  182. package/dist/governance-runtime.js.map +1 -0
  183. package/dist/index.d.ts +3 -0
  184. package/dist/index.d.ts.map +1 -0
  185. package/{index.js → dist/index.js} +5 -0
  186. package/dist/index.js.map +1 -0
  187. package/dist/llm/index.d.ts +4 -0
  188. package/dist/llm/index.d.ts.map +1 -0
  189. package/dist/llm/index.js +20 -0
  190. package/dist/llm/index.js.map +1 -0
  191. package/dist/llm/providers/claude-agent-sdk.d.ts +3 -0
  192. package/dist/llm/providers/claude-agent-sdk.d.ts.map +1 -0
  193. package/dist/llm/providers/claude-agent-sdk.js +174 -0
  194. package/dist/llm/providers/claude-agent-sdk.js.map +1 -0
  195. package/dist/llm/providers/claude-code.d.ts +8 -0
  196. package/dist/llm/providers/claude-code.d.ts.map +1 -0
  197. package/dist/llm/providers/claude-code.js +171 -0
  198. package/dist/llm/providers/claude-code.js.map +1 -0
  199. package/dist/llm/providers/dql-agent-provider.d.ts +5 -0
  200. package/dist/llm/providers/dql-agent-provider.d.ts.map +1 -0
  201. package/dist/llm/providers/dql-agent-provider.js +287 -0
  202. package/dist/llm/providers/dql-agent-provider.js.map +1 -0
  203. package/dist/llm/tools.d.ts +9 -0
  204. package/dist/llm/tools.d.ts.map +1 -0
  205. package/dist/llm/tools.js +112 -0
  206. package/dist/llm/tools.js.map +1 -0
  207. package/dist/llm/types.d.ts +72 -0
  208. package/dist/llm/types.d.ts.map +1 -0
  209. package/dist/llm/types.js +2 -0
  210. package/dist/llm/types.js.map +1 -0
  211. package/dist/local-runtime.d.ts +142 -0
  212. package/dist/local-runtime.d.ts.map +1 -0
  213. package/dist/local-runtime.js +4859 -0
  214. package/dist/local-runtime.js.map +1 -0
  215. package/dist/local-runtime.test.d.ts +2 -0
  216. package/dist/local-runtime.test.d.ts.map +1 -0
  217. package/dist/local-runtime.test.js +241 -0
  218. package/dist/local-runtime.test.js.map +1 -0
  219. package/dist/metricflow.d.ts +35 -0
  220. package/dist/metricflow.d.ts.map +1 -0
  221. package/dist/metricflow.js +122 -0
  222. package/dist/metricflow.js.map +1 -0
  223. package/dist/metricflow.test.d.ts +2 -0
  224. package/dist/metricflow.test.d.ts.map +1 -0
  225. package/dist/metricflow.test.js +54 -0
  226. package/dist/metricflow.test.js.map +1 -0
  227. package/dist/open-browser.d.ts +2 -0
  228. package/dist/open-browser.d.ts.map +1 -0
  229. package/dist/open-browser.js +29 -0
  230. package/dist/open-browser.js.map +1 -0
  231. package/dist/schedule/alerts.d.ts +5 -0
  232. package/dist/schedule/alerts.d.ts.map +1 -0
  233. package/dist/schedule/alerts.js +54 -0
  234. package/dist/schedule/alerts.js.map +1 -0
  235. package/dist/schedule/discovery.d.ts +4 -0
  236. package/dist/schedule/discovery.d.ts.map +1 -0
  237. package/dist/schedule/discovery.js +36 -0
  238. package/dist/schedule/discovery.js.map +1 -0
  239. package/dist/schedule/notifiers/email.d.ts +3 -0
  240. package/dist/schedule/notifiers/email.d.ts.map +1 -0
  241. package/dist/schedule/notifiers/email.js +76 -0
  242. package/dist/schedule/notifiers/email.js.map +1 -0
  243. package/dist/schedule/notifiers/file.d.ts +3 -0
  244. package/dist/schedule/notifiers/file.d.ts.map +1 -0
  245. package/dist/schedule/notifiers/file.js +50 -0
  246. package/dist/schedule/notifiers/file.js.map +1 -0
  247. package/dist/schedule/notifiers/index.d.ts +10 -0
  248. package/dist/schedule/notifiers/index.d.ts.map +1 -0
  249. package/dist/schedule/notifiers/index.js +33 -0
  250. package/dist/schedule/notifiers/index.js.map +1 -0
  251. package/dist/schedule/notifiers/slack.d.ts +3 -0
  252. package/dist/schedule/notifiers/slack.d.ts.map +1 -0
  253. package/dist/schedule/notifiers/slack.js +58 -0
  254. package/dist/schedule/notifiers/slack.js.map +1 -0
  255. package/dist/schedule/runner.d.ts +14 -0
  256. package/dist/schedule/runner.d.ts.map +1 -0
  257. package/dist/schedule/runner.js +225 -0
  258. package/dist/schedule/runner.js.map +1 -0
  259. package/dist/schedule/runs.d.ts +5 -0
  260. package/dist/schedule/runs.d.ts.map +1 -0
  261. package/dist/schedule/runs.js +41 -0
  262. package/dist/schedule/runs.js.map +1 -0
  263. package/dist/schedule/service.d.ts +13 -0
  264. package/dist/schedule/service.d.ts.map +1 -0
  265. package/dist/schedule/service.js +87 -0
  266. package/dist/schedule/service.js.map +1 -0
  267. package/dist/schedule/types.d.ts +70 -0
  268. package/dist/schedule/types.d.ts.map +1 -0
  269. package/dist/schedule/types.js +2 -0
  270. package/dist/schedule/types.js.map +1 -0
  271. package/dist/semantic-import.d.ts +135 -0
  272. package/dist/semantic-import.d.ts.map +1 -0
  273. package/dist/semantic-import.js +979 -0
  274. package/dist/semantic-import.js.map +1 -0
  275. package/dist/semantic-import.test.d.ts +2 -0
  276. package/dist/semantic-import.test.d.ts.map +1 -0
  277. package/dist/semantic-import.test.js +95 -0
  278. package/dist/semantic-import.test.js.map +1 -0
  279. package/dist/settings/provider-settings.d.ts +33 -0
  280. package/dist/settings/provider-settings.d.ts.map +1 -0
  281. package/dist/settings/provider-settings.js +91 -0
  282. package/dist/settings/provider-settings.js.map +1 -0
  283. package/package.json +29 -21
@@ -0,0 +1,390 @@
1
+ import { createHash } from 'node:crypto';
2
+ import { existsSync, mkdirSync, readdirSync, readFileSync, statSync, writeFileSync } from 'node:fs';
3
+ import { basename, extname, join, relative, resolve } from 'node:path';
4
+ const SQL_EXTENSIONS = new Set(['.sql']);
5
+ const ACTIVE_IMPORT_KINDS = new Set(['raw-sql-file', 'raw-sql-folder']);
6
+ export function createBlockStudioImportSession(projectRoot, options) {
7
+ const inputPath = resolveInputPath(projectRoot, options.inputPath);
8
+ const sourceKind = resolveSourceKind(inputPath, options.sourceKind);
9
+ if (!ACTIVE_IMPORT_KINDS.has(sourceKind)) {
10
+ throw new Error(`${sourceKind} import is planned but not implemented in this build.`);
11
+ }
12
+ const now = new Date().toISOString();
13
+ const defaults = {
14
+ domain: sanitizeDomain(options.domain || 'imported'),
15
+ owner: options.owner?.trim() ?? '',
16
+ tags: normalizeTags(['imported', 'raw-sql', ...(options.tags ?? [])]),
17
+ };
18
+ const statements = collectSqlStatements(inputPath);
19
+ const importId = buildImportId(sourceKind, inputPath, now);
20
+ const candidates = statements.map((statement) => buildSqlCandidate({
21
+ importId,
22
+ sourceKind,
23
+ projectRoot,
24
+ statement,
25
+ defaults,
26
+ }));
27
+ const manifest = {
28
+ id: importId,
29
+ sourceKind,
30
+ inputPath: displayPath(projectRoot, inputPath),
31
+ createdAt: now,
32
+ updatedAt: now,
33
+ defaults,
34
+ candidateIds: candidates.map((candidate) => candidate.id),
35
+ };
36
+ const session = { ...manifest, candidates };
37
+ writeBlockStudioImportSession(projectRoot, session);
38
+ return session;
39
+ }
40
+ export function loadBlockStudioImportSession(projectRoot, importId) {
41
+ const root = importRoot(projectRoot, importId);
42
+ const manifestPath = join(root, 'manifest.json');
43
+ if (!existsSync(manifestPath))
44
+ throw new Error(`Import session not found: ${importId}`);
45
+ const manifest = JSON.parse(readFileSync(manifestPath, 'utf-8'));
46
+ const candidates = manifest.candidateIds.map((candidateId) => readBlockStudioImportCandidate(projectRoot, importId, candidateId));
47
+ return { ...manifest, candidates };
48
+ }
49
+ export function writeBlockStudioImportSession(projectRoot, session) {
50
+ const root = importRoot(projectRoot, session.id);
51
+ const candidatesDir = join(root, 'candidates');
52
+ mkdirSync(candidatesDir, { recursive: true });
53
+ const manifest = {
54
+ id: session.id,
55
+ sourceKind: session.sourceKind,
56
+ inputPath: session.inputPath,
57
+ createdAt: session.createdAt,
58
+ updatedAt: new Date().toISOString(),
59
+ defaults: session.defaults,
60
+ candidateIds: session.candidates.map((candidate) => candidate.id),
61
+ };
62
+ writeFileSync(join(root, 'manifest.json'), JSON.stringify(manifest, null, 2) + '\n', 'utf-8');
63
+ for (const candidate of session.candidates) {
64
+ writeBlockStudioImportCandidate(projectRoot, session.id, candidate);
65
+ }
66
+ }
67
+ export function writeBlockStudioImportCandidate(projectRoot, importId, candidate) {
68
+ const candidatesDir = join(importRoot(projectRoot, importId), 'candidates');
69
+ mkdirSync(candidatesDir, { recursive: true });
70
+ writeFileSync(join(candidatesDir, `${candidate.id}.json`), JSON.stringify(candidate, null, 2) + '\n', 'utf-8');
71
+ }
72
+ export function readBlockStudioImportCandidate(projectRoot, importId, candidateId) {
73
+ const candidatePath = join(importRoot(projectRoot, importId), 'candidates', `${candidateId}.json`);
74
+ if (!existsSync(candidatePath))
75
+ throw new Error(`Import candidate not found: ${candidateId}`);
76
+ return JSON.parse(readFileSync(candidatePath, 'utf-8'));
77
+ }
78
+ export function updateBlockStudioImportCandidate(projectRoot, importId, candidateId, patch) {
79
+ const candidate = readBlockStudioImportCandidate(projectRoot, importId, candidateId);
80
+ const next = { ...candidate };
81
+ if (patch.name !== undefined)
82
+ next.name = patch.name;
83
+ if (patch.domain !== undefined)
84
+ next.domain = sanitizeDomain(patch.domain);
85
+ if (patch.description !== undefined)
86
+ next.description = patch.description;
87
+ if (patch.owner !== undefined)
88
+ next.owner = patch.owner;
89
+ if (patch.tags !== undefined)
90
+ next.tags = normalizeTags(patch.tags);
91
+ if (patch.sql !== undefined)
92
+ next.sql = patch.sql;
93
+ if (patch.reviewStatus !== undefined)
94
+ next.reviewStatus = patch.reviewStatus;
95
+ if (patch.name || patch.domain || patch.description || patch.owner || patch.tags || patch.sql) {
96
+ next.dqlSource = candidateToDqlSource(next);
97
+ next.lineage = {
98
+ ...next.lineage,
99
+ sourceTables: extractSourceTables(next.sql),
100
+ parameters: extractSqlParameters(next.sql),
101
+ };
102
+ next.confidence = scoreSqlCandidate(next.sql, next.lineage);
103
+ }
104
+ writeBlockStudioImportCandidate(projectRoot, importId, next);
105
+ return next;
106
+ }
107
+ export function candidateToDqlSource(candidate) {
108
+ const tags = normalizeTags(candidate.tags).map((tag) => dqlString(tag)).join(', ');
109
+ const sql = candidate.sql.trim().replace(/"""/g, '\\"\\"\\"');
110
+ return `block ${dqlString(candidate.name)} {
111
+ domain = ${dqlString(sanitizeDomain(candidate.domain))}
112
+ type = "custom"
113
+ description = ${dqlString(candidate.description)}
114
+ tags = [${tags}]
115
+ owner = ${dqlString(candidate.owner)}
116
+
117
+ query = """
118
+ ${sql}
119
+ """
120
+
121
+ visualization {
122
+ chart = "table"
123
+ }
124
+
125
+ tests {
126
+ assert row_count > 0
127
+ }
128
+ }
129
+ `;
130
+ }
131
+ function collectSqlStatements(inputPath) {
132
+ const stats = statSync(inputPath);
133
+ const files = stats.isDirectory() ? walkSqlFiles(inputPath) : [inputPath];
134
+ if (files.length === 0)
135
+ throw new Error('No .sql files found to import.');
136
+ const statements = [];
137
+ for (const file of files) {
138
+ const source = readFileSync(file, 'utf-8');
139
+ const split = splitSqlStatements(source);
140
+ split.forEach((sql, index) => {
141
+ statements.push({
142
+ sourcePath: file,
143
+ sql,
144
+ statementIndex: index + 1,
145
+ totalStatements: split.length,
146
+ });
147
+ });
148
+ }
149
+ if (statements.length === 0)
150
+ throw new Error('No SQL statements found to import.');
151
+ return statements;
152
+ }
153
+ function buildSqlCandidate(options) {
154
+ const statement = options.statement;
155
+ const sourcePath = displayPath(options.projectRoot, statement.sourcePath);
156
+ const metadata = extractStatementMetadata(statement.sql);
157
+ const baseName = metadata.name || basename(statement.sourcePath, extname(statement.sourcePath));
158
+ const name = statement.totalStatements > 1
159
+ ? `${baseName} ${statement.statementIndex}`
160
+ : baseName;
161
+ const sourceTables = extractSourceTables(statement.sql);
162
+ const parameters = extractSqlParameters(statement.sql);
163
+ const warnings = [
164
+ ...(parameters.length > 0 ? [`Contains parameters: ${parameters.join(', ')}`] : []),
165
+ ...(statement.sql.includes('"""') ? ['SQL contains triple quotes that were escaped in the DQL draft.'] : []),
166
+ ];
167
+ const lineage = {
168
+ sourceTables,
169
+ parameters,
170
+ warnings,
171
+ statementIndex: statement.statementIndex,
172
+ totalStatements: statement.totalStatements,
173
+ };
174
+ const candidate = {
175
+ id: buildCandidateId(sourcePath, statement.statementIndex, statement.sql),
176
+ sourceKind: options.sourceKind,
177
+ sourcePath,
178
+ name: titleizeName(name),
179
+ domain: options.defaults.domain,
180
+ description: metadata.description || `Imported from ${sourcePath}`,
181
+ owner: options.defaults.owner,
182
+ tags: normalizeTags(options.defaults.tags),
183
+ sql: statement.sql.trim(),
184
+ dqlSource: '',
185
+ validation: null,
186
+ preview: null,
187
+ lineage,
188
+ confidence: scoreSqlCandidate(statement.sql, lineage),
189
+ reviewStatus: 'draft',
190
+ };
191
+ candidate.dqlSource = candidateToDqlSource(candidate);
192
+ return candidate;
193
+ }
194
+ function splitSqlStatements(source) {
195
+ const statements = [];
196
+ let start = 0;
197
+ let single = false;
198
+ let double = false;
199
+ let backtick = false;
200
+ let lineComment = false;
201
+ let blockComment = false;
202
+ let dollarQuote = null;
203
+ for (let i = 0; i < source.length; i += 1) {
204
+ const char = source[i];
205
+ const next = source[i + 1];
206
+ if (lineComment) {
207
+ if (char === '\n')
208
+ lineComment = false;
209
+ continue;
210
+ }
211
+ if (blockComment) {
212
+ if (char === '*' && next === '/') {
213
+ blockComment = false;
214
+ i += 1;
215
+ }
216
+ continue;
217
+ }
218
+ if (dollarQuote) {
219
+ if (source.startsWith(dollarQuote, i)) {
220
+ i += dollarQuote.length - 1;
221
+ dollarQuote = null;
222
+ }
223
+ continue;
224
+ }
225
+ if (!single && !double && !backtick) {
226
+ if (char === '-' && next === '-') {
227
+ lineComment = true;
228
+ i += 1;
229
+ continue;
230
+ }
231
+ if (char === '/' && next === '*') {
232
+ blockComment = true;
233
+ i += 1;
234
+ continue;
235
+ }
236
+ if (char === '$') {
237
+ const tag = source.slice(i).match(/^\$[A-Za-z_][A-Za-z0-9_]*\$|^\$\$/)?.[0];
238
+ if (tag) {
239
+ dollarQuote = tag;
240
+ i += tag.length - 1;
241
+ continue;
242
+ }
243
+ }
244
+ }
245
+ if (!double && !backtick && char === "'" && source[i - 1] !== '\\')
246
+ single = !single;
247
+ else if (!single && !backtick && char === '"' && source[i - 1] !== '\\')
248
+ double = !double;
249
+ else if (!single && !double && char === '`')
250
+ backtick = !backtick;
251
+ if (!single && !double && !backtick && char === ';') {
252
+ const statement = source.slice(start, i).trim();
253
+ if (statement)
254
+ statements.push(statement);
255
+ start = i + 1;
256
+ }
257
+ }
258
+ const trailing = source.slice(start).trim();
259
+ if (trailing)
260
+ statements.push(trailing);
261
+ return statements;
262
+ }
263
+ function extractStatementMetadata(sql) {
264
+ const leading = sql.split(/\r?\n/).slice(0, 12).join('\n');
265
+ const name = leading.match(/(?:--|\/\*)\s*(?:name|block)\s*:\s*([^*\n]+)/i)?.[1]?.trim() ?? '';
266
+ const description = leading.match(/(?:--|\/\*)\s*(?:description|desc)\s*:\s*([^*\n]+)/i)?.[1]?.trim() ?? '';
267
+ if (description)
268
+ return { name, description };
269
+ const firstComment = leading.match(/^\s*--\s*(?!name\s*:|block\s*:)(.+)$/im)?.[1]?.trim() ?? '';
270
+ return { name, description: firstComment };
271
+ }
272
+ function extractSourceTables(sql) {
273
+ const tables = new Set();
274
+ const cleaned = stripSqlComments(sql);
275
+ const regex = /\b(?:from|join|update|into)\s+([`"[]?[A-Za-z0-9_./:-]+(?:\.[A-Za-z0-9_./:-]+)*[`"\]]?)/gi;
276
+ let match;
277
+ while ((match = regex.exec(cleaned))) {
278
+ const raw = match[1].replace(/^[`"[]|[`"\]]$/g, '');
279
+ if (!raw || raw.startsWith('('))
280
+ continue;
281
+ if (/^(select|values|unnest|lateral)$/i.test(raw))
282
+ continue;
283
+ tables.add(raw);
284
+ }
285
+ return Array.from(tables);
286
+ }
287
+ function extractSqlParameters(sql) {
288
+ const params = new Set();
289
+ const cleaned = stripSqlComments(sql);
290
+ let match;
291
+ const handlebars = /\{\{\s*([A-Za-z_][A-Za-z0-9_.-]*)\s*\}\}/g;
292
+ while ((match = handlebars.exec(cleaned)))
293
+ params.add(match[1]);
294
+ const colon = /(^|[^:]):([A-Za-z_][A-Za-z0-9_]*)\b/g;
295
+ while ((match = colon.exec(cleaned)))
296
+ params.add(match[2]);
297
+ const dollar = /(^|[^$])\$([A-Za-z_][A-Za-z0-9_]*)\b/g;
298
+ while ((match = dollar.exec(cleaned)))
299
+ params.add(match[2]);
300
+ return Array.from(params);
301
+ }
302
+ function stripSqlComments(sql) {
303
+ return sql
304
+ .replace(/\/\*[\s\S]*?\*\//g, ' ')
305
+ .replace(/--[^\n\r]*/g, ' ');
306
+ }
307
+ function scoreSqlCandidate(sql, lineage) {
308
+ let score = /\bselect\b/i.test(sql) ? 0.76 : 0.62;
309
+ if (lineage.sourceTables.length > 0)
310
+ score += 0.1;
311
+ if (/\bgroup\s+by\b/i.test(sql))
312
+ score += 0.04;
313
+ if (lineage.parameters.length > 0)
314
+ score -= 0.12;
315
+ if (lineage.totalStatements > 1)
316
+ score -= 0.03;
317
+ return Math.max(0.35, Math.min(0.92, Number(score.toFixed(2))));
318
+ }
319
+ function walkSqlFiles(root) {
320
+ const files = [];
321
+ const stack = [root];
322
+ while (stack.length > 0) {
323
+ const dir = stack.pop();
324
+ for (const entry of readdirSync(dir, { withFileTypes: true })) {
325
+ if (entry.name === 'node_modules' || entry.name === '.git' || entry.name === '.dql')
326
+ continue;
327
+ const full = join(dir, entry.name);
328
+ if (entry.isDirectory())
329
+ stack.push(full);
330
+ else if (entry.isFile() && SQL_EXTENSIONS.has(extname(entry.name).toLowerCase()))
331
+ files.push(full);
332
+ }
333
+ }
334
+ return files.sort();
335
+ }
336
+ function resolveInputPath(projectRoot, inputPath) {
337
+ if (!inputPath.trim())
338
+ throw new Error('Import path is required.');
339
+ const resolved = resolve(projectRoot, inputPath);
340
+ if (!existsSync(resolved))
341
+ throw new Error(`Import path not found: ${inputPath}`);
342
+ return resolved;
343
+ }
344
+ function resolveSourceKind(inputPath, requested) {
345
+ if (requested && requested !== 'raw-sql')
346
+ return requested;
347
+ const stats = statSync(inputPath);
348
+ if (stats.isDirectory())
349
+ return 'raw-sql-folder';
350
+ if (extname(inputPath).toLowerCase() === '.sql')
351
+ return 'raw-sql-file';
352
+ throw new Error('Only .sql files and folders are supported in this import build.');
353
+ }
354
+ function importRoot(projectRoot, importId) {
355
+ return join(projectRoot, '.dql', 'imports', importId);
356
+ }
357
+ function buildImportId(sourceKind, inputPath, createdAt) {
358
+ const hash = createHash('sha1').update(`${sourceKind}:${inputPath}:${createdAt}`).digest('hex').slice(0, 10);
359
+ return `imp_${hash}`;
360
+ }
361
+ function buildCandidateId(sourcePath, statementIndex, sql) {
362
+ const hash = createHash('sha1').update(`${sourcePath}:${statementIndex}:${sql}`).digest('hex').slice(0, 12);
363
+ return `cand_${hash}`;
364
+ }
365
+ function displayPath(projectRoot, absPath) {
366
+ const rel = relative(projectRoot, absPath).replaceAll('\\', '/');
367
+ return rel && !rel.startsWith('..') ? rel : absPath.replaceAll('\\', '/');
368
+ }
369
+ function sanitizeDomain(domain) {
370
+ return domain
371
+ .trim()
372
+ .toLowerCase()
373
+ .replace(/[^a-z0-9/_-]+/g, '-')
374
+ .replace(/^\/+|\/+$/g, '') || 'imported';
375
+ }
376
+ function normalizeTags(tags) {
377
+ return Array.from(new Set(tags.map((tag) => tag.trim().toLowerCase()).filter(Boolean)));
378
+ }
379
+ function titleizeName(name) {
380
+ return name
381
+ .replace(/\.[^.]+$/, '')
382
+ .replace(/[_-]+/g, ' ')
383
+ .replace(/\s+/g, ' ')
384
+ .trim()
385
+ .replace(/\b\w/g, (char) => char.toUpperCase()) || 'Imported Query';
386
+ }
387
+ function dqlString(value) {
388
+ return JSON.stringify(value ?? '');
389
+ }
390
+ //# sourceMappingURL=block-studio-import.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"block-studio-import.js","sourceRoot":"","sources":["../src/block-studio-import.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=block-studio-import.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"block-studio-import.test.d.ts","sourceRoot":"","sources":["../src/block-studio-import.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,106 @@
1
+ import { describe, expect, it, afterEach } from 'vitest';
2
+ import { mkdtempSync, readFileSync, rmSync, writeFileSync, mkdirSync } from 'node:fs';
3
+ import { join } from 'node:path';
4
+ import { tmpdir } from 'node:os';
5
+ import { candidateToDqlSource, createBlockStudioImportSession, loadBlockStudioImportSession, updateBlockStudioImportCandidate, } from './block-studio-import.js';
6
+ const tempDirs = [];
7
+ afterEach(() => {
8
+ while (tempDirs.length > 0) {
9
+ const dir = tempDirs.pop();
10
+ if (dir)
11
+ rmSync(dir, { recursive: true, force: true });
12
+ }
13
+ });
14
+ function tempProject() {
15
+ const dir = mkdtempSync(join(tmpdir(), 'dql-import-'));
16
+ tempDirs.push(dir);
17
+ return dir;
18
+ }
19
+ describe('Block Studio SQL import', () => {
20
+ it('previews a single SQL file as a draft block candidate', () => {
21
+ const root = tempProject();
22
+ writeFileSync(join(root, 'revenue.sql'), `-- name: revenue by region
23
+ -- description: Revenue by region from legacy BI
24
+ select region, sum(revenue) as total_revenue
25
+ from marts.orders
26
+ group by region;
27
+ `);
28
+ const session = createBlockStudioImportSession(root, {
29
+ inputPath: 'revenue.sql',
30
+ domain: 'finance',
31
+ owner: 'analytics',
32
+ });
33
+ expect(session.candidates).toHaveLength(1);
34
+ const candidate = session.candidates[0];
35
+ expect(candidate.sourceKind).toBe('raw-sql-file');
36
+ expect(candidate.name).toBe('Revenue By Region');
37
+ expect(candidate.domain).toBe('finance');
38
+ expect(candidate.owner).toBe('analytics');
39
+ expect(candidate.description).toBe('Revenue by region from legacy BI');
40
+ expect(candidate.lineage.sourceTables).toEqual(['marts.orders']);
41
+ expect(candidate.dqlSource).toContain('block "Revenue By Region"');
42
+ expect(candidate.dqlSource).toContain('query = """');
43
+ expect(candidate.reviewStatus).toBe('draft');
44
+ expect(readFileSync(join(root, '.dql', 'imports', session.id, 'manifest.json'), 'utf-8')).toContain(candidate.id);
45
+ });
46
+ it('splits multi-statement SQL without breaking semicolons inside strings', () => {
47
+ const root = tempProject();
48
+ writeFileSync(join(root, 'legacy.sql'), `
49
+ select 'a;b' as label, count(*) as n from raw.events;
50
+ select region, count(*) as n from raw.accounts group by region;
51
+ `);
52
+ const session = createBlockStudioImportSession(root, {
53
+ inputPath: 'legacy.sql',
54
+ domain: 'ops',
55
+ });
56
+ expect(session.candidates).toHaveLength(2);
57
+ expect(session.candidates[0].sql).toContain("'a;b'");
58
+ expect(session.candidates[0].lineage.sourceTables).toEqual(['raw.events']);
59
+ expect(session.candidates[1].lineage.sourceTables).toEqual(['raw.accounts']);
60
+ expect(session.candidates[0].lineage.totalStatements).toBe(2);
61
+ });
62
+ it('imports every SQL file in a folder and reloads the persisted session', () => {
63
+ const root = tempProject();
64
+ mkdirSync(join(root, 'queries'));
65
+ writeFileSync(join(root, 'queries', 'a.sql'), 'select * from source_a;');
66
+ writeFileSync(join(root, 'queries', 'b.sql'), 'select * from source_b;');
67
+ const session = createBlockStudioImportSession(root, {
68
+ inputPath: 'queries',
69
+ domain: 'shared reporting',
70
+ });
71
+ const reloaded = loadBlockStudioImportSession(root, session.id);
72
+ expect(reloaded.sourceKind).toBe('raw-sql-folder');
73
+ expect(reloaded.candidates.map((candidate) => candidate.lineage.sourceTables[0]).sort()).toEqual(['source_a', 'source_b']);
74
+ expect(reloaded.defaults.domain).toBe('shared-reporting');
75
+ });
76
+ it('detects parameters and refreshes generated DQL when a candidate is updated', () => {
77
+ const root = tempProject();
78
+ writeFileSync(join(root, 'parameterized.sql'), 'select * from orders where region = :region and dt >= {{ start_date }};');
79
+ const session = createBlockStudioImportSession(root, { inputPath: 'parameterized.sql' });
80
+ expect(session.candidates[0].lineage.parameters.sort()).toEqual(['region', 'start_date']);
81
+ const updated = updateBlockStudioImportCandidate(root, session.id, session.candidates[0].id, {
82
+ name: 'Orders Filtered',
83
+ domain: 'sales',
84
+ sql: 'select * from orders where region = :region;',
85
+ });
86
+ expect(updated.name).toBe('Orders Filtered');
87
+ expect(updated.domain).toBe('sales');
88
+ expect(updated.dqlSource).toContain('block "Orders Filtered"');
89
+ expect(updated.lineage.parameters).toEqual(['region']);
90
+ });
91
+ it('generates a valid DQL block shape from a candidate', () => {
92
+ const source = candidateToDqlSource({
93
+ name: 'Legacy Query',
94
+ domain: 'finance',
95
+ description: 'Imported legacy SQL',
96
+ owner: 'owner',
97
+ tags: ['imported', 'raw-sql'],
98
+ sql: 'select * from finance.orders',
99
+ });
100
+ expect(source).toContain('domain = "finance"');
101
+ expect(source).toContain('tags = ["imported", "raw-sql"]');
102
+ expect(source).toContain('visualization {');
103
+ expect(source).toContain('chart = "table"');
104
+ });
105
+ });
106
+ //# sourceMappingURL=block-studio-import.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"block-studio-import.test.js","sourceRoot":"","sources":["../src/block-studio-import.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface BlockTemplate {
2
+ id: string;
3
+ name: string;
4
+ description: string;
5
+ content: string;
6
+ }
7
+ export declare function listBlockTemplates(): BlockTemplate[];
8
+ //# sourceMappingURL=block-templates.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"block-templates.d.ts","sourceRoot":"","sources":["../src/block-templates.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC5B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,MAAM,CAAC;CACjB;AA2DD,wBAAgB,kBAAkB,IAAI,aAAa,EAAE,CAEpD"}
@@ -0,0 +1,60 @@
1
+ const BLOCK_TEMPLATES = [
2
+ {
3
+ id: 'metric-report',
4
+ name: 'Metric Report',
5
+ description: 'Single metric query with semantic references and ordering.',
6
+ content: `SELECT
7
+ @metric(total_revenue) AS total_revenue
8
+ ORDER BY 1 DESC
9
+ LIMIT 50;
10
+ `,
11
+ },
12
+ {
13
+ id: 'dimension-analysis',
14
+ name: 'Dimension Analysis',
15
+ description: 'Group a metric by a business dimension.',
16
+ content: `SELECT
17
+ @dim(segment) AS segment,
18
+ @metric(total_revenue) AS total_revenue
19
+ GROUP BY 1
20
+ ORDER BY 2 DESC
21
+ LIMIT 25;
22
+ `,
23
+ },
24
+ {
25
+ id: 'time-series',
26
+ name: 'Time Series',
27
+ description: 'Time-based trend using a semantic date dimension.',
28
+ content: `SELECT
29
+ @dim(order_date) AS order_date,
30
+ @metric(total_revenue) AS total_revenue
31
+ GROUP BY 1
32
+ ORDER BY 1
33
+ LIMIT 365;
34
+ `,
35
+ },
36
+ {
37
+ id: 'kpi-dashboard',
38
+ name: 'KPI Dashboard',
39
+ description: 'Multiple KPI snapshot query.',
40
+ content: `SELECT
41
+ @metric(total_revenue) AS total_revenue,
42
+ @metric(total_orders) AS total_orders,
43
+ @metric(avg_order_value) AS avg_order_value;
44
+ `,
45
+ },
46
+ {
47
+ id: 'data-quality',
48
+ name: 'Data Quality',
49
+ description: 'Simple assertion-style query for row quality checks.',
50
+ content: `SELECT
51
+ COUNT(*) AS row_count,
52
+ COUNT(*) FILTER (WHERE @dim(segment) IS NULL) AS null_segment_rows
53
+ FROM orders;
54
+ `,
55
+ },
56
+ ];
57
+ export function listBlockTemplates() {
58
+ return BLOCK_TEMPLATES;
59
+ }
60
+ //# sourceMappingURL=block-templates.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"block-templates.js","sourceRoot":"","sources":["../src/block-templates.ts"],"names":[],"mappings":"AAOA,MAAM,eAAe,GAAoB;IACvC;QACE,EAAE,EAAE,eAAe;QACnB,IAAI,EAAE,eAAe;QACrB,WAAW,EAAE,4DAA4D;QACzE,OAAO,EAAE;;;;CAIZ;KACE;IACD;QACE,EAAE,EAAE,oBAAoB;QACxB,IAAI,EAAE,oBAAoB;QAC1B,WAAW,EAAE,yCAAyC;QACtD,OAAO,EAAE;;;;;;CAMZ;KACE;IACD;QACE,EAAE,EAAE,aAAa;QACjB,IAAI,EAAE,aAAa;QACnB,WAAW,EAAE,mDAAmD;QAChE,OAAO,EAAE;;;;;;CAMZ;KACE;IACD;QACE,EAAE,EAAE,eAAe;QACnB,IAAI,EAAE,eAAe;QACrB,WAAW,EAAE,8BAA8B;QAC3C,OAAO,EAAE;;;;CAIZ;KACE;IACD;QACE,EAAE,EAAE,cAAc;QAClB,IAAI,EAAE,cAAc;QACpB,WAAW,EAAE,sDAAsD;QACnE,OAAO,EAAE;;;;CAIZ;KACE;CACF,CAAC;AAEF,MAAM,UAAU,kBAAkB;IAChC,OAAO,eAAe,CAAC;AACzB,CAAC"}
@@ -0,0 +1,19 @@
1
+ /**
2
+ * `dql agent` — block-first answer loop on the command line.
3
+ *
4
+ * dql agent ask "what was revenue last week?"
5
+ * [--provider claude|openai|gemini|ollama]
6
+ * [--user alice@acme.com] (filters Skills + records feedback as this user)
7
+ * [--domain growth] (scopes KG search)
8
+ * [--format json] (emits structured JSON instead of prose)
9
+ *
10
+ * dql agent reindex
11
+ * Rebuilds .dql/cache/agent-kg.sqlite from the project's manifest +
12
+ * Skills folder. Equivalent to `dql app reindex`.
13
+ *
14
+ * dql agent feedback <up|down> --block <id> --question "..."
15
+ * Records feedback into the KG. Used by clients without MCP access.
16
+ */
17
+ import type { CLIFlags } from '../args.js';
18
+ export declare function runAgent(sub: string | null, rest: string[], flags: CLIFlags): Promise<void>;
19
+ //# sourceMappingURL=agent.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"agent.d.ts","sourceRoot":"","sources":["../../src/commands/agent.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAiBH,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAG3C,wBAAsB,QAAQ,CAC5B,GAAG,EAAE,MAAM,GAAG,IAAI,EAClB,IAAI,EAAE,MAAM,EAAE,EACd,KAAK,EAAE,QAAQ,GACd,OAAO,CAAC,IAAI,CAAC,CAmBf"}