simplemdg-dev-cli 2.0.4 → 2.4.5

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 (148) hide show
  1. package/README.md +62 -354
  2. package/USER_GUIDE.md +55 -376
  3. package/dist/commands/cds.command.js +69 -60
  4. package/dist/commands/cds.command.js.map +1 -1
  5. package/dist/commands/cf-db.command.d.ts +2 -0
  6. package/dist/commands/cf-db.command.js +606 -0
  7. package/dist/commands/cf-db.command.js.map +1 -0
  8. package/dist/commands/cf.command.js +291 -280
  9. package/dist/commands/cf.command.js.map +1 -1
  10. package/dist/commands/gitlab.command.d.ts +2 -0
  11. package/dist/commands/gitlab.command.js +351 -0
  12. package/dist/commands/gitlab.command.js.map +1 -0
  13. package/dist/commands/npmrc.command.js +50 -44
  14. package/dist/commands/npmrc.command.js.map +1 -1
  15. package/dist/core/cache.d.ts +1 -1
  16. package/dist/core/cache.js +58 -31
  17. package/dist/core/cache.js.map +1 -1
  18. package/dist/core/cds.js +32 -22
  19. package/dist/core/cds.js.map +1 -1
  20. package/dist/core/cf-env-parser.d.ts +1 -1
  21. package/dist/core/cf-env-parser.js +4 -1
  22. package/dist/core/cf-env-parser.js.map +1 -1
  23. package/dist/core/cf.d.ts +1 -1
  24. package/dist/core/cf.js +46 -31
  25. package/dist/core/cf.js.map +1 -1
  26. package/dist/core/db/db-btp.d.ts +48 -0
  27. package/dist/core/db/db-btp.js +162 -0
  28. package/dist/core/db/db-btp.js.map +1 -0
  29. package/dist/core/db/db-cache.d.ts +40 -0
  30. package/dist/core/db/db-cache.js +188 -0
  31. package/dist/core/db/db-cache.js.map +1 -0
  32. package/dist/core/db/db-connection.d.ts +22 -0
  33. package/dist/core/db/db-connection.js +73 -0
  34. package/dist/core/db/db-connection.js.map +1 -0
  35. package/dist/core/db/db-crypto.d.ts +3 -0
  36. package/dist/core/db/db-crypto.js +54 -0
  37. package/dist/core/db/db-crypto.js.map +1 -0
  38. package/dist/core/db/db-hana-adapter.d.ts +36 -0
  39. package/dist/core/db/db-hana-adapter.js +251 -0
  40. package/dist/core/db/db-hana-adapter.js.map +1 -0
  41. package/dist/core/db/db-metadata.d.ts +25 -0
  42. package/dist/core/db/db-metadata.js +150 -0
  43. package/dist/core/db/db-metadata.js.map +1 -0
  44. package/dist/core/db/db-postgres-adapter.d.ts +34 -0
  45. package/dist/core/db/db-postgres-adapter.js +259 -0
  46. package/dist/core/db/db-postgres-adapter.js.map +1 -0
  47. package/dist/core/db/db-query-files.d.ts +20 -0
  48. package/dist/core/db/db-query-files.js +106 -0
  49. package/dist/core/db/db-query-files.js.map +1 -0
  50. package/dist/core/db/db-query-history.d.ts +5 -0
  51. package/dist/core/db/db-query-history.js +49 -0
  52. package/dist/core/db/db-query-history.js.map +1 -0
  53. package/dist/core/db/db-row.d.ts +28 -0
  54. package/dist/core/db/db-row.js +123 -0
  55. package/dist/core/db/db-row.js.map +1 -0
  56. package/dist/core/db/db-studio-client.d.ts +1 -0
  57. package/dist/core/db/db-studio-client.js +401 -0
  58. package/dist/core/db/db-studio-client.js.map +1 -0
  59. package/dist/core/db/db-studio-html.d.ts +4 -0
  60. package/dist/core/db/db-studio-html.js +83 -0
  61. package/dist/core/db/db-studio-html.js.map +1 -0
  62. package/dist/core/db/db-studio-server.d.ts +11 -0
  63. package/dist/core/db/db-studio-server.js +528 -0
  64. package/dist/core/db/db-studio-server.js.map +1 -0
  65. package/dist/core/db/db-studio-styles.d.ts +1 -0
  66. package/dist/core/db/db-studio-styles.js +225 -0
  67. package/dist/core/db/db-studio-styles.js.map +1 -0
  68. package/dist/core/db/db-types.d.ts +214 -0
  69. package/dist/core/db/db-types.js +3 -0
  70. package/dist/core/db/db-types.js.map +1 -0
  71. package/dist/core/db/db-vcap-parser.d.ts +7 -0
  72. package/dist/core/db/db-vcap-parser.js +137 -0
  73. package/dist/core/db/db-vcap-parser.js.map +1 -0
  74. package/dist/core/doctor.d.ts +1 -1
  75. package/dist/core/doctor.js +14 -8
  76. package/dist/core/doctor.js.map +1 -1
  77. package/dist/core/guide.js +31 -26
  78. package/dist/core/guide.js.map +1 -1
  79. package/dist/core/install.d.ts +1 -1
  80. package/dist/core/install.js +17 -11
  81. package/dist/core/install.js.map +1 -1
  82. package/dist/core/navigator.d.ts +17 -0
  83. package/dist/core/navigator.js +140 -0
  84. package/dist/core/navigator.js.map +1 -0
  85. package/dist/core/npmrc.js +29 -16
  86. package/dist/core/npmrc.js.map +1 -1
  87. package/dist/core/process.js +11 -6
  88. package/dist/core/process.js.map +1 -1
  89. package/dist/core/prompts.js +16 -8
  90. package/dist/core/prompts.js.map +1 -1
  91. package/dist/core/repository.d.ts +1 -1
  92. package/dist/core/repository.js +16 -9
  93. package/dist/core/repository.js.map +1 -1
  94. package/dist/core/scanner.d.ts +1 -1
  95. package/dist/core/scanner.js +13 -7
  96. package/dist/core/scanner.js.map +1 -1
  97. package/dist/core/tooling.d.ts +28 -0
  98. package/dist/core/tooling.js +168 -0
  99. package/dist/core/tooling.js.map +1 -0
  100. package/dist/core/types.js +2 -1
  101. package/dist/core/version-conflict.d.ts +2 -2
  102. package/dist/core/version-conflict.js +11 -6
  103. package/dist/core/version-conflict.js.map +1 -1
  104. package/dist/index.js +65 -48
  105. package/dist/index.js.map +1 -1
  106. package/dist/types-local.js +2 -1
  107. package/package.json +12 -6
  108. package/src/commands/cds.command.ts +529 -0
  109. package/src/commands/cf-db.command.ts +636 -0
  110. package/src/commands/cf.command.ts +3345 -0
  111. package/src/commands/gitlab.command.ts +373 -0
  112. package/src/commands/npmrc.command.ts +581 -0
  113. package/src/core/cache.ts +332 -0
  114. package/src/core/cds.ts +278 -0
  115. package/src/core/cf-env-parser.ts +131 -0
  116. package/src/core/cf.ts +271 -0
  117. package/src/core/db/db-btp.ts +207 -0
  118. package/src/core/db/db-cache.ts +242 -0
  119. package/src/core/db/db-connection.ts +79 -0
  120. package/src/core/db/db-crypto.ts +53 -0
  121. package/src/core/db/db-hana-adapter.ts +306 -0
  122. package/src/core/db/db-metadata.ts +174 -0
  123. package/src/core/db/db-postgres-adapter.ts +293 -0
  124. package/src/core/db/db-query-files.ts +130 -0
  125. package/src/core/db/db-query-history.ts +53 -0
  126. package/src/core/db/db-row.ts +157 -0
  127. package/src/core/db/db-studio-client.ts +397 -0
  128. package/src/core/db/db-studio-html.ts +85 -0
  129. package/src/core/db/db-studio-server.ts +626 -0
  130. package/src/core/db/db-studio-styles.ts +221 -0
  131. package/src/core/db/db-types.ts +243 -0
  132. package/src/core/db/db-vcap-parser.ts +182 -0
  133. package/src/core/doctor.ts +70 -0
  134. package/src/core/guide.ts +261 -0
  135. package/src/core/install.ts +91 -0
  136. package/src/core/navigator.ts +164 -0
  137. package/src/core/npmrc.ts +171 -0
  138. package/src/core/process.ts +75 -0
  139. package/src/core/prompts.ts +225 -0
  140. package/src/core/repository.ts +36 -0
  141. package/src/core/scanner.ts +41 -0
  142. package/src/core/tooling.ts +207 -0
  143. package/src/core/types.ts +152 -0
  144. package/src/core/version-conflict.ts +46 -0
  145. package/src/index.ts +460 -0
  146. package/src/types/external.d.ts +3 -0
  147. package/src/types-local.ts +11 -0
  148. package/tsconfig.json +17 -0
@@ -0,0 +1,528 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.startStudioServer = startStudioServer;
7
+ const node_http_1 = __importDefault(require("node:http"));
8
+ const node_net_1 = __importDefault(require("node:net"));
9
+ const execa_1 = require("execa");
10
+ const chalk_1 = __importDefault(require("chalk"));
11
+ const db_studio_html_1 = require("./db-studio-html");
12
+ const db_connection_1 = require("./db-connection");
13
+ const db_cache_1 = require("./db-cache");
14
+ const db_connection_2 = require("./db-connection");
15
+ const db_metadata_1 = require("./db-metadata");
16
+ const db_query_files_1 = require("./db-query-files");
17
+ const db_row_1 = require("./db-row");
18
+ const db_query_history_1 = require("./db-query-history");
19
+ const db_btp_1 = require("./db-btp");
20
+ function isPortAvailable(port) {
21
+ return new Promise((resolve) => {
22
+ const tester = node_net_1.default.createServer();
23
+ tester.once("error", () => resolve(false));
24
+ tester.once("listening", () => tester.close(() => resolve(true)));
25
+ tester.listen(port, "127.0.0.1");
26
+ });
27
+ }
28
+ async function findAvailablePort(preferredPort) {
29
+ for (let candidate = preferredPort; candidate < preferredPort + 50; candidate += 1) {
30
+ if (await isPortAvailable(candidate)) {
31
+ return candidate;
32
+ }
33
+ }
34
+ throw new Error(`No available port found between ${preferredPort} and ${preferredPort + 49}`);
35
+ }
36
+ async function readJsonBody(req) {
37
+ const chunks = [];
38
+ for await (const chunk of req) {
39
+ chunks.push(Buffer.from(chunk));
40
+ }
41
+ const raw = Buffer.concat(chunks).toString("utf8").trim();
42
+ if (!raw) {
43
+ return {};
44
+ }
45
+ return JSON.parse(raw);
46
+ }
47
+ function sendJson(res, value, status = 200) {
48
+ const payload = JSON.stringify(value);
49
+ res.writeHead(status, { "content-type": "application/json; charset=utf-8" });
50
+ res.end(payload);
51
+ }
52
+ function sendText(res, value, contentType, fileName) {
53
+ const headers = { "content-type": contentType };
54
+ if (fileName) {
55
+ headers["content-disposition"] = `attachment; filename="${fileName}"`;
56
+ }
57
+ res.writeHead(200, headers);
58
+ res.end(value);
59
+ }
60
+ function toCsv(fields, rows) {
61
+ const escapeCell = (value) => {
62
+ const text = value === null || value === undefined ? "" : typeof value === "object" ? JSON.stringify(value) : String(value);
63
+ return `"${text.replace(/"/g, '""')}"`;
64
+ };
65
+ const header = fields.map(escapeCell).join(",");
66
+ const lines = rows.map((row) => fields.map((field) => escapeCell(row[field])).join(","));
67
+ return [header, ...lines].join("\n");
68
+ }
69
+ function getString(body, key) {
70
+ const value = body[key];
71
+ return typeof value === "string" ? value : "";
72
+ }
73
+ function getNumber(body, key, fallback) {
74
+ const value = Number(body[key]);
75
+ return Number.isFinite(value) ? value : fallback;
76
+ }
77
+ const VALID_ENVIRONMENTS = new Set(["DEV", "QAS", "PROD", "SANDBOX", "CUSTOM"]);
78
+ function getEnvironment(body) {
79
+ const value = getString(body, "environment").toUpperCase();
80
+ return VALID_ENVIRONMENTS.has(value) ? value : undefined;
81
+ }
82
+ function draftFromBody(body) {
83
+ const type = getString(body, "type") === "hana" ? "hana" : "postgresql";
84
+ return {
85
+ name: getString(body, "name") || `${type} connection`,
86
+ color: getString(body, "color") || undefined,
87
+ environment: getEnvironment(body),
88
+ isFavorite: body.isFavorite === undefined ? undefined : Boolean(body.isFavorite),
89
+ type,
90
+ host: getString(body, "host"),
91
+ port: getNumber(body, "port", type === "hana" ? 443 : 5432),
92
+ database: getString(body, "database") || undefined,
93
+ schema: getString(body, "schema") || undefined,
94
+ username: getString(body, "username"),
95
+ password: getString(body, "password"),
96
+ ssl: body.ssl === undefined ? true : Boolean(body.ssl),
97
+ sslValidateCertificate: Boolean(body.sslValidateCertificate),
98
+ tags: Array.isArray(body.tags) ? body.tags : undefined,
99
+ };
100
+ }
101
+ function getObject(body, key) {
102
+ const value = body[key];
103
+ return value && typeof value === "object" && !Array.isArray(value) ? value : {};
104
+ }
105
+ function resolvedFromDraft(draft) {
106
+ const now = new Date().toISOString();
107
+ return {
108
+ id: "draft",
109
+ name: draft.name,
110
+ type: draft.type,
111
+ host: draft.host,
112
+ port: draft.port,
113
+ database: draft.database,
114
+ schema: draft.schema,
115
+ username: draft.username,
116
+ password: draft.password,
117
+ ssl: draft.ssl,
118
+ sslValidateCertificate: draft.sslValidateCertificate,
119
+ createdAt: now,
120
+ updatedAt: now,
121
+ };
122
+ }
123
+ async function openBrowser(url) {
124
+ const command = process.platform === "win32" ? "cmd" : process.platform === "darwin" ? "open" : "xdg-open";
125
+ const args = process.platform === "win32" ? ["/c", "start", "", url] : [url];
126
+ await (0, execa_1.execa)(command, args, { reject: false, detached: true, stdio: "ignore" }).catch(() => undefined);
127
+ }
128
+ async function startStudioServer(options = {}) {
129
+ const preferredPort = options.port && options.port > 0 ? options.port : 45888;
130
+ const port = await findAvailablePort(preferredPort);
131
+ const pool = new db_connection_1.StudioConnectionPool({ queryTimeoutMs: options.queryTimeoutMs });
132
+ const serverReadOnlyDefault = options.readOnly ?? false;
133
+ const router = async (req, res) => {
134
+ const url = new URL(req.url ?? "/", `http://127.0.0.1:${port}`);
135
+ const pathname = url.pathname;
136
+ const method = req.method ?? "GET";
137
+ if (pathname === "/" && method === "GET") {
138
+ res.writeHead(200, { "content-type": "text/html; charset=utf-8" });
139
+ res.end((0, db_studio_html_1.renderStudioHtml)({ readOnlyDefault: serverReadOnlyDefault }));
140
+ return;
141
+ }
142
+ // --- Connections ---------------------------------------------------------
143
+ if (pathname === "/api/connections" && method === "GET") {
144
+ sendJson(res, { connections: await (0, db_cache_1.listPublicConnections)() });
145
+ return;
146
+ }
147
+ if (pathname === "/api/connections/test" && method === "POST") {
148
+ const body = await readJsonBody(req);
149
+ const resolved = await (0, db_cache_1.getResolvedConnection)(getString(body, "connectionId"));
150
+ const result = await (0, db_connection_2.testConnectionProfile)(resolved, { queryTimeoutMs: options.queryTimeoutMs });
151
+ sendJson(res, result);
152
+ return;
153
+ }
154
+ if (pathname === "/api/connections/test-draft" && method === "POST") {
155
+ const body = await readJsonBody(req);
156
+ const result = await (0, db_connection_2.testConnectionProfile)(resolvedFromDraft(draftFromBody(body)), { queryTimeoutMs: options.queryTimeoutMs });
157
+ sendJson(res, result);
158
+ return;
159
+ }
160
+ if (pathname === "/api/connections/create" && method === "POST") {
161
+ const body = await readJsonBody(req);
162
+ const profile = await (0, db_cache_1.upsertConnectionFromDraft)(draftFromBody(body));
163
+ const { encryptedPassword: _omit, ...publicProfile } = profile;
164
+ void _omit;
165
+ sendJson(res, { connection: publicProfile });
166
+ return;
167
+ }
168
+ if (pathname === "/api/connections/rename" && method === "POST") {
169
+ const body = await readJsonBody(req);
170
+ const profile = await (0, db_cache_1.renameConnection)(getString(body, "id"), getString(body, "name"));
171
+ sendJson(res, { id: profile.id, name: profile.name });
172
+ return;
173
+ }
174
+ if (pathname === "/api/connections/update" && method === "POST") {
175
+ const body = await readJsonBody(req);
176
+ const patch = {};
177
+ if (typeof body.name === "string")
178
+ patch.name = body.name;
179
+ if (typeof body.color === "string")
180
+ patch.color = body.color;
181
+ if (body.environment !== undefined)
182
+ patch.environment = getEnvironment(body);
183
+ if (body.isFavorite !== undefined)
184
+ patch.isFavorite = Boolean(body.isFavorite);
185
+ if (Array.isArray(body.tags))
186
+ patch.tags = body.tags;
187
+ const profile = await (0, db_cache_1.updateConnectionFields)(getString(body, "id"), patch);
188
+ const { encryptedPassword: _omit, ...publicProfile } = profile;
189
+ void _omit;
190
+ sendJson(res, { connection: publicProfile });
191
+ return;
192
+ }
193
+ if (pathname === "/api/connections/duplicate" && method === "POST") {
194
+ const body = await readJsonBody(req);
195
+ const profile = await (0, db_cache_1.duplicateConnection)(getString(body, "id"));
196
+ sendJson(res, { id: profile.id, name: profile.name });
197
+ return;
198
+ }
199
+ if (pathname === "/api/connections/remove" && method === "POST") {
200
+ const body = await readJsonBody(req);
201
+ await pool.closeConnection(getString(body, "id"));
202
+ const removed = await (0, db_cache_1.removeConnection)(getString(body, "id"));
203
+ sendJson(res, { removed });
204
+ return;
205
+ }
206
+ if (pathname === "/api/connections/import-from-app" && method === "POST") {
207
+ const body = await readJsonBody(req);
208
+ const { profile } = await (0, db_btp_1.importConnectionFromApp)({
209
+ app: getString(body, "app"),
210
+ serviceName: getString(body, "serviceName") || undefined,
211
+ type: (getString(body, "type") || undefined),
212
+ });
213
+ const { encryptedPassword: _omitPassword, ...publicProfile } = profile;
214
+ void _omitPassword;
215
+ sendJson(res, { connection: publicProfile });
216
+ return;
217
+ }
218
+ // --- BTP -----------------------------------------------------------------
219
+ if (pathname === "/api/btp/current-target" && method === "GET") {
220
+ const session = await (0, db_btp_1.ensureCloudFoundrySession)();
221
+ const target = await (0, db_btp_1.getCloudFoundryTargetSummary)();
222
+ sendJson(res, {
223
+ loggedIn: session.loggedIn,
224
+ message: session.message,
225
+ target,
226
+ productionWarning: (0, db_metadata_1.looksLikeProduction)(target.org, target.space),
227
+ });
228
+ return;
229
+ }
230
+ if (pathname === "/api/btp/apps" && method === "GET") {
231
+ const session = await (0, db_btp_1.ensureCloudFoundrySession)();
232
+ if (!session.loggedIn) {
233
+ sendJson(res, { loggedIn: false, message: session.message, apps: [] });
234
+ return;
235
+ }
236
+ const apps = await (0, db_btp_1.listCloudFoundryAppsWithCache)({ refresh: url.searchParams.get("refresh") === "true" });
237
+ sendJson(res, { loggedIn: true, apps });
238
+ return;
239
+ }
240
+ if (pathname === "/api/btp/env" && method === "POST") {
241
+ const body = await readJsonBody(req);
242
+ const candidates = await (0, db_btp_1.detectAppDatabaseServices)(getString(body, "app"));
243
+ // Never expose passwords to the browser.
244
+ const safeCandidates = candidates.map(({ password: _password, ...rest }) => {
245
+ void _password;
246
+ return rest;
247
+ });
248
+ sendJson(res, { services: safeCandidates });
249
+ return;
250
+ }
251
+ // --- Catalog -------------------------------------------------------------
252
+ if (pathname === "/api/catalog/schemas" && method === "GET") {
253
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
254
+ sendJson(res, { schemas: await adapter.listSchemas() });
255
+ return;
256
+ }
257
+ if (pathname === "/api/catalog/objects" && method === "GET") {
258
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
259
+ const kindsParam = url.searchParams.get("kinds");
260
+ const kinds = kindsParam ? kindsParam.split(",").filter(Boolean) : undefined;
261
+ const objects = await adapter.listObjects({
262
+ schema: url.searchParams.get("schema") ?? undefined,
263
+ search: url.searchParams.get("search") ?? undefined,
264
+ kinds,
265
+ });
266
+ sendJson(res, { objects });
267
+ return;
268
+ }
269
+ if (pathname === "/api/catalog/columns" && method === "GET") {
270
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
271
+ const schema = url.searchParams.get("schema") ?? "";
272
+ const table = url.searchParams.get("table") ?? "";
273
+ const [columns, indexes] = await Promise.all([
274
+ adapter.listColumns(schema, table),
275
+ adapter.listIndexes(schema, table).catch(() => []),
276
+ ]);
277
+ sendJson(res, { columns, indexes });
278
+ return;
279
+ }
280
+ if (pathname === "/api/catalog/ddl" && method === "GET") {
281
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
282
+ const schema = url.searchParams.get("schema") ?? "";
283
+ const table = url.searchParams.get("table") ?? "";
284
+ const columns = await adapter.listColumns(schema, table);
285
+ sendJson(res, { ddl: (0, db_metadata_1.generateCreateTableDdl)(adapter.type, schema, table, columns) });
286
+ return;
287
+ }
288
+ if (pathname === "/api/catalog/indexes" && method === "GET") {
289
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
290
+ sendJson(res, { indexes: await adapter.listIndexes(url.searchParams.get("schema") ?? "", url.searchParams.get("table") ?? "") });
291
+ return;
292
+ }
293
+ if (pathname === "/api/catalog/primary-key" && method === "GET") {
294
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
295
+ const primaryKey = await adapter.getPrimaryKey(url.searchParams.get("schema") ?? "", url.searchParams.get("table") ?? "");
296
+ sendJson(res, { primaryKey });
297
+ return;
298
+ }
299
+ if (pathname === "/api/catalog/constraints" && method === "GET") {
300
+ const adapter = await pool.getAdapter(url.searchParams.get("connectionId") ?? "");
301
+ const schema = url.searchParams.get("schema") ?? "";
302
+ const table = url.searchParams.get("table") ?? "";
303
+ const [primaryKey, indexes] = await Promise.all([
304
+ adapter.getPrimaryKey(schema, table),
305
+ adapter.listIndexes(schema, table).catch(() => []),
306
+ ]);
307
+ sendJson(res, { primaryKey, indexes });
308
+ return;
309
+ }
310
+ // --- Table data ----------------------------------------------------------
311
+ if (pathname === "/api/table/data" && method === "POST") {
312
+ const body = await readJsonBody(req);
313
+ const adapter = await pool.getAdapter(getString(body, "connectionId"));
314
+ const result = await adapter.getTableData({
315
+ schema: getString(body, "schema"),
316
+ table: getString(body, "table"),
317
+ limit: getNumber(body, "limit", 100),
318
+ offset: getNumber(body, "offset", 0),
319
+ where: getString(body, "where") || undefined,
320
+ orderBy: getString(body, "orderBy") || undefined,
321
+ orderDirection: getString(body, "orderDirection") === "desc" ? "desc" : "asc",
322
+ });
323
+ sendJson(res, { result });
324
+ return;
325
+ }
326
+ if (pathname === "/api/table/count" && method === "POST") {
327
+ const body = await readJsonBody(req);
328
+ const adapter = await pool.getAdapter(getString(body, "connectionId"));
329
+ const count = await adapter.countRows(getString(body, "schema"), getString(body, "table"));
330
+ sendJson(res, { count });
331
+ return;
332
+ }
333
+ if ((pathname === "/api/table/row/update" || pathname === "/api/table/row/insert" || pathname === "/api/table/row/delete") && method === "POST") {
334
+ const body = await readJsonBody(req);
335
+ const readOnly = body.readOnly === undefined ? serverReadOnlyDefault : Boolean(body.readOnly);
336
+ if (readOnly) {
337
+ sendJson(res, { ok: false, blocked: true, error: "Read-only mode is on. Turn it off to modify data." });
338
+ return;
339
+ }
340
+ const adapter = await pool.getAdapter(getString(body, "connectionId"));
341
+ const schema = getString(body, "schema");
342
+ const table = getString(body, "table");
343
+ try {
344
+ let result;
345
+ if (pathname.endsWith("/update")) {
346
+ result = await (0, db_row_1.updateRow)(adapter, { schema, table, changes: getObject(body, "changes"), keys: getObject(body, "keys") });
347
+ }
348
+ else if (pathname.endsWith("/insert")) {
349
+ result = await (0, db_row_1.insertRow)(adapter, { schema, table, values: getObject(body, "values") });
350
+ }
351
+ else {
352
+ result = await (0, db_row_1.deleteRow)(adapter, { schema, table, keys: getObject(body, "keys") });
353
+ }
354
+ sendJson(res, { ok: true, result });
355
+ }
356
+ catch (error) {
357
+ sendJson(res, { ok: false, error: error instanceof Error ? error.message : String(error) });
358
+ }
359
+ return;
360
+ }
361
+ if (pathname === "/api/table/save-changes" && method === "POST") {
362
+ const body = await readJsonBody(req);
363
+ const readOnly = body.readOnly === undefined ? serverReadOnlyDefault : Boolean(body.readOnly);
364
+ if (readOnly) {
365
+ sendJson(res, { ok: false, blocked: true, error: "Read-only mode is on. Turn it off to save changes." });
366
+ return;
367
+ }
368
+ const adapter = await pool.getAdapter(getString(body, "connectionId"));
369
+ const result = await (0, db_row_1.saveTableChanges)(adapter, {
370
+ schema: getString(body, "schema"),
371
+ table: getString(body, "table"),
372
+ primaryKeyColumns: Array.isArray(body.primaryKeyColumns) ? body.primaryKeyColumns : [],
373
+ updates: Array.isArray(body.updates) ? body.updates : [],
374
+ inserts: Array.isArray(body.inserts) ? body.inserts : [],
375
+ deletes: Array.isArray(body.deletes) ? body.deletes : [],
376
+ });
377
+ sendJson(res, { ok: true, result });
378
+ return;
379
+ }
380
+ if (pathname === "/api/table/sql" && method === "POST") {
381
+ const body = await readJsonBody(req);
382
+ const adapter = await pool.getAdapter(getString(body, "connectionId"));
383
+ const schema = getString(body, "schema");
384
+ const table = getString(body, "table");
385
+ sendJson(res, {
386
+ select: (0, db_metadata_1.generateSelectSql)(adapter.type, schema, table, getNumber(body, "limit", 100)),
387
+ count: (0, db_metadata_1.generateCountSql)(adapter.type, schema, table),
388
+ });
389
+ return;
390
+ }
391
+ // --- Query run -----------------------------------------------------------
392
+ if (pathname === "/api/query/run" && method === "POST") {
393
+ const body = await readJsonBody(req);
394
+ const connectionId = getString(body, "connectionId");
395
+ const sql = getString(body, "sql");
396
+ const limit = getNumber(body, "limit", 0);
397
+ const readOnly = body.readOnly === undefined ? serverReadOnlyDefault : Boolean(body.readOnly);
398
+ const confirmDangerous = Boolean(body.confirmDangerous);
399
+ if (!connectionId) {
400
+ sendJson(res, { ok: false, error: "Select a connection first." });
401
+ return;
402
+ }
403
+ const safety = (0, db_metadata_1.analyzeSqlSafety)(sql, { readOnly });
404
+ if (safety.blockedByReadOnly) {
405
+ sendJson(res, { ok: false, blocked: true, safety, error: `Read-only mode blocks: ${safety.matchedKeywords.join(", ")}` });
406
+ return;
407
+ }
408
+ if (safety.isDestructive && !confirmDangerous) {
409
+ sendJson(res, { ok: false, needsConfirmation: true, safety });
410
+ return;
411
+ }
412
+ const connection = await (0, db_cache_1.getResolvedConnection)(connectionId).catch(() => undefined);
413
+ const adapter = await pool.getAdapter(connectionId);
414
+ const effectiveSql = (0, db_metadata_1.appendSafeLimit)(adapter.type, sql, limit);
415
+ try {
416
+ const result = await adapter.runQuery(effectiveSql, { maxRows: limit > 0 ? limit : undefined });
417
+ await (0, db_query_history_1.appendQueryHistory)({
418
+ connectionId,
419
+ connectionName: connection?.name,
420
+ connectionType: adapter.type,
421
+ sql,
422
+ durationMs: result.durationMs,
423
+ success: true,
424
+ rowCount: result.rowCount,
425
+ });
426
+ sendJson(res, { ok: true, result, safety, effectiveSql });
427
+ }
428
+ catch (error) {
429
+ const message = error instanceof Error ? error.message : String(error);
430
+ await (0, db_query_history_1.appendQueryHistory)({
431
+ connectionId,
432
+ connectionName: connection?.name,
433
+ connectionType: adapter.type,
434
+ sql,
435
+ durationMs: 0,
436
+ success: false,
437
+ error: message,
438
+ });
439
+ sendJson(res, { ok: false, error: message });
440
+ }
441
+ return;
442
+ }
443
+ // --- Saved queries -------------------------------------------------------
444
+ if (pathname === "/api/queries" && method === "GET") {
445
+ sendJson(res, { queries: await (0, db_query_files_1.listSavedQueries)() });
446
+ return;
447
+ }
448
+ if (pathname === "/api/queries" && method === "POST") {
449
+ const body = await readJsonBody(req);
450
+ const query = await (0, db_query_files_1.saveQuery)({
451
+ name: getString(body, "name"),
452
+ sql: getString(body, "sql"),
453
+ connectionId: getString(body, "connectionId") || undefined,
454
+ connectionType: (getString(body, "connectionType") || undefined),
455
+ tags: Array.isArray(body.tags) ? body.tags : undefined,
456
+ });
457
+ sendJson(res, { query });
458
+ return;
459
+ }
460
+ if (pathname.startsWith("/api/queries/") && method === "PUT") {
461
+ const id = decodeURIComponent(pathname.slice("/api/queries/".length));
462
+ const body = await readJsonBody(req);
463
+ const name = getString(body, "name");
464
+ const sql = getString(body, "sql");
465
+ const query = sql
466
+ ? await (0, db_query_files_1.saveQuery)({ id, name, sql, connectionId: getString(body, "connectionId") || undefined })
467
+ : await (0, db_query_files_1.renameSavedQuery)(id, name);
468
+ sendJson(res, { query });
469
+ return;
470
+ }
471
+ if (pathname.startsWith("/api/queries/") && method === "DELETE") {
472
+ const id = decodeURIComponent(pathname.slice("/api/queries/".length));
473
+ sendJson(res, { deleted: await (0, db_query_files_1.deleteSavedQuery)(id) });
474
+ return;
475
+ }
476
+ // --- History -------------------------------------------------------------
477
+ if (pathname === "/api/history" && method === "GET") {
478
+ sendJson(res, { history: await (0, db_query_history_1.listQueryHistory)(100) });
479
+ return;
480
+ }
481
+ // --- Export --------------------------------------------------------------
482
+ if (pathname === "/api/export/csv" && method === "POST") {
483
+ const body = await readJsonBody(req);
484
+ const fields = Array.isArray(body.fields) ? body.fields : [];
485
+ const rows = Array.isArray(body.rows) ? body.rows : [];
486
+ sendText(res, toCsv(fields, rows), "text/csv; charset=utf-8", "result.csv");
487
+ return;
488
+ }
489
+ if (pathname === "/api/export/json" && method === "POST") {
490
+ const body = await readJsonBody(req);
491
+ const rows = Array.isArray(body.rows) ? body.rows : [];
492
+ sendText(res, JSON.stringify(rows, null, 2), "application/json; charset=utf-8", "result.json");
493
+ return;
494
+ }
495
+ res.writeHead(404, { "content-type": "application/json; charset=utf-8" });
496
+ res.end(JSON.stringify({ error: "Not found" }));
497
+ };
498
+ const server = node_http_1.default.createServer((req, res) => {
499
+ router(req, res).catch((error) => {
500
+ const message = error instanceof Error ? error.message : String(error);
501
+ if (!res.headersSent) {
502
+ sendJson(res, { error: message }, 500);
503
+ }
504
+ else {
505
+ res.end();
506
+ }
507
+ });
508
+ });
509
+ await new Promise((resolve) => server.listen(port, "127.0.0.1", resolve));
510
+ const url = `http://127.0.0.1:${port}`;
511
+ console.log(chalk_1.default.green(`SimpleMDG CF DB Studio: ${url}`));
512
+ if (serverReadOnlyDefault) {
513
+ console.log(chalk_1.default.yellow("Read-only mode is ON. Write/DDL statements are blocked."));
514
+ }
515
+ console.log(chalk_1.default.gray("Server is bound to 127.0.0.1 only. Press Ctrl+C to stop."));
516
+ if (!process.env.SMDG_STUDIO_NO_OPEN) {
517
+ await openBrowser(url);
518
+ }
519
+ return {
520
+ url,
521
+ port,
522
+ close: async () => {
523
+ await pool.closeAll();
524
+ await new Promise((resolve) => server.close(() => resolve()));
525
+ },
526
+ };
527
+ }
528
+ //# sourceMappingURL=db-studio-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"db-studio-server.js","sourceRoot":"","sources":["../../../src/core/db/db-studio-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare const STUDIO_STYLES = "\n:root{\n --bg:#0b0f17;--bg-2:#0e1420;--bg-3:#121a28;--panel:#0d131e;--elev:#16203250;\n --border:#1f2c44;--border-2:#27374f;--text:#dce6f5;--muted:#8295b5;--faint:#5b6c8a;\n --accent:#3b82f6;--accent-2:#1d4ed8;--green:#22c55e;--red:#ef4444;--yellow:#f59e0b;\n --amber:#fbbf24;--chip:#1a2840;--sel:#16243d;--editbg:#3a2f08;--delbg:#3a1414;--insbg:#0f2e18;\n}\n*{box-sizing:border-box}\nhtml,body{height:100%;margin:0}\nbody{background:var(--bg);color:var(--text);font:13px/1.45 \"Segoe UI\",Roboto,Arial,sans-serif;overflow:hidden}\nbutton,input,select,textarea{font:inherit;color:inherit;outline:none}\n::-webkit-scrollbar{width:10px;height:10px}\n::-webkit-scrollbar-thumb{background:#22314c;border-radius:8px}\n::-webkit-scrollbar-track{background:transparent}\n.app{display:flex;flex-direction:column;height:100vh}\n.hidden{display:none!important}\n.spin{width:14px;height:14px;border:2px solid var(--border-2);border-top-color:var(--accent);border-radius:50%;display:inline-block;animation:sp .7s linear infinite;vertical-align:middle}\n@keyframes sp{to{transform:rotate(360deg)}}\nsvg.ic{width:15px;height:15px;vertical-align:-2px;fill:none;stroke:currentColor;stroke-width:1.7;stroke-linecap:round;stroke-linejoin:round}\n\n/* top bar */\n.topbar{display:flex;align-items:center;gap:9px;padding:8px 12px;background:linear-gradient(180deg,#101a2b,#0d1521);border-bottom:1px solid var(--border);flex:0 0 auto}\n.brand{font-weight:700;color:#cfe0ff;letter-spacing:.2px;white-space:nowrap}\n.brand .b2{color:var(--faint);font-weight:600}\n.badge{padding:3px 9px;border-radius:999px;background:var(--chip);border:1px solid var(--border);color:var(--muted);font-size:12px;white-space:nowrap;display:inline-flex;align-items:center;gap:5px}\n.badge.on{color:#cfe0ff;border-color:var(--accent)}\n.badge.hana{color:#7dd3fc;border-color:#0e7490}\n.badge.pg{color:#a5b4fc;border-color:#4338ca}\n.badge.ro{cursor:pointer}\n.badge.ro.active{color:#fff;background:#7c2d12;border-color:#b45309}\n.badge.prod{color:#fff;background:#7f1d1d;border-color:#b91c1c}\n.grow{flex:1}\n.top-search{flex:0 1 320px}\n.iconbtn{background:transparent;border:1px solid var(--border);border-radius:8px;padding:6px 8px;color:var(--muted);cursor:pointer;display:inline-flex;align-items:center;gap:6px}\n.iconbtn:hover{color:#cfe0ff;border-color:var(--border-2);background:var(--bg-3)}\n.iconbtn.primary{background:var(--accent);border-color:var(--accent-2);color:#fff}\n.iconbtn.primary:hover{filter:brightness(1.08);color:#fff}\n\n/* body */\n.body{display:flex;flex:1;min-height:0}\n.sidebar{width:320px;min-width:220px;max-width:560px;background:var(--panel);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow:hidden}\n.resizer{width:5px;cursor:col-resize}\n.resizer:hover{background:var(--accent)}\n.workspace{flex:1;display:flex;flex-direction:column;min-width:0}\n\n/* sidebar sections */\n.side-sec{display:flex;flex-direction:column;min-height:0;border-bottom:1px solid var(--border)}\n.side-sec.flex{flex:1}\n.side-head{display:flex;align-items:center;gap:7px;padding:9px 11px;cursor:pointer;color:var(--muted);user-select:none}\n.side-head:hover{color:#cfe0ff}\n.side-head .h-title{font-size:11px;text-transform:uppercase;letter-spacing:.7px;font-weight:700;flex:1}\n.side-head .chev{transition:transform .15s}\n.side-sec.collapsed .chev{transform:rotate(-90deg)}\n.side-sec.collapsed .side-body{display:none}\n.side-body{padding:0 9px 10px;overflow:auto;min-height:0}\n.side-sec.flex .side-body{flex:1}\n.side-actions{display:flex;gap:6px;margin-bottom:7px}\n\n/* search box */\n.searchbox{display:flex;align-items:center;gap:6px;background:var(--bg-3);border:1px solid var(--border);border-radius:8px;padding:5px 8px;margin-bottom:7px}\n.searchbox svg{color:var(--faint)}\n.searchbox input{flex:1;background:transparent;border:0;color:var(--text)}\n.searchbox .sbtn{background:transparent;border:0;color:var(--muted);cursor:pointer}\n\n/* buttons + inputs */\n.btn{background:var(--accent);border:1px solid var(--accent-2);color:#fff;border-radius:8px;padding:6px 11px;cursor:pointer}\n.btn:hover{filter:brightness(1.08)}\n.btn.sec{background:#22304a;border-color:var(--border-2);color:#cfe0ff}\n.btn.ghost{background:transparent;border-color:var(--border);color:var(--muted)}\n.btn.danger{background:#7f1d1d;border-color:#b91c1c;color:#fff}\n.btn.sm{padding:3px 8px;font-size:12px}\n.btn:disabled{opacity:.45;cursor:not-allowed;filter:none}\n.input,.select{width:100%;background:var(--bg-3);border:1px solid var(--border);border-radius:8px;padding:7px 9px;color:var(--text)}\n.field{display:flex;flex-direction:column;gap:4px;margin-bottom:9px}\n.field label{color:var(--muted);font-size:12px}\n.note{color:var(--muted);font-size:12px}\n.faint{color:var(--faint)}\n.row{display:flex;gap:7px;align-items:center;flex-wrap:wrap}\n.right{justify-content:flex-end}\n\n/* connection cards */\n.conn-card{position:relative;background:var(--bg-3);border:1px solid var(--border);border-left:3px solid var(--faint);border-radius:9px;padding:8px 10px;margin-bottom:7px;cursor:pointer}\n.conn-card:hover{border-color:var(--border-2);background:#152033}\n.conn-card.active{border-color:var(--accent);background:var(--sel)}\n.conn-top{display:flex;align-items:center;gap:7px}\n.conn-name{font-weight:600;flex:1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n.conn-sub{color:var(--muted);font-size:12px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;margin-top:2px}\n.conn-tags{display:flex;gap:5px;margin-top:5px;flex-wrap:wrap}\n.tag{font-size:10.5px;padding:1px 7px;border-radius:999px;background:var(--chip);border:1px solid var(--border);color:var(--muted)}\n.tag.env-DEV{color:#86efac;border-color:#166534}\n.tag.env-QAS{color:#fde68a;border-color:#a16207}\n.tag.env-PROD{color:#fecaca;border-color:#b91c1c}\n.tag.env-SANDBOX{color:#a5b4fc;border-color:#4338ca}\n.tag.type{color:#7dd3fc}\n.star{cursor:pointer;color:var(--faint)}\n.star.on{color:var(--amber)}\n.skel{height:56px;border-radius:9px;margin-bottom:7px;background:linear-gradient(90deg,#121a28,#1a2740,#121a28);background-size:200% 100%;animation:shimmer 1.3s infinite}\n@keyframes shimmer{0%{background-position:200% 0}100%{background-position:-200% 0}}\n\n/* tree */\n.tree{font-size:12.5px}\n.trow{display:flex;align-items:center;gap:5px;padding:3px 5px;border-radius:6px;cursor:pointer;white-space:nowrap}\n.trow:hover{background:var(--bg-3)}\n.trow.sel{background:var(--sel)}\n.tchev{width:13px;color:var(--faint);flex:0 0 auto;text-align:center;transition:transform .12s}\n.tchev.open{transform:rotate(90deg)}\n.tchev.leaf{visibility:hidden}\n.ticon{flex:0 0 auto;color:var(--muted);display:inline-flex}\n.ticon.tbl{color:#7dd3fc}.ticon.viw{color:#c4b5fd}.ticon.prc{color:#fca5a5}.ticon.fun{color:#fcd34d}.ticon.syn{color:#67e8f9}.ticon.sch{color:#93c5fd}.ticon.db{color:#86efac}.ticon.fld{color:var(--muted)}\n.tlabel{flex:1;overflow:hidden;text-overflow:ellipsis}\n.tbadge{color:var(--faint);font-size:11px}\n.tchildren{margin-left:13px;border-left:1px solid var(--border);padding-left:3px}\n.tnote{color:var(--faint);font-size:11.5px;padding:3px 6px}\n.tsearch{margin:3px 0 4px}\n\n/* workspace tabs */\n.tabbar{display:flex;align-items:stretch;background:var(--bg-2);border-bottom:1px solid var(--border);overflow-x:auto;flex:0 0 auto;min-height:38px}\n.wtab{display:flex;align-items:center;gap:7px;padding:8px 13px;border-right:1px solid var(--border);color:var(--muted);cursor:pointer;white-space:nowrap;max-width:260px}\n.wtab:hover{color:var(--text);background:var(--bg-3)}\n.wtab.active{color:#bcd4ff;background:var(--bg-3);box-shadow:inset 0 -2px 0 var(--accent)}\n.wtab .t-ico{display:inline-flex;color:var(--faint)}\n.wtab .t-title{overflow:hidden;text-overflow:ellipsis}\n.wtab .dot{width:7px;height:7px;border-radius:50%;background:var(--amber)}\n.wtab .x{color:var(--faint);border-radius:4px;padding:0 3px}\n.wtab .x:hover{color:#fff;background:#37425c}\n.tabcontent{flex:1;min-height:0;position:relative;background:var(--bg)}\n.tabpane{position:absolute;inset:0;display:flex;flex-direction:column;overflow:auto}\n\n/* welcome */\n.welcome{padding:26px;max-width:1000px;margin:0 auto;width:100%}\n.welcome h1{margin:0 0 4px;font-size:22px}\n.welcome .lede{color:var(--muted);margin-bottom:22px}\n.wcards{display:grid;grid-template-columns:repeat(auto-fit,minmax(210px,1fr));gap:12px;margin-bottom:24px}\n.wcard{background:var(--bg-2);border:1px solid var(--border);border-radius:12px;padding:16px;cursor:pointer}\n.wcard:hover{border-color:var(--accent);background:var(--bg-3)}\n.wcard .wc-ic{color:var(--accent);margin-bottom:8px}\n.wcard h3{margin:0 0 4px;font-size:14px}\n.wcard p{margin:0;color:var(--muted);font-size:12px}\n.wcols{display:grid;grid-template-columns:1fr 1fr;gap:18px}\n.wcol h4{color:var(--muted);font-size:11px;text-transform:uppercase;letter-spacing:.6px;margin:0 0 8px}\n.wlist .wli{padding:8px 10px;border:1px solid var(--border);border-radius:8px;margin-bottom:6px;cursor:pointer;background:var(--bg-2)}\n.wlist .wli:hover{border-color:var(--border-2);background:var(--bg-3)}\n\n/* toolbar + editor */\n.toolbar{display:flex;gap:6px;align-items:center;flex-wrap:wrap;padding:8px 10px;border-bottom:1px solid var(--border);background:var(--bg-2)}\n.pane-body{flex:1;min-height:0;display:flex;flex-direction:column;padding:10px;gap:8px;overflow:auto}\n.editor{width:100%;min-height:160px;resize:vertical;background:#0a1018;border:1px solid var(--border);border-radius:8px;padding:11px;color:#e7eefc;font-family:Consolas,\"Cascadia Code\",monospace;font-size:13px}\n.errbox{background:#2a0f10;border:1px solid #7f1d1d;color:#fca5a5;border-radius:8px;padding:9px 11px;white-space:pre-wrap;font-family:Consolas,monospace;font-size:12px}\n\n/* grid */\n.gridwrap{flex:1;min-height:120px;overflow:auto;border:1px solid var(--border);border-radius:8px;background:var(--bg-3)}\ntable.grid{border-collapse:separate;border-spacing:0;width:100%;font-size:12.5px}\ntable.grid th,table.grid td{border-bottom:1px solid var(--border);border-right:1px solid var(--border);padding:6px 9px;text-align:left;white-space:nowrap;max-width:440px;overflow:hidden;text-overflow:ellipsis}\ntable.grid th{position:sticky;top:0;z-index:1;background:#172339;color:#cfe0ff;cursor:pointer;user-select:none}\ntable.grid th .sort{color:var(--accent);margin-left:4px}\ntable.grid tr:hover td{background:#13203450}\ntable.grid td.num{text-align:right;color:#a7f3d0}\ntable.grid th.rowhdr,table.grid td.rowhdr{width:54px;text-align:right;color:var(--faint);background:#15203550;cursor:pointer;position:sticky;left:0}\ntable.grid tr.selrow td{background:var(--sel)}\ntable.grid td.edited{background:var(--editbg);box-shadow:inset 0 0 0 1px var(--amber)}\ntable.grid tr.row-del td{background:var(--delbg);text-decoration:line-through;color:#fca5a5}\ntable.grid tr.row-ins td{background:var(--insbg)}\ntable.grid tr.row-err td{box-shadow:inset 0 0 0 1px var(--red)}\n.rowflag{display:inline-block;width:8px;height:8px;border-radius:50%;margin-right:4px}\n.rowflag.d{background:var(--amber)}.rowflag.del{background:var(--red)}.rowflag.ins{background:var(--green)}\n.cellinput{width:100%;background:#0a1018;border:1px solid var(--accent);border-radius:4px;color:#fff;padding:3px 5px;font:inherit}\n.rowerr-msg{color:#fca5a5;font-size:11px}\n\n/* sql tabs inside editor */\n.sqltabs{display:flex;gap:4px;flex-wrap:wrap;padding:6px 10px 0}\n.sqltab{padding:4px 10px;background:var(--bg-3);border:1px solid var(--border);border-bottom:0;border-radius:7px 7px 0 0;cursor:pointer;color:var(--muted);font-size:12px}\n.sqltab.active{color:#cfe0ff;border-color:var(--accent);background:var(--sel)}\n.sqltab .x{margin-left:7px}\n\n/* status bar */\n.statusbar{display:flex;align-items:center;gap:16px;padding:5px 12px;background:var(--bg-2);border-top:1px solid var(--border);color:var(--muted);font-size:12px;flex:0 0 auto}\n.st-item{display:inline-flex;align-items:center;gap:6px}\n.st-dot{width:8px;height:8px;border-radius:50%;background:var(--faint)}\n.st-dot.ok{background:var(--green)}.st-dot.err{background:var(--red)}.st-dot.run{background:var(--amber);animation:sp 1s linear infinite}\n.st-pending{color:var(--amber)}\n\n/* metadata */\n.meta-tabs{display:flex;gap:5px;padding:8px 10px 0}\n.meta-tab{padding:5px 11px;border:1px solid var(--border);border-bottom:0;border-radius:7px 7px 0 0;cursor:pointer;color:var(--muted);font-size:12px}\n.meta-tab.active{color:#cfe0ff;border-color:var(--accent);background:var(--sel)}\n.kvs{display:grid;grid-template-columns:160px 1fr;gap:5px 14px;padding:6px}\n.kvs .k{color:var(--muted)}\n.pill{display:inline-block;padding:1px 7px;border-radius:999px;background:var(--chip);border:1px solid var(--border);font-size:11px;color:var(--muted)}\n.pill.pk{color:#fde68a;border-color:#a16207}\n\n/* modal + wizard */\n.modal{position:fixed;inset:0;background:rgba(2,6,15,.62);display:flex;align-items:center;justify-content:center;z-index:60}\n.dialog{background:var(--bg-2);border:1px solid var(--border-2);border-radius:14px;padding:18px;width:560px;max-width:94vw;max-height:90vh;overflow:auto;box-shadow:0 24px 80px rgba(0,0,0,.5)}\n.dialog h3{margin:0 0 14px}\n.steps{display:flex;gap:6px;margin-bottom:14px;flex-wrap:wrap}\n.step{flex:1;min-width:80px;text-align:center;font-size:11px;color:var(--faint);padding:6px 4px;border-radius:8px;background:var(--bg-3);border:1px solid var(--border)}\n.step.active{color:#cfe0ff;border-color:var(--accent)}\n.step.done{color:var(--green)}\n.wlistbox{max-height:240px;overflow:auto;border:1px solid var(--border);border-radius:9px}\n.wrow{padding:8px 10px;border-bottom:1px solid var(--border);cursor:pointer;display:flex;align-items:center;gap:8px}\n.wrow:hover{background:var(--bg-3)}\n.wrow.sel{background:var(--sel)}\n.swatches{display:flex;gap:7px;flex-wrap:wrap}\n.swatch{width:24px;height:24px;border-radius:7px;cursor:pointer;border:2px solid transparent}\n.swatch.sel{border-color:#fff}\n\n/* context menu */\n.ctxmenu{position:fixed;z-index:80;background:var(--bg-2);border:1px solid var(--border-2);border-radius:9px;padding:5px;min-width:190px;box-shadow:0 16px 50px rgba(0,0,0,.5)}\n.ctxitem{display:flex;align-items:center;gap:9px;padding:7px 10px;border-radius:7px;cursor:pointer;color:var(--text);font-size:12.5px}\n.ctxitem:hover{background:var(--accent);color:#fff}\n.ctxitem.danger{color:#fca5a5}\n.ctxitem.danger:hover{background:#7f1d1d;color:#fff}\n.ctxsep{height:1px;background:var(--border);margin:4px 6px}\n\n/* toasts */\n.toasts{position:fixed;right:16px;bottom:16px;display:flex;flex-direction:column;gap:8px;z-index:90}\n.toast{background:var(--bg-2);border:1px solid var(--border-2);border-left:3px solid var(--accent);border-radius:9px;padding:9px 13px;min-width:240px;max-width:380px;box-shadow:0 10px 34px rgba(0,0,0,.4);animation:slideup .18s ease}\n.toast.ok{border-left-color:var(--green)}.toast.err{border-left-color:var(--red)}.toast.warn{border-left-color:var(--yellow)}\n@keyframes slideup{from{transform:translateY(8px);opacity:0}to{transform:translateY(0);opacity:1}}\n.empty{color:var(--faint);padding:16px;text-align:center}\n";