@bytebase/dbhub 0.21.0 → 0.21.1

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/README.md CHANGED
@@ -117,4 +117,10 @@ See [Testing](.claude/skills/testing/SKILL.md) and [Debug](https://dbhub.ai/conf
117
117
 
118
118
  ## Star History
119
119
 
120
- ![Star History Chart](https://api.star-history.com/svg?repos=bytebase/dbhub&type=Date)
120
+ <a href="https://www.star-history.com/?repos=bytebase%2Fdbhub&type=date&legend=top-left">
121
+ <picture>
122
+ <source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/image?repos=bytebase/dbhub&type=date&theme=dark&legend=top-left" />
123
+ <source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/image?repos=bytebase/dbhub&type=date&legend=top-left" />
124
+ <img alt="Star History Chart" src="https://api.star-history.com/image?repos=bytebase/dbhub&type=date&legend=top-left" />
125
+ </picture>
126
+ </a>
@@ -1,3 +1,6 @@
1
+ import {
2
+ isDriverNotInstalled
3
+ } from "./chunk-GWPUVYLO.js";
1
4
  import {
2
5
  ConnectorRegistry,
3
6
  SafeURL,
@@ -1314,8 +1317,18 @@ function buildDSNFromSource(source) {
1314
1317
  }
1315
1318
 
1316
1319
  // src/utils/aws-rds-signer.ts
1317
- import { Signer } from "@aws-sdk/rds-signer";
1318
1320
  async function generateRdsAuthToken(params) {
1321
+ let Signer;
1322
+ try {
1323
+ ({ Signer } = await import("@aws-sdk/rds-signer"));
1324
+ } catch (error) {
1325
+ if (isDriverNotInstalled(error, "@aws-sdk/rds-signer")) {
1326
+ throw new Error(
1327
+ 'AWS IAM authentication requires the "@aws-sdk/rds-signer" package. Install it with: pnpm add @aws-sdk/rds-signer'
1328
+ );
1329
+ }
1330
+ throw error;
1331
+ }
1319
1332
  const signer = new Signer({
1320
1333
  hostname: params.hostname,
1321
1334
  port: params.port,
@@ -0,0 +1,35 @@
1
+ // src/utils/module-loader.ts
2
+ var MISSING_MODULE_RE = /Cannot find (?:package|module) '([^']+)'/;
3
+ function isDriverNotInstalled(err, driver) {
4
+ if (!(err instanceof Error) || !("code" in err) || err.code !== "ERR_MODULE_NOT_FOUND") {
5
+ return false;
6
+ }
7
+ const match = err.message.match(MISSING_MODULE_RE);
8
+ if (!match) {
9
+ return false;
10
+ }
11
+ const missingSpecifier = match[1];
12
+ return missingSpecifier === driver || missingSpecifier.startsWith(`${driver}/`);
13
+ }
14
+ async function loadConnectors(connectorModules) {
15
+ await Promise.all(
16
+ connectorModules.map(async ({ load, name, driver }) => {
17
+ try {
18
+ await load();
19
+ } catch (err) {
20
+ if (isDriverNotInstalled(err, driver)) {
21
+ console.error(
22
+ `Skipping ${name} connector: driver package "${driver}" not installed.`
23
+ );
24
+ } else {
25
+ throw err;
26
+ }
27
+ }
28
+ })
29
+ );
30
+ }
31
+
32
+ export {
33
+ isDriverNotInstalled,
34
+ loadConnectors
35
+ };
package/dist/index.js CHANGED
@@ -12,7 +12,10 @@ import {
12
12
  resolveSourceConfigs,
13
13
  resolveTomlConfigPath,
14
14
  resolveTransport
15
- } from "./chunk-GRGEI5QT.js";
15
+ } from "./chunk-6JFMPX62.js";
16
+ import {
17
+ loadConnectors
18
+ } from "./chunk-GWPUVYLO.js";
16
19
  import {
17
20
  quoteQualifiedIdentifier
18
21
  } from "./chunk-JFWX35TB.js";
@@ -90,20 +93,76 @@ function createToolSuccessResponse(data, meta = {}) {
90
93
 
91
94
  // src/utils/allowed-keywords.ts
92
95
  var allowedKeywords = {
93
- postgres: ["select", "with", "explain", "analyze", "show"],
94
- mysql: ["select", "with", "explain", "analyze", "show", "describe", "desc"],
95
- mariadb: ["select", "with", "explain", "analyze", "show", "describe", "desc"],
96
- sqlite: ["select", "with", "explain", "analyze", "pragma"],
96
+ postgres: ["select", "with", "explain", "show"],
97
+ mysql: ["select", "with", "explain", "show", "describe", "desc"],
98
+ mariadb: ["select", "with", "explain", "show", "describe", "desc"],
99
+ sqlite: ["select", "with", "explain", "pragma"],
97
100
  sqlserver: ["select", "with", "explain", "showplan"]
98
101
  };
102
+ var mutatingKeywords = [
103
+ "insert",
104
+ "update",
105
+ "delete",
106
+ "drop",
107
+ "alter",
108
+ "create",
109
+ "truncate",
110
+ "merge",
111
+ "grant",
112
+ "revoke",
113
+ "rename"
114
+ ];
115
+ var mutatingPattern = new RegExp(
116
+ `\\b(?:${mutatingKeywords.join("|")})\\b`,
117
+ "i"
118
+ );
119
+ var mutatingPatternWithReplace = new RegExp(
120
+ `\\b(?:${mutatingKeywords.join("|")}|replace\\s+(?:(?:low_priority|delayed)\\s+)?into)\\b`,
121
+ "i"
122
+ );
123
+ var mutatingPatterns = {
124
+ postgres: mutatingPattern,
125
+ mysql: mutatingPatternWithReplace,
126
+ mariadb: mutatingPatternWithReplace,
127
+ sqlite: mutatingPatternWithReplace,
128
+ sqlserver: mutatingPattern
129
+ };
130
+ var selectIntoPattern = /\bselect\b[\s\S]+\binto\b/i;
131
+ var explainAnalyzePattern = /^explain\s+(?:\([^)]*\banalyze\b(?!\s*(?:=\s*)?(?:false|off|0)\b)[^)]*\)|\banalyze\b(?!\s*(?:=\s*)?(?:false|off|0)\b)(?:\s+verbose\b)?)/i;
99
132
  function isReadOnlySQL(sql, connectorType) {
100
- const cleanedSQL = stripCommentsAndStrings(sql, connectorType).trim().toLowerCase();
133
+ return checkReadOnly(
134
+ stripCommentsAndStrings(sql, connectorType).trim().toLowerCase(),
135
+ connectorType
136
+ );
137
+ }
138
+ function checkReadOnly(cleanedSQL, connectorType) {
101
139
  if (!cleanedSQL) {
102
140
  return false;
103
141
  }
104
- const firstWord = cleanedSQL.split(/\s+/)[0];
142
+ const firstWord = cleanedSQL.match(/\S+/)?.[0] ?? "";
105
143
  const keywordList = allowedKeywords[connectorType] || [];
106
- return keywordList.includes(firstWord);
144
+ if (!keywordList.includes(firstWord)) {
145
+ return false;
146
+ }
147
+ if (firstWord === "with") {
148
+ const pattern = mutatingPatterns[connectorType] ?? mutatingPattern;
149
+ if (pattern.test(cleanedSQL)) {
150
+ return false;
151
+ }
152
+ }
153
+ if ((firstWord === "select" || firstWord === "with") && selectIntoPattern.test(cleanedSQL)) {
154
+ return false;
155
+ }
156
+ if (firstWord === "explain") {
157
+ const m = explainAnalyzePattern.exec(cleanedSQL);
158
+ if (m) {
159
+ const afterExplain = cleanedSQL.slice(m[0].length).trim();
160
+ if (afterExplain && !checkReadOnly(afterExplain, connectorType)) {
161
+ return false;
162
+ }
163
+ }
164
+ }
165
+ return true;
107
166
  }
108
167
 
109
168
  // src/requests/store.ts
@@ -1362,7 +1421,7 @@ See documentation for more details on configuring database connections.
1362
1421
  const sources = sourceConfigsData.sources;
1363
1422
  console.error(`Configuration source: ${sourceConfigsData.source}`);
1364
1423
  await connectorManager.connectWithSources(sources);
1365
- const { initializeToolRegistry: initializeToolRegistry2 } = await import("./registry-XSMMLRC4.js");
1424
+ const { initializeToolRegistry: initializeToolRegistry2 } = await import("./registry-I2JQWFDX.js");
1366
1425
  initializeToolRegistry2({
1367
1426
  sources: sourceConfigsData.sources,
1368
1427
  tools: sourceConfigsData.tools
@@ -1488,41 +1547,10 @@ See documentation for more details on configuring database connections.
1488
1547
  }
1489
1548
  }
1490
1549
 
1491
- // src/utils/module-loader.ts
1492
- var MISSING_MODULE_RE = /Cannot find (?:package|module) '([^']+)'/;
1493
- function isDriverNotInstalled(err, driver) {
1494
- if (!(err instanceof Error) || !("code" in err) || err.code !== "ERR_MODULE_NOT_FOUND") {
1495
- return false;
1496
- }
1497
- const match = err.message.match(MISSING_MODULE_RE);
1498
- if (!match) {
1499
- return false;
1500
- }
1501
- const missingSpecifier = match[1];
1502
- return missingSpecifier === driver || missingSpecifier.startsWith(`${driver}/`);
1503
- }
1504
- async function loadConnectors(connectorModules2) {
1505
- await Promise.all(
1506
- connectorModules2.map(async ({ load, name, driver }) => {
1507
- try {
1508
- await load();
1509
- } catch (err) {
1510
- if (isDriverNotInstalled(err, driver)) {
1511
- console.error(
1512
- `Skipping ${name} connector: driver package "${driver}" not installed.`
1513
- );
1514
- } else {
1515
- throw err;
1516
- }
1517
- }
1518
- })
1519
- );
1520
- }
1521
-
1522
1550
  // src/index.ts
1523
1551
  var connectorModules = [
1524
- { load: () => import("./postgres-B7YSSZMH.js"), name: "PostgreSQL", driver: "pg" },
1525
- { load: () => import("./sqlserver-FDBRUELV.js"), name: "SQL Server", driver: "mssql" },
1552
+ { load: () => import("./postgres-4PCNQDGV.js"), name: "PostgreSQL", driver: "pg" },
1553
+ { load: () => import("./sqlserver-JBR6X37Z.js"), name: "SQL Server", driver: "mssql" },
1526
1554
  { load: () => import("./sqlite-FSCLCRIH.js"), name: "SQLite", driver: "better-sqlite3" },
1527
1555
  { load: () => import("./mysql-I35IQ2GH.js"), name: "MySQL", driver: "mysql2" },
1528
1556
  { load: () => import("./mariadb-VGTS4WXE.js"), name: "MariaDB", driver: "mariadb" }
@@ -303,7 +303,7 @@ var PostgresConnector = class _PostgresConnector {
303
303
  JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
304
304
  WHERE c.relname = $1
305
305
  AND n.nspname = $2
306
- AND c.relkind IN ('r','p','m','f')
306
+ AND c.relkind IN ('r','p','m','f','v')
307
307
  `,
308
308
  [tableName, schemaToUse]
309
309
  );
@@ -2,7 +2,8 @@ import {
2
2
  ToolRegistry,
3
3
  getToolRegistry,
4
4
  initializeToolRegistry
5
- } from "./chunk-GRGEI5QT.js";
5
+ } from "./chunk-6JFMPX62.js";
6
+ import "./chunk-GWPUVYLO.js";
6
7
  import "./chunk-C7WEAPX4.js";
7
8
  export {
8
9
  ToolRegistry,
@@ -1,3 +1,6 @@
1
+ import {
2
+ isDriverNotInstalled
3
+ } from "./chunk-GWPUVYLO.js";
1
4
  import {
2
5
  SQLRowLimiter
3
6
  } from "./chunk-BRXZ5ZQB.js";
@@ -9,7 +12,6 @@ import {
9
12
 
10
13
  // src/connectors/sqlserver/index.ts
11
14
  import sql from "mssql";
12
- import { DefaultAzureCredential } from "@azure/identity";
13
15
  var SQLServerDSNParser = class {
14
16
  async parse(dsn, config) {
15
17
  const connectionTimeoutSeconds = config?.connectionTimeoutSeconds;
@@ -74,6 +76,17 @@ Expected: ${expectedFormat}`
74
76
  };
75
77
  switch (options.authentication) {
76
78
  case "azure-active-directory-access-token": {
79
+ let DefaultAzureCredential;
80
+ try {
81
+ ({ DefaultAzureCredential } = await import("@azure/identity"));
82
+ } catch (importError) {
83
+ if (isDriverNotInstalled(importError, "@azure/identity")) {
84
+ throw new Error(
85
+ 'Azure AD authentication requires the "@azure/identity" package. Install it with: pnpm add @azure/identity'
86
+ );
87
+ }
88
+ throw importError;
89
+ }
77
90
  try {
78
91
  const credential = new DefaultAzureCredential();
79
92
  const token = await credential.getToken("https://database.windows.net/");
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@bytebase/dbhub",
3
- "version": "0.21.0",
3
+ "version": "0.21.1",
4
4
  "mcpName": "io.github.bytebase/dbhub",
5
5
  "description": "Minimal, token-efficient Database MCP Server for PostgreSQL, MySQL, SQL Server, SQLite, MariaDB",
6
6
  "repository": {
@@ -30,7 +30,8 @@
30
30
  "test": "vitest run",
31
31
  "test:unit": "vitest run --project unit",
32
32
  "test:watch": "vitest",
33
- "test:integration": "vitest run --project integration"
33
+ "test:integration": "vitest run --project integration",
34
+ "test:build": "node scripts/smoke-test-build.mjs"
34
35
  },
35
36
  "engines": {
36
37
  "node": ">=20"
@@ -39,8 +40,6 @@
39
40
  "author": "",
40
41
  "license": "MIT",
41
42
  "dependencies": {
42
- "@aws-sdk/rds-signer": "^3.1001.0",
43
- "@azure/identity": "^4.8.0",
44
43
  "@iarna/toml": "^2.2.5",
45
44
  "@modelcontextprotocol/sdk": "^1.25.1",
46
45
  "dotenv": "^16.4.7",
@@ -50,6 +49,8 @@
50
49
  "zod": "^3.24.2"
51
50
  },
52
51
  "optionalDependencies": {
52
+ "@aws-sdk/rds-signer": "^3.1001.0",
53
+ "@azure/identity": "^4.8.0",
53
54
  "better-sqlite3": "^11.9.0",
54
55
  "mariadb": "^3.4.0",
55
56
  "mssql": "^11.0.1",