@sigma4life/mysql-mcp-server 1.0.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 (149) hide show
  1. package/.env.example +35 -0
  2. package/.github/workflows/ci.yml +18 -0
  3. package/AUTHENTICATION.md +24 -0
  4. package/CLAUDE.md +37 -0
  5. package/CONTRIBUTING.md +19 -0
  6. package/LICENSE +21 -0
  7. package/QUERY_ACCESS_SETUP.md +65 -0
  8. package/README.md +144 -0
  9. package/SECURITY.md +10 -0
  10. package/TESTING.md +54 -0
  11. package/dist/cli.d.ts +3 -0
  12. package/dist/cli.d.ts.map +1 -0
  13. package/dist/cli.js +197 -0
  14. package/dist/cli.js.map +1 -0
  15. package/dist/core/cache.d.ts +11 -0
  16. package/dist/core/cache.d.ts.map +1 -0
  17. package/dist/core/cache.js +30 -0
  18. package/dist/core/cache.js.map +1 -0
  19. package/dist/core/config.d.ts +24 -0
  20. package/dist/core/config.d.ts.map +1 -0
  21. package/dist/core/config.js +63 -0
  22. package/dist/core/config.js.map +1 -0
  23. package/dist/core/database-url.d.ts +11 -0
  24. package/dist/core/database-url.d.ts.map +1 -0
  25. package/dist/core/database-url.js +37 -0
  26. package/dist/core/database-url.js.map +1 -0
  27. package/dist/core/database-url.test.d.ts +2 -0
  28. package/dist/core/database-url.test.d.ts.map +1 -0
  29. package/dist/core/database-url.test.js +22 -0
  30. package/dist/core/database-url.test.js.map +1 -0
  31. package/dist/core/logger.d.ts +3 -0
  32. package/dist/core/logger.d.ts.map +1 -0
  33. package/dist/core/logger.js +17 -0
  34. package/dist/core/logger.js.map +1 -0
  35. package/dist/core/mcp.d.ts +14 -0
  36. package/dist/core/mcp.d.ts.map +1 -0
  37. package/dist/core/mcp.js +23 -0
  38. package/dist/core/mcp.js.map +1 -0
  39. package/dist/core/query-safety.d.ts +10 -0
  40. package/dist/core/query-safety.d.ts.map +1 -0
  41. package/dist/core/query-safety.js +50 -0
  42. package/dist/core/query-safety.js.map +1 -0
  43. package/dist/core/query-safety.test.d.ts +2 -0
  44. package/dist/core/query-safety.test.d.ts.map +1 -0
  45. package/dist/core/query-safety.test.js +30 -0
  46. package/dist/core/query-safety.test.js.map +1 -0
  47. package/dist/core/schema-types.d.ts +53 -0
  48. package/dist/core/schema-types.d.ts.map +1 -0
  49. package/dist/core/schema-types.js +2 -0
  50. package/dist/core/schema-types.js.map +1 -0
  51. package/dist/core/security/access-control.d.ts +32 -0
  52. package/dist/core/security/access-control.d.ts.map +1 -0
  53. package/dist/core/security/access-control.js +244 -0
  54. package/dist/core/security/access-control.js.map +1 -0
  55. package/dist/core/security/config-loader.d.ts +20 -0
  56. package/dist/core/security/config-loader.d.ts.map +1 -0
  57. package/dist/core/security/config-loader.js +227 -0
  58. package/dist/core/security/config-loader.js.map +1 -0
  59. package/dist/core/security/types.d.ts +64 -0
  60. package/dist/core/security/types.d.ts.map +1 -0
  61. package/dist/core/security/types.js +28 -0
  62. package/dist/core/security/types.js.map +1 -0
  63. package/dist/core/sql-parser.d.ts +23 -0
  64. package/dist/core/sql-parser.d.ts.map +1 -0
  65. package/dist/core/sql-parser.js +460 -0
  66. package/dist/core/sql-parser.js.map +1 -0
  67. package/dist/core/sql-parser.test.d.ts +2 -0
  68. package/dist/core/sql-parser.test.d.ts.map +1 -0
  69. package/dist/core/sql-parser.test.js +21 -0
  70. package/dist/core/sql-parser.test.js.map +1 -0
  71. package/dist/handlers/accessible-schema.d.ts +53 -0
  72. package/dist/handlers/accessible-schema.d.ts.map +1 -0
  73. package/dist/handlers/accessible-schema.js +192 -0
  74. package/dist/handlers/accessible-schema.js.map +1 -0
  75. package/dist/handlers/data.d.ts +17 -0
  76. package/dist/handlers/data.d.ts.map +1 -0
  77. package/dist/handlers/data.js +30 -0
  78. package/dist/handlers/data.js.map +1 -0
  79. package/dist/handlers/relationships.d.ts +14 -0
  80. package/dist/handlers/relationships.d.ts.map +1 -0
  81. package/dist/handlers/relationships.js +77 -0
  82. package/dist/handlers/relationships.js.map +1 -0
  83. package/dist/handlers/schema.d.ts +14 -0
  84. package/dist/handlers/schema.d.ts.map +1 -0
  85. package/dist/handlers/schema.js +104 -0
  86. package/dist/handlers/schema.js.map +1 -0
  87. package/dist/handlers/search.d.ts +14 -0
  88. package/dist/handlers/search.d.ts.map +1 -0
  89. package/dist/handlers/search.js +18 -0
  90. package/dist/handlers/search.js.map +1 -0
  91. package/dist/handlers/validation.d.ts +32 -0
  92. package/dist/handlers/validation.d.ts.map +1 -0
  93. package/dist/handlers/validation.js +116 -0
  94. package/dist/handlers/validation.js.map +1 -0
  95. package/dist/index.d.ts +3 -0
  96. package/dist/index.d.ts.map +1 -0
  97. package/dist/index.js +294 -0
  98. package/dist/index.js.map +1 -0
  99. package/dist/mysql/connection.d.ts +15 -0
  100. package/dist/mysql/connection.d.ts.map +1 -0
  101. package/dist/mysql/connection.js +57 -0
  102. package/dist/mysql/connection.js.map +1 -0
  103. package/dist/mysql/identifiers.d.ts +3 -0
  104. package/dist/mysql/identifiers.d.ts.map +1 -0
  105. package/dist/mysql/identifiers.js +10 -0
  106. package/dist/mysql/identifiers.js.map +1 -0
  107. package/dist/mysql/identifiers.test.d.ts +2 -0
  108. package/dist/mysql/identifiers.test.d.ts.map +1 -0
  109. package/dist/mysql/identifiers.test.js +11 -0
  110. package/dist/mysql/identifiers.test.js.map +1 -0
  111. package/dist/mysql/queries.d.ts +49 -0
  112. package/dist/mysql/queries.d.ts.map +1 -0
  113. package/dist/mysql/queries.js +206 -0
  114. package/dist/mysql/queries.js.map +1 -0
  115. package/docs/clients/claude-code.md +28 -0
  116. package/docs/clients/claude-desktop.md +35 -0
  117. package/docs/clients/codex.md +28 -0
  118. package/docs/clients/cursor.md +26 -0
  119. package/docs/clients/opencode.md +35 -0
  120. package/docs/clients/vscode.md +25 -0
  121. package/package.json +49 -0
  122. package/query-access.example.json +21 -0
  123. package/src/cli.ts +221 -0
  124. package/src/core/cache.ts +41 -0
  125. package/src/core/config.ts +97 -0
  126. package/src/core/database-url.test.ts +28 -0
  127. package/src/core/database-url.ts +47 -0
  128. package/src/core/logger.ts +24 -0
  129. package/src/core/mcp.ts +23 -0
  130. package/src/core/query-safety.test.ts +36 -0
  131. package/src/core/query-safety.ts +63 -0
  132. package/src/core/schema-types.ts +58 -0
  133. package/src/core/security/access-control.ts +321 -0
  134. package/src/core/security/config-loader.ts +315 -0
  135. package/src/core/security/types.ts +114 -0
  136. package/src/core/sql-parser.test.ts +37 -0
  137. package/src/core/sql-parser.ts +572 -0
  138. package/src/handlers/accessible-schema.ts +314 -0
  139. package/src/handlers/data.ts +66 -0
  140. package/src/handlers/relationships.ts +114 -0
  141. package/src/handlers/schema.ts +154 -0
  142. package/src/handlers/search.ts +34 -0
  143. package/src/handlers/validation.ts +165 -0
  144. package/src/index.ts +337 -0
  145. package/src/mysql/connection.ts +68 -0
  146. package/src/mysql/identifiers.test.ts +12 -0
  147. package/src/mysql/identifiers.ts +10 -0
  148. package/src/mysql/queries.ts +285 -0
  149. package/tsconfig.json +24 -0
@@ -0,0 +1,192 @@
1
+ import { resolveDatabase, resolveSchema } from '../core/config.js';
2
+ import { logger } from '../core/logger.js';
3
+ import { getAccessControlConfig, getTableConfigForSchema, isAccessControlInitialized, } from '../core/security/access-control.js';
4
+ import { getTableInfo } from './schema.js';
5
+ import { findTables } from './search.js';
6
+ function requireAccessConfig(database) {
7
+ if (!isAccessControlInitialized()) {
8
+ throw new Error('Access control not configured. Set QUERY_ACCESS_CONFIG to enable accessible schema introspection.');
9
+ }
10
+ const config = getAccessControlConfig();
11
+ if (!config.databases[database.toUpperCase()]) {
12
+ throw new Error(`Database '${database}' is not configured in query access control. ` +
13
+ `Configured databases: ${Object.keys(config.databases).join(', ') || '(none)'}`);
14
+ }
15
+ return config;
16
+ }
17
+ function getConfiguredSchemas(config, database) {
18
+ const dbConfig = config.databases[database.toUpperCase()];
19
+ if (!dbConfig) {
20
+ return [];
21
+ }
22
+ return dbConfig.schemas ? Object.keys(dbConfig.schemas) : [database];
23
+ }
24
+ function isTableAccessible(tableName, tableConfig) {
25
+ const listLower = tableConfig.list.map((table) => table.toLowerCase());
26
+ const tableNameLower = tableName.toLowerCase();
27
+ if (tableConfig.mode === 'whitelist' && !listLower.includes(tableNameLower)) {
28
+ return {
29
+ accessible: false,
30
+ reason: `Table not in whitelist. Allowed tables: ${tableConfig.list.join(', ') || '(none)'}`,
31
+ };
32
+ }
33
+ if (tableConfig.mode === 'blacklist' && listLower.includes(tableNameLower)) {
34
+ return { accessible: false, reason: 'Table is in blacklist' };
35
+ }
36
+ return { accessible: true };
37
+ }
38
+ function findColumnPolicy(tableName, columnAccess) {
39
+ for (const [table, policy] of Object.entries(columnAccess)) {
40
+ if (table.toLowerCase() === tableName.toLowerCase()) {
41
+ return policy;
42
+ }
43
+ }
44
+ return null;
45
+ }
46
+ function filterColumns(columns, tableName, columnAccess) {
47
+ const policy = findColumnPolicy(tableName, columnAccess);
48
+ if (!policy) {
49
+ return { accessibleColumns: columns };
50
+ }
51
+ const policyColumns = policy.columns.map((column) => column.toLowerCase());
52
+ if (policy.mode === 'inclusion') {
53
+ return {
54
+ accessibleColumns: columns.filter((column) => policyColumns.includes(column.name.toLowerCase())),
55
+ allowedColumnsList: policy.columns,
56
+ mode: 'inclusion',
57
+ };
58
+ }
59
+ return {
60
+ accessibleColumns: columns.filter((column) => !policyColumns.includes(column.name.toLowerCase())),
61
+ blockedColumns: policy.columns,
62
+ mode: 'exclusion',
63
+ };
64
+ }
65
+ function annotateColumnsWithAccess(columns, tableName, columnAccess) {
66
+ const policy = findColumnPolicy(tableName, columnAccess);
67
+ if (!policy) {
68
+ return {
69
+ annotatedColumns: columns.map((column) => ({ ...column, isAccessible: true })),
70
+ };
71
+ }
72
+ const policyColumns = policy.columns.map((column) => column.toLowerCase());
73
+ return {
74
+ annotatedColumns: columns.map((column) => {
75
+ const listed = policyColumns.includes(column.name.toLowerCase());
76
+ const isAccessible = policy.mode === 'inclusion' ? listed : !listed;
77
+ return {
78
+ ...column,
79
+ isAccessible,
80
+ accessDeniedReason: isAccessible
81
+ ? undefined
82
+ : policy.mode === 'inclusion'
83
+ ? `Column not in inclusion list. Allowed: ${policy.columns.join(', ')}`
84
+ : `Column in exclusion list: ${policy.columns.join(', ')}`,
85
+ };
86
+ }),
87
+ mode: policy.mode,
88
+ };
89
+ }
90
+ export async function getAccessibleSchema(args) {
91
+ const database = resolveDatabase(args.database);
92
+ const schema = resolveSchema(args.schema);
93
+ const config = requireAccessConfig(database);
94
+ const notes = [];
95
+ const schemaConfig = getTableConfigForSchema(config, database, schema);
96
+ if (!schemaConfig) {
97
+ throw new Error(`Schema '${schema}' is not configured for query access in database '${database}'.`);
98
+ }
99
+ const { tableConfig, columnAccess } = schemaConfig;
100
+ const tableNames = tableConfig.mode === 'whitelist'
101
+ ? tableConfig.list
102
+ : (await findTables({ database })).map((table) => table.tableName);
103
+ const blacklist = tableConfig.mode === 'blacklist'
104
+ ? new Set(tableConfig.list.map((table) => table.toLowerCase()))
105
+ : new Set();
106
+ const accessibleTableNames = tableNames.filter((table) => !blacklist.has(table.toLowerCase()));
107
+ if (tableConfig.mode === 'blacklist' && tableConfig.list.length > 0) {
108
+ notes.push(`${tableConfig.list.length} table(s) blocked by blacklist: ${tableConfig.list.join(', ')}`);
109
+ }
110
+ const tables = [];
111
+ for (const tableName of accessibleTableNames) {
112
+ try {
113
+ const tableInfo = await getTableInfo({ database, table: tableName });
114
+ const { accessibleColumns, blockedColumns, allowedColumnsList, mode } = filterColumns(tableInfo.columns, tableInfo.name, columnAccess);
115
+ tables.push({
116
+ schema,
117
+ name: tableInfo.name,
118
+ type: tableInfo.type,
119
+ accessibleColumns,
120
+ columnAccessMode: mode,
121
+ blockedColumns,
122
+ allowedColumnsList,
123
+ });
124
+ }
125
+ catch (error) {
126
+ logger.warn(`Could not get info for table ${tableName}: ${error instanceof Error ? error.message : String(error)}`);
127
+ }
128
+ }
129
+ return {
130
+ database,
131
+ requireExplicitColumns: config.requireExplicitColumns,
132
+ configuredSchemas: getConfiguredSchemas(config, database),
133
+ tables,
134
+ notes: notes.length ? notes : undefined,
135
+ };
136
+ }
137
+ export async function getAccessibleTableInfo(args) {
138
+ const database = resolveDatabase(args.database);
139
+ const schema = resolveSchema(args.schema);
140
+ const config = requireAccessConfig(database);
141
+ let tableInfo;
142
+ try {
143
+ tableInfo = await getTableInfo({ database, table: args.table });
144
+ }
145
+ catch (error) {
146
+ return {
147
+ database,
148
+ schema,
149
+ table: args.table,
150
+ type: 'TABLE',
151
+ isAccessible: false,
152
+ accessDeniedReason: error instanceof Error ? error.message : String(error),
153
+ };
154
+ }
155
+ const schemaConfig = getTableConfigForSchema(config, database, schema);
156
+ if (!schemaConfig) {
157
+ return {
158
+ database,
159
+ schema,
160
+ table: tableInfo.name,
161
+ type: tableInfo.type,
162
+ isAccessible: false,
163
+ accessDeniedReason: `Schema '${schema}' is not configured for query access in database '${database}'`,
164
+ };
165
+ }
166
+ const tableAccess = isTableAccessible(tableInfo.name, schemaConfig.tableConfig);
167
+ if (!tableAccess.accessible) {
168
+ return {
169
+ database,
170
+ schema,
171
+ table: tableInfo.name,
172
+ type: tableInfo.type,
173
+ isAccessible: false,
174
+ accessDeniedReason: tableAccess.reason,
175
+ };
176
+ }
177
+ const { annotatedColumns, mode } = annotateColumnsWithAccess(tableInfo.columns, tableInfo.name, schemaConfig.columnAccess);
178
+ return {
179
+ database,
180
+ schema,
181
+ table: tableInfo.name,
182
+ type: tableInfo.type,
183
+ isAccessible: true,
184
+ columnAccessMode: mode,
185
+ columns: annotatedColumns,
186
+ indexes: tableInfo.indexes,
187
+ foreignKeys: tableInfo.foreignKeys,
188
+ accessibleColumnCount: annotatedColumns.filter((column) => column.isAccessible).length,
189
+ totalColumnCount: annotatedColumns.length,
190
+ };
191
+ }
192
+ //# sourceMappingURL=accessible-schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"accessible-schema.js","sourceRoot":"","sources":["../../src/handlers/accessible-schema.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,17 @@
1
+ export interface DataQueryResult {
2
+ originalQuery: string;
3
+ executedQuery: string;
4
+ wasModified: boolean;
5
+ modifications: string[];
6
+ rows: unknown[];
7
+ rowCount: number;
8
+ executionTimeMs: number;
9
+ limitReached: boolean;
10
+ columnNames?: string[];
11
+ }
12
+ export declare function executeQuery(args: {
13
+ database?: string;
14
+ query: string;
15
+ parameters?: Record<string, unknown>;
16
+ }): Promise<DataQueryResult>;
17
+ //# sourceMappingURL=data.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"data.d.ts","sourceRoot":"","sources":["../../src/handlers/data.ts"],"names":[],"mappings":"AAcA,MAAM,WAAW,eAAe;IAC9B,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,OAAO,CAAC;IACrB,aAAa,EAAE,MAAM,EAAE,CAAC;IACxB,IAAI,EAAE,OAAO,EAAE,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,eAAe,EAAE,MAAM,CAAC;IACxB,YAAY,EAAE,OAAO,CAAC;IACtB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;CACxB;AAED,wBAAsB,YAAY,CAAC,IAAI,EAAE;IACvC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACtC,GAAG,OAAO,CAAC,eAAe,CAAC,CAmC3B"}
@@ -0,0 +1,30 @@
1
+ import { appConfig, resolveDatabase } from '../core/config.js';
2
+ import { logger } from '../core/logger.js';
3
+ import { enforceRowLimit, validateQuerySafety, } from '../core/query-safety.js';
4
+ import { getAccessControlConfig, isAccessControlInitialized, validateQueryAccess, } from '../core/security/access-control.js';
5
+ import { db } from '../mysql/connection.js';
6
+ export async function executeQuery(args) {
7
+ const database = resolveDatabase(args.database);
8
+ const startTime = Date.now();
9
+ logger.info(`Executing read query on database: ${database}`);
10
+ validateQuerySafety(args.query);
11
+ if (!isAccessControlInitialized()) {
12
+ throw new Error('Access control not configured. Data queries are blocked until QUERY_ACCESS_CONFIG is set.');
13
+ }
14
+ validateQueryAccess(args.query, database, getAccessControlConfig());
15
+ const modResult = enforceRowLimit(args.query, appConfig.query.maxRows);
16
+ const rows = await db.query(modResult.modifiedQuery, args.parameters);
17
+ const executionTimeMs = Date.now() - startTime;
18
+ return {
19
+ originalQuery: args.query,
20
+ executedQuery: modResult.modifiedQuery,
21
+ wasModified: modResult.wasModified,
22
+ modifications: modResult.modifications,
23
+ rows,
24
+ rowCount: rows.length,
25
+ executionTimeMs,
26
+ limitReached: rows.length === modResult.appliedLimitValue,
27
+ columnNames: rows.length ? Object.keys(rows[0]) : [],
28
+ };
29
+ }
30
+ //# sourceMappingURL=data.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"data.js","sourceRoot":"","sources":["../../src/handlers/data.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAC/D,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EAEL,eAAe,EACf,mBAAmB,GACpB,MAAM,yBAAyB,CAAC;AACjC,OAAO,EACL,sBAAsB,EACtB,0BAA0B,EAC1B,mBAAmB,GACpB,MAAM,oCAAoC,CAAC;AAC5C,OAAO,EAAE,EAAE,EAAE,MAAM,wBAAwB,CAAC;AAc5C,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,IAIlC;IACC,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAE7B,MAAM,CAAC,IAAI,CAAC,qCAAqC,QAAQ,EAAE,CAAC,CAAC;IAE7D,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAEhC,IAAI,CAAC,0BAA0B,EAAE,EAAE,CAAC;QAClC,MAAM,IAAI,KAAK,CACb,2FAA2F,CAC5F,CAAC;IACJ,CAAC;IAED,mBAAmB,CAAC,IAAI,CAAC,KAAK,EAAE,QAAQ,EAAE,sBAAsB,EAAE,CAAC,CAAC;IAEpE,MAAM,SAAS,GAA4B,eAAe,CACxD,IAAI,CAAC,KAAK,EACV,SAAS,CAAC,KAAK,CAAC,OAAO,CACxB,CAAC;IAEF,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,KAAK,CAAM,SAAS,CAAC,aAAa,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3E,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;IAE/C,OAAO;QACL,aAAa,EAAE,IAAI,CAAC,KAAK;QACzB,aAAa,EAAE,SAAS,CAAC,aAAa;QACtC,WAAW,EAAE,SAAS,CAAC,WAAW;QAClC,aAAa,EAAE,SAAS,CAAC,aAAa;QACtC,IAAI;QACJ,QAAQ,EAAE,IAAI,CAAC,MAAM;QACrB,eAAe;QACf,YAAY,EAAE,IAAI,CAAC,MAAM,KAAK,SAAS,CAAC,iBAAiB;QACzD,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE;KACrD,CAAC;AACJ,CAAC"}
@@ -0,0 +1,14 @@
1
+ import { Relationship } from '../mysql/queries.js';
2
+ interface RelationshipPath {
3
+ path: Relationship[];
4
+ joinCondition: string;
5
+ }
6
+ export declare function getRelationships(args: {
7
+ database?: string;
8
+ fromTable: string;
9
+ toTable?: string;
10
+ maxDepth?: number;
11
+ schema?: string;
12
+ }): Promise<RelationshipPath[]>;
13
+ export {};
14
+ //# sourceMappingURL=relationships.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"relationships.d.ts","sourceRoot":"","sources":["../../src/handlers/relationships.ts"],"names":[],"mappings":"AAGA,OAAO,EAEL,YAAY,EACb,MAAM,qBAAqB,CAAC;AAE7B,UAAU,gBAAgB;IACxB,IAAI,EAAE,YAAY,EAAE,CAAC;IACrB,aAAa,EAAE,MAAM,CAAC;CACvB;AAED,wBAAsB,gBAAgB,CAAC,IAAI,EAAE;IAC3C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAwB9B"}
@@ -0,0 +1,77 @@
1
+ import { cache } from '../core/cache.js';
2
+ import { resolveDatabase, resolveSchema } from '../core/config.js';
3
+ import { logger } from '../core/logger.js';
4
+ import { getRelationships as getMysqlRelationships, } from '../mysql/queries.js';
5
+ export async function getRelationships(args) {
6
+ const database = resolveDatabase(args.database);
7
+ resolveSchema(args.schema);
8
+ const { fromTable, toTable, maxDepth = 2 } = args;
9
+ const cacheKey = `relationships:${database}:${fromTable}:${toTable || 'all'}:${maxDepth}`;
10
+ const cached = cache.get(cacheKey);
11
+ if (cached) {
12
+ return cached;
13
+ }
14
+ const relationships = await getMysqlRelationships();
15
+ const paths = toTable
16
+ ? findPaths(fromTable, toTable, relationships, maxDepth)
17
+ : relationships
18
+ .filter((rel) => rel.fromTable === fromTable || rel.toTable === fromTable)
19
+ .map((rel) => ({
20
+ path: [rel],
21
+ joinCondition: buildJoinCondition([rel]),
22
+ }));
23
+ cache.set(cacheKey, paths);
24
+ logger.info(`Found ${paths.length} relationship path(s) for ${fromTable}`);
25
+ return paths;
26
+ }
27
+ function findPaths(fromTable, toTable, relationships, maxDepth) {
28
+ const paths = [];
29
+ const visited = new Set();
30
+ function dfs(currentTable, currentPath, depth) {
31
+ if (depth > maxDepth) {
32
+ return;
33
+ }
34
+ if (currentTable === toTable && currentPath.length > 0) {
35
+ paths.push({
36
+ path: [...currentPath],
37
+ joinCondition: buildJoinCondition(currentPath),
38
+ });
39
+ return;
40
+ }
41
+ visited.add(currentTable);
42
+ for (const rel of relationships) {
43
+ if (rel.fromTable === currentTable && !visited.has(rel.toTable)) {
44
+ currentPath.push(rel);
45
+ dfs(rel.toTable, currentPath, depth + 1);
46
+ currentPath.pop();
47
+ }
48
+ if (rel.toTable === currentTable && !visited.has(rel.fromTable)) {
49
+ currentPath.push(rel);
50
+ dfs(rel.fromTable, currentPath, depth + 1);
51
+ currentPath.pop();
52
+ }
53
+ }
54
+ visited.delete(currentTable);
55
+ }
56
+ dfs(fromTable, [], 0);
57
+ return paths;
58
+ }
59
+ function buildJoinCondition(path) {
60
+ if (path.length === 0) {
61
+ return '';
62
+ }
63
+ const conditions = [];
64
+ let currentTable = path[0].fromTable;
65
+ for (const rel of path) {
66
+ if (rel.fromTable === currentTable) {
67
+ conditions.push(`JOIN ${rel.toTable} ON ${rel.fromTable}.${rel.fromColumn} = ${rel.toTable}.${rel.toColumn}`);
68
+ currentTable = rel.toTable;
69
+ }
70
+ else {
71
+ conditions.push(`JOIN ${rel.fromTable} ON ${rel.toTable}.${rel.toColumn} = ${rel.fromTable}.${rel.fromColumn}`);
72
+ currentTable = rel.fromTable;
73
+ }
74
+ }
75
+ return conditions.join('\n');
76
+ }
77
+ //# sourceMappingURL=relationships.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"relationships.js","sourceRoot":"","sources":["../../src/handlers/relationships.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ import { SchemaResult, TableMetadata } from '../core/schema-types.js';
2
+ export declare function getSchema(args: {
3
+ database?: string;
4
+ tables?: string[];
5
+ schema?: string;
6
+ includeRelationships?: boolean;
7
+ includeStatistics?: boolean;
8
+ }): Promise<SchemaResult>;
9
+ export declare function getTableInfo(args: {
10
+ database?: string;
11
+ table: string;
12
+ schema?: string;
13
+ }): Promise<TableMetadata>;
14
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/handlers/schema.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,YAAY,EACZ,aAAa,EACd,MAAM,yBAAyB,CAAC;AAoFjC,wBAAsB,SAAS,CAAC,IAAI,EAAE;IACpC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,oBAAoB,CAAC,EAAE,OAAO,CAAC;IAC/B,iBAAiB,CAAC,EAAE,OAAO,CAAC;CAC7B,GAAG,OAAO,CAAC,YAAY,CAAC,CAiCxB;AAED,wBAAsB,YAAY,CAAC,IAAI,EAAE;IACvC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,GAAG,OAAO,CAAC,aAAa,CAAC,CAmBzB"}
@@ -0,0 +1,104 @@
1
+ import { cache } from '../core/cache.js';
2
+ import { resolveDatabase, resolveSchema } from '../core/config.js';
3
+ import { logger } from '../core/logger.js';
4
+ import { getColumns, getForeignKeys, getIndexes, getPrimaryKeys, getStatistics, listTables, } from '../mysql/queries.js';
5
+ import { validateDatabaseObject } from './validation.js';
6
+ function byTable(rows) {
7
+ const map = new Map();
8
+ for (const row of rows) {
9
+ const list = map.get(row.tableName) || [];
10
+ list.push(row);
11
+ map.set(row.tableName, list);
12
+ }
13
+ return map;
14
+ }
15
+ async function buildMetadata(tableNames, includeRelationships = true, includeStatistics = false) {
16
+ const tables = await listTables(tableNames);
17
+ const actualTableNames = tables.map((table) => table.tableName);
18
+ const [columns, primaryKeys, foreignKeys, indexes, statistics] = await Promise.all([
19
+ getColumns(actualTableNames),
20
+ getPrimaryKeys(actualTableNames),
21
+ includeRelationships ? getForeignKeys(actualTableNames) : Promise.resolve([]),
22
+ getIndexes(actualTableNames),
23
+ includeStatistics ? getStatistics(actualTableNames) : Promise.resolve([]),
24
+ ]);
25
+ const columnsByTable = byTable(columns);
26
+ const pksByTable = byTable(primaryKeys);
27
+ const fksByTable = byTable(foreignKeys);
28
+ const indexesByTable = byTable(indexes);
29
+ const statsByTable = byTable(statistics);
30
+ return tables.map((table) => {
31
+ const metadata = {
32
+ schema: table.schemaName,
33
+ name: table.tableName,
34
+ type: table.tableType === 'VIEW' ? 'VIEW' : 'TABLE',
35
+ columns: (columnsByTable.get(table.tableName) || []).map(({ tableName: _tableName, ...column }) => ({
36
+ ...column,
37
+ nullable: Boolean(column.nullable),
38
+ isIdentity: Boolean(column.isIdentity),
39
+ isComputed: Boolean(column.isComputed),
40
+ isPrimaryKey: Boolean(column.isPrimaryKey),
41
+ isForeignKey: Boolean(column.isForeignKey),
42
+ })),
43
+ indexes: (indexesByTable.get(table.tableName) || []).map(({ tableName: _tableName, ...index }) => ({
44
+ ...index,
45
+ isUnique: Boolean(index.isUnique),
46
+ isPrimaryKey: Boolean(index.isPrimaryKey),
47
+ })),
48
+ };
49
+ const primaryKey = pksByTable.get(table.tableName)?.[0];
50
+ if (primaryKey) {
51
+ const { tableName: _tableName, ...pk } = primaryKey;
52
+ metadata.primaryKey = pk;
53
+ }
54
+ if (includeRelationships) {
55
+ metadata.foreignKeys = (fksByTable.get(table.tableName) || []).map(({ tableName: _tableName, ...fk }) => fk);
56
+ }
57
+ const stats = statsByTable.get(table.tableName)?.[0];
58
+ if (includeStatistics && stats) {
59
+ const { tableName: _tableName, ...tableStats } = stats;
60
+ metadata.statistics = tableStats;
61
+ }
62
+ return metadata;
63
+ });
64
+ }
65
+ export async function getSchema(args) {
66
+ const database = resolveDatabase(args.database);
67
+ resolveSchema(args.schema);
68
+ const { tables, includeRelationships = true, includeStatistics = false, } = args;
69
+ const cacheKey = `schema:${database}:${tables?.join(',') || 'all'}:${includeRelationships}:${includeStatistics}`;
70
+ const cached = cache.get(cacheKey);
71
+ if (cached) {
72
+ return cached;
73
+ }
74
+ if (tables?.length) {
75
+ const validation = await Promise.all(tables.map((table) => validateDatabaseObject(database, table)));
76
+ const missing = validation.filter((result) => !result.valid);
77
+ if (missing.length) {
78
+ throw new Error(missing.map((result) => result.message).join('\n'));
79
+ }
80
+ }
81
+ const result = {
82
+ schema: await buildMetadata(tables, includeRelationships, includeStatistics),
83
+ };
84
+ cache.set(cacheKey, result);
85
+ logger.info(`Retrieved schema for ${result.schema.length} objects from ${database}`);
86
+ return result;
87
+ }
88
+ export async function getTableInfo(args) {
89
+ const database = resolveDatabase(args.database);
90
+ resolveSchema(args.schema);
91
+ const validation = await validateDatabaseObject(database, args.table);
92
+ if (!validation.valid || !validation.table?.actualName) {
93
+ throw new Error(validation.message);
94
+ }
95
+ const tableName = validation.table.actualName.includes('.')
96
+ ? validation.table.actualName.split('.').pop()
97
+ : validation.table.actualName;
98
+ const metadata = await buildMetadata([tableName], true, false);
99
+ if (!metadata[0]) {
100
+ throw new Error(`Table '${args.table}' not found in database '${database}'`);
101
+ }
102
+ return metadata[0];
103
+ }
104
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/handlers/schema.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ import { ObjectSearchResult, TableSearchResult } from '../mysql/queries.js';
2
+ export declare function findTables(args: {
3
+ database?: string;
4
+ pattern?: string;
5
+ hasColumn?: string;
6
+ schema?: string;
7
+ }): Promise<TableSearchResult[]>;
8
+ export declare function searchObjects(args: {
9
+ database?: string;
10
+ search: string;
11
+ schema?: string;
12
+ type?: string;
13
+ }): Promise<ObjectSearchResult[]>;
14
+ //# sourceMappingURL=search.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.d.ts","sourceRoot":"","sources":["../../src/handlers/search.ts"],"names":[],"mappings":"AAEA,OAAO,EAGL,kBAAkB,EAClB,iBAAiB,EAClB,MAAM,qBAAqB,CAAC;AAE7B,wBAAsB,UAAU,CAAC,IAAI,EAAE;IACrC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,GAAG,OAAO,CAAC,iBAAiB,EAAE,CAAC,CAM/B;AAED,wBAAsB,aAAa,CAAC,IAAI,EAAE;IACxC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC,CAMhC"}
@@ -0,0 +1,18 @@
1
+ import { resolveDatabase, resolveSchema } from '../core/config.js';
2
+ import { logger } from '../core/logger.js';
3
+ import { findTables as findMysqlTables, searchObjects as searchMysqlObjects, } from '../mysql/queries.js';
4
+ export async function findTables(args) {
5
+ const database = resolveDatabase(args.database);
6
+ resolveSchema(args.schema);
7
+ const tables = await findMysqlTables(args.pattern, args.hasColumn);
8
+ logger.info(`Found ${tables.length} tables matching criteria in ${database}`);
9
+ return tables;
10
+ }
11
+ export async function searchObjects(args) {
12
+ const database = resolveDatabase(args.database);
13
+ resolveSchema(args.schema);
14
+ const results = await searchMysqlObjects(args.search, args.type);
15
+ logger.info(`Found ${results.length} matches for '${args.search}' in ${database}`);
16
+ return results;
17
+ }
18
+ //# sourceMappingURL=search.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.js","sourceRoot":"","sources":["../../src/handlers/search.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AACnE,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EACL,UAAU,IAAI,eAAe,EAC7B,aAAa,IAAI,kBAAkB,GAGpC,MAAM,qBAAqB,CAAC;AAE7B,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,IAKhC;IACC,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChD,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC3B,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;IACnE,MAAM,CAAC,IAAI,CAAC,SAAS,MAAM,CAAC,MAAM,gCAAgC,QAAQ,EAAE,CAAC,CAAC;IAC9E,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,IAKnC;IACC,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChD,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC3B,MAAM,OAAO,GAAG,MAAM,kBAAkB,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;IACjE,MAAM,CAAC,IAAI,CAAC,SAAS,OAAO,CAAC,MAAM,iBAAiB,IAAI,CAAC,MAAM,QAAQ,QAAQ,EAAE,CAAC,CAAC;IACnF,OAAO,OAAO,CAAC;AACjB,CAAC"}
@@ -0,0 +1,32 @@
1
+ export interface ValidationResult {
2
+ exists: boolean;
3
+ actualName?: string;
4
+ suggestions?: string[];
5
+ message: string;
6
+ }
7
+ export interface DatabaseValidation extends ValidationResult {
8
+ databases?: string[];
9
+ }
10
+ export interface SchemaValidation extends ValidationResult {
11
+ schemas?: string[];
12
+ }
13
+ export interface TableValidation extends ValidationResult {
14
+ tables?: Array<{
15
+ schema: string;
16
+ table: string;
17
+ fullName: string;
18
+ type?: string;
19
+ rowCount?: number | null;
20
+ }>;
21
+ }
22
+ export declare function validateDatabase(database?: string): Promise<DatabaseValidation>;
23
+ export declare function validateSchema(database: string | undefined, schema?: string): Promise<SchemaValidation>;
24
+ export declare function validateTable(database: string | undefined, table: string, schema?: string): Promise<TableValidation>;
25
+ export declare function validateDatabaseObject(database?: string, table?: string, schema?: string): Promise<{
26
+ valid: boolean;
27
+ database: DatabaseValidation;
28
+ schema?: SchemaValidation;
29
+ table?: TableValidation;
30
+ message: string;
31
+ }>;
32
+ //# sourceMappingURL=validation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.d.ts","sourceRoot":"","sources":["../../src/handlers/validation.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,gBAAgB;IAC/B,MAAM,EAAE,OAAO,CAAC;IAChB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,kBAAmB,SAAQ,gBAAgB;IAC1D,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;CACtB;AAED,MAAM,WAAW,gBAAiB,SAAQ,gBAAgB;IACxD,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC;CACpB;AAED,MAAM,WAAW,eAAgB,SAAQ,gBAAgB;IACvD,MAAM,CAAC,EAAE,KAAK,CAAC;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;KAAE,CAAC,CAAC;CAC9G;AAED,wBAAsB,gBAAgB,CAAC,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,kBAAkB,CAAC,CASrF;AAED,wBAAsB,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,SAAS,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAQ7G;AAED,wBAAsB,aAAa,CACjC,QAAQ,EAAE,MAAM,GAAG,SAAS,EAC5B,KAAK,EAAE,MAAM,EACb,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC,eAAe,CAAC,CAuE1B;AAED,wBAAsB,sBAAsB,CAC1C,QAAQ,CAAC,EAAE,MAAM,EACjB,KAAK,CAAC,EAAE,MAAM,EACd,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC;IACT,KAAK,EAAE,OAAO,CAAC;IACf,QAAQ,EAAE,kBAAkB,CAAC;IAC7B,MAAM,CAAC,EAAE,gBAAgB,CAAC;IAC1B,KAAK,CAAC,EAAE,eAAe,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC,CAiCD"}