@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.
- package/.env.example +35 -0
- package/.github/workflows/ci.yml +18 -0
- package/AUTHENTICATION.md +24 -0
- package/CLAUDE.md +37 -0
- package/CONTRIBUTING.md +19 -0
- package/LICENSE +21 -0
- package/QUERY_ACCESS_SETUP.md +65 -0
- package/README.md +144 -0
- package/SECURITY.md +10 -0
- package/TESTING.md +54 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +197 -0
- package/dist/cli.js.map +1 -0
- package/dist/core/cache.d.ts +11 -0
- package/dist/core/cache.d.ts.map +1 -0
- package/dist/core/cache.js +30 -0
- package/dist/core/cache.js.map +1 -0
- package/dist/core/config.d.ts +24 -0
- package/dist/core/config.d.ts.map +1 -0
- package/dist/core/config.js +63 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/database-url.d.ts +11 -0
- package/dist/core/database-url.d.ts.map +1 -0
- package/dist/core/database-url.js +37 -0
- package/dist/core/database-url.js.map +1 -0
- package/dist/core/database-url.test.d.ts +2 -0
- package/dist/core/database-url.test.d.ts.map +1 -0
- package/dist/core/database-url.test.js +22 -0
- package/dist/core/database-url.test.js.map +1 -0
- package/dist/core/logger.d.ts +3 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +17 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/mcp.d.ts +14 -0
- package/dist/core/mcp.d.ts.map +1 -0
- package/dist/core/mcp.js +23 -0
- package/dist/core/mcp.js.map +1 -0
- package/dist/core/query-safety.d.ts +10 -0
- package/dist/core/query-safety.d.ts.map +1 -0
- package/dist/core/query-safety.js +50 -0
- package/dist/core/query-safety.js.map +1 -0
- package/dist/core/query-safety.test.d.ts +2 -0
- package/dist/core/query-safety.test.d.ts.map +1 -0
- package/dist/core/query-safety.test.js +30 -0
- package/dist/core/query-safety.test.js.map +1 -0
- package/dist/core/schema-types.d.ts +53 -0
- package/dist/core/schema-types.d.ts.map +1 -0
- package/dist/core/schema-types.js +2 -0
- package/dist/core/schema-types.js.map +1 -0
- package/dist/core/security/access-control.d.ts +32 -0
- package/dist/core/security/access-control.d.ts.map +1 -0
- package/dist/core/security/access-control.js +244 -0
- package/dist/core/security/access-control.js.map +1 -0
- package/dist/core/security/config-loader.d.ts +20 -0
- package/dist/core/security/config-loader.d.ts.map +1 -0
- package/dist/core/security/config-loader.js +227 -0
- package/dist/core/security/config-loader.js.map +1 -0
- package/dist/core/security/types.d.ts +64 -0
- package/dist/core/security/types.d.ts.map +1 -0
- package/dist/core/security/types.js +28 -0
- package/dist/core/security/types.js.map +1 -0
- package/dist/core/sql-parser.d.ts +23 -0
- package/dist/core/sql-parser.d.ts.map +1 -0
- package/dist/core/sql-parser.js +460 -0
- package/dist/core/sql-parser.js.map +1 -0
- package/dist/core/sql-parser.test.d.ts +2 -0
- package/dist/core/sql-parser.test.d.ts.map +1 -0
- package/dist/core/sql-parser.test.js +21 -0
- package/dist/core/sql-parser.test.js.map +1 -0
- package/dist/handlers/accessible-schema.d.ts +53 -0
- package/dist/handlers/accessible-schema.d.ts.map +1 -0
- package/dist/handlers/accessible-schema.js +192 -0
- package/dist/handlers/accessible-schema.js.map +1 -0
- package/dist/handlers/data.d.ts +17 -0
- package/dist/handlers/data.d.ts.map +1 -0
- package/dist/handlers/data.js +30 -0
- package/dist/handlers/data.js.map +1 -0
- package/dist/handlers/relationships.d.ts +14 -0
- package/dist/handlers/relationships.d.ts.map +1 -0
- package/dist/handlers/relationships.js +77 -0
- package/dist/handlers/relationships.js.map +1 -0
- package/dist/handlers/schema.d.ts +14 -0
- package/dist/handlers/schema.d.ts.map +1 -0
- package/dist/handlers/schema.js +104 -0
- package/dist/handlers/schema.js.map +1 -0
- package/dist/handlers/search.d.ts +14 -0
- package/dist/handlers/search.d.ts.map +1 -0
- package/dist/handlers/search.js +18 -0
- package/dist/handlers/search.js.map +1 -0
- package/dist/handlers/validation.d.ts +32 -0
- package/dist/handlers/validation.d.ts.map +1 -0
- package/dist/handlers/validation.js +116 -0
- package/dist/handlers/validation.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +294 -0
- package/dist/index.js.map +1 -0
- package/dist/mysql/connection.d.ts +15 -0
- package/dist/mysql/connection.d.ts.map +1 -0
- package/dist/mysql/connection.js +57 -0
- package/dist/mysql/connection.js.map +1 -0
- package/dist/mysql/identifiers.d.ts +3 -0
- package/dist/mysql/identifiers.d.ts.map +1 -0
- package/dist/mysql/identifiers.js +10 -0
- package/dist/mysql/identifiers.js.map +1 -0
- package/dist/mysql/identifiers.test.d.ts +2 -0
- package/dist/mysql/identifiers.test.d.ts.map +1 -0
- package/dist/mysql/identifiers.test.js +11 -0
- package/dist/mysql/identifiers.test.js.map +1 -0
- package/dist/mysql/queries.d.ts +49 -0
- package/dist/mysql/queries.d.ts.map +1 -0
- package/dist/mysql/queries.js +206 -0
- package/dist/mysql/queries.js.map +1 -0
- package/docs/clients/claude-code.md +28 -0
- package/docs/clients/claude-desktop.md +35 -0
- package/docs/clients/codex.md +28 -0
- package/docs/clients/cursor.md +26 -0
- package/docs/clients/opencode.md +35 -0
- package/docs/clients/vscode.md +25 -0
- package/package.json +49 -0
- package/query-access.example.json +21 -0
- package/src/cli.ts +221 -0
- package/src/core/cache.ts +41 -0
- package/src/core/config.ts +97 -0
- package/src/core/database-url.test.ts +28 -0
- package/src/core/database-url.ts +47 -0
- package/src/core/logger.ts +24 -0
- package/src/core/mcp.ts +23 -0
- package/src/core/query-safety.test.ts +36 -0
- package/src/core/query-safety.ts +63 -0
- package/src/core/schema-types.ts +58 -0
- package/src/core/security/access-control.ts +321 -0
- package/src/core/security/config-loader.ts +315 -0
- package/src/core/security/types.ts +114 -0
- package/src/core/sql-parser.test.ts +37 -0
- package/src/core/sql-parser.ts +572 -0
- package/src/handlers/accessible-schema.ts +314 -0
- package/src/handlers/data.ts +66 -0
- package/src/handlers/relationships.ts +114 -0
- package/src/handlers/schema.ts +154 -0
- package/src/handlers/search.ts +34 -0
- package/src/handlers/validation.ts +165 -0
- package/src/index.ts +337 -0
- package/src/mysql/connection.ts +68 -0
- package/src/mysql/identifiers.test.ts +12 -0
- package/src/mysql/identifiers.ts +10 -0
- package/src/mysql/queries.ts +285 -0
- 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":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACzC,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAKnE,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EACL,UAAU,EACV,cAAc,EACd,UAAU,EACV,cAAc,EACd,aAAa,EACb,UAAU,GACX,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AAEzD,SAAS,OAAO,CAAkC,IAAS;IACzD,MAAM,GAAG,GAAG,IAAI,GAAG,EAAe,CAAC;IACnC,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,IAAI,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;QAC1C,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACf,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;IAC/B,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,KAAK,UAAU,aAAa,CAC1B,UAAqB,EACrB,oBAAoB,GAAG,IAAI,EAC3B,iBAAiB,GAAG,KAAK;IAEzB,MAAM,MAAM,GAAG,MAAM,UAAU,CAAC,UAAU,CAAC,CAAC;IAC5C,MAAM,gBAAgB,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAEhE,MAAM,CAAC,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;QACjF,UAAU,CAAC,gBAAgB,CAAC;QAC5B,cAAc,CAAC,gBAAgB,CAAC;QAChC,oBAAoB,CAAC,CAAC,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QAC7E,UAAU,CAAC,gBAAgB,CAAC;QAC5B,iBAAiB,CAAC,CAAC,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;KAC1E,CAAC,CAAC;IAEH,MAAM,cAAc,GAAG,OAAO,CAAC,OAAgB,CAAC,CAAC;IACjD,MAAM,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;IACxC,MAAM,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;IACxC,MAAM,cAAc,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;IACxC,MAAM,YAAY,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;IAEzC,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;QAC1B,MAAM,QAAQ,GAAkB;YAC9B,MAAM,EAAE,KAAK,CAAC,UAAU;YACxB,IAAI,EAAE,KAAK,CAAC,SAAS;YACrB,IAAI,EAAE,KAAK,CAAC,SAAS,KAAK,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO;YACnD,OAAO,EAAE,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,MAAM,EAAE,EAAE,EAAE,CAAC,CAAC;gBAClG,GAAG,MAAM;gBACT,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC;gBAClC,UAAU,EAAE,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC;gBACtC,UAAU,EAAE,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC;gBACtC,YAAY,EAAE,OAAO,CAAC,MAAM,CAAC,YAAY,CAAC;gBAC1C,YAAY,EAAE,OAAO,CAAC,MAAM,CAAC,YAAY,CAAC;aAC3C,CAAC,CAAC;YACH,OAAO,EAAE,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;gBACjG,GAAG,KAAK;gBACR,QAAQ,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC;gBACjC,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC;aAC1C,CAAC,CAAC;SACJ,CAAC;QAEF,MAAM,UAAU,GAAG,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACxD,IAAI,UAAU,EAAE,CAAC;YACf,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,GAAG,UAAU,CAAC;YACpD,QAAQ,CAAC,UAAU,GAAG,EAAE,CAAC;QAC3B,CAAC;QAED,IAAI,oBAAoB,EAAE,CAAC;YACzB,QAAQ,CAAC,WAAW,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC;QAC/G,CAAC;QAED,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACrD,IAAI,iBAAiB,IAAI,KAAK,EAAE,CAAC;YAC/B,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,UAAU,EAAE,GAAG,KAAK,CAAC;YACvD,QAAQ,CAAC,UAAU,GAAG,UAAU,CAAC;QACnC,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,IAM/B;IACC,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChD,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAE3B,MAAM,EACJ,MAAM,EACN,oBAAoB,GAAG,IAAI,EAC3B,iBAAiB,GAAG,KAAK,GAC1B,GAAG,IAAI,CAAC;IAET,MAAM,QAAQ,GAAG,UAAU,QAAQ,IAAI,MAAM,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,KAAK,IAAI,oBAAoB,IAAI,iBAAiB,EAAE,CAAC;IACjH,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAe,QAAQ,CAAC,CAAC;IACjD,IAAI,MAAM,EAAE,CAAC;QACX,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,IAAI,MAAM,EAAE,MAAM,EAAE,CAAC;QACnB,MAAM,UAAU,GAAG,MAAM,OAAO,CAAC,GAAG,CAClC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,sBAAsB,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC,CAC/D,CAAC;QACF,MAAM,OAAO,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC7D,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;QACtE,CAAC;IACH,CAAC;IAED,MAAM,MAAM,GAAG;QACb,MAAM,EAAE,MAAM,aAAa,CAAC,MAAM,EAAE,oBAAoB,EAAE,iBAAiB,CAAC;KAC7E,CAAC;IAEF,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC5B,MAAM,CAAC,IAAI,CAAC,wBAAwB,MAAM,CAAC,MAAM,CAAC,MAAM,iBAAiB,QAAQ,EAAE,CAAC,CAAC;IACrF,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,IAIlC;IACC,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChD,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAE3B,MAAM,UAAU,GAAG,MAAM,sBAAsB,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IACtE,IAAI,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,UAAU,EAAE,CAAC;QACvD,MAAM,IAAI,KAAK,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;IACtC,CAAC;IAED,MAAM,SAAS,GAAG,UAAU,CAAC,KAAK,CAAC,UAAU,CAAC,QAAQ,CAAC,GAAG,CAAC;QACzD,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAG;QAC/C,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,UAAU,CAAC;IAChC,MAAM,QAAQ,GAAG,MAAM,aAAa,CAAC,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC;IAE/D,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;QACjB,MAAM,IAAI,KAAK,CAAC,UAAU,IAAI,CAAC,KAAK,4BAA4B,QAAQ,GAAG,CAAC,CAAC;IAC/E,CAAC;IAED,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC;AACrB,CAAC"}
|
|
@@ -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"}
|