@rashidazarang/airtable-mcp 3.1.0 → 3.2.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.
- package/README.md +62 -25
- package/bin/airtable-mcp.js +12 -32
- package/dist/typescript/airtable-mcp-server.js +77 -0
- package/dist/typescript/airtable-mcp-server.js.map +1 -0
- package/dist/typescript/app/airtable-client.js +325 -0
- package/dist/typescript/app/airtable-client.js.map +1 -0
- package/dist/typescript/app/config.js +141 -0
- package/dist/typescript/app/config.js.map +1 -0
- package/dist/typescript/app/context.js +3 -0
- package/dist/typescript/app/context.js.map +1 -0
- package/dist/typescript/app/exceptions.js +85 -0
- package/dist/typescript/app/exceptions.js.map +1 -0
- package/dist/typescript/app/governance.js +58 -0
- package/dist/typescript/app/governance.js.map +1 -0
- package/dist/typescript/app/logger.js +47 -0
- package/dist/typescript/app/logger.js.map +1 -0
- package/dist/typescript/app/rateLimiter.js +37 -0
- package/dist/typescript/app/rateLimiter.js.map +1 -0
- package/dist/typescript/app/tools/create.js +54 -0
- package/dist/typescript/app/tools/create.js.map +1 -0
- package/dist/typescript/app/tools/describe.js +146 -0
- package/dist/typescript/app/tools/describe.js.map +1 -0
- package/dist/typescript/app/tools/handleError.js +54 -0
- package/dist/typescript/app/tools/handleError.js.map +1 -0
- package/dist/typescript/app/tools/index.js +24 -0
- package/dist/typescript/app/tools/index.js.map +1 -0
- package/dist/typescript/app/tools/listBases.js +52 -0
- package/dist/typescript/app/tools/listBases.js.map +1 -0
- package/dist/typescript/app/tools/listExceptions.js +18 -0
- package/dist/typescript/app/tools/listExceptions.js.map +1 -0
- package/dist/typescript/app/tools/listGovernance.js +17 -0
- package/dist/typescript/app/tools/listGovernance.js.map +1 -0
- package/dist/typescript/app/tools/query.js +126 -0
- package/dist/typescript/app/tools/query.js.map +1 -0
- package/dist/typescript/app/tools/update.js +56 -0
- package/dist/typescript/app/tools/update.js.map +1 -0
- package/dist/typescript/app/tools/upsert.js +65 -0
- package/dist/typescript/app/tools/upsert.js.map +1 -0
- package/dist/typescript/app/tools/webhooks.js +44 -0
- package/dist/typescript/app/tools/webhooks.js.map +1 -0
- package/dist/typescript/app/types.js +282 -0
- package/dist/typescript/app/types.js.map +1 -0
- package/dist/typescript/apps-sdk/mappers.js +70 -0
- package/dist/typescript/apps-sdk/mappers.js.map +1 -0
- package/dist/typescript/errors.js +75 -0
- package/dist/typescript/errors.js.map +1 -0
- package/dist/typescript/index.js +27 -0
- package/dist/typescript/index.js.map +1 -0
- package/package.json +49 -30
- package/tsconfig.json +10 -4
- package/types/typescript/airtable-mcp-server.d.ts +2 -0
- package/types/typescript/app/airtable-client.d.ts +49 -0
- package/types/typescript/app/config.d.ts +16 -0
- package/types/typescript/app/context.d.ts +12 -0
- package/types/typescript/app/exceptions.d.ts +12 -0
- package/types/typescript/app/governance.d.ts +18 -0
- package/types/typescript/app/logger.d.ts +13 -0
- package/types/typescript/app/rateLimiter.d.ts +13 -0
- package/types/typescript/app/tools/create.d.ts +3 -0
- package/types/typescript/app/tools/describe.d.ts +3 -0
- package/types/typescript/app/tools/handleError.d.ts +8 -0
- package/types/typescript/app/tools/index.d.ts +3 -0
- package/types/typescript/app/tools/listBases.d.ts +33 -0
- package/types/typescript/app/tools/listExceptions.d.ts +3 -0
- package/types/typescript/app/tools/listGovernance.d.ts +3 -0
- package/types/typescript/app/tools/query.d.ts +3 -0
- package/types/typescript/app/tools/update.d.ts +3 -0
- package/types/typescript/app/tools/upsert.d.ts +3 -0
- package/types/typescript/app/tools/webhooks.d.ts +3 -0
- package/types/typescript/app/types.d.ts +830 -0
- package/types/typescript/apps-sdk/mappers.d.ts +53 -0
- package/types/typescript/errors.d.ts +55 -0
- package/types/typescript/index.d.ts +10 -0
- package/types/typescript/prompt-templates.d.ts +5 -0
- package/types/typescript/tools-schemas.d.ts +5 -0
- package/airtable_simple.js +0 -1561
- package/airtable_simple_production.js +0 -1564
- package/dist/airtable-mcp-server.js +0 -660
- package/dist/airtable-mcp-server.js.map +0 -1
- package/dist/test-suite.js +0 -421
- package/dist/test-suite.js.map +0 -1
- package/examples/airtable-crud-example.js +0 -203
- package/examples/building-mcp.md +0 -6666
- package/examples/claude_config.json +0 -4
- package/examples/claude_simple_config.json +0 -7
- package/examples/env-demo.js +0 -172
- package/examples/example-tasks-update.json +0 -23
- package/examples/example-tasks.json +0 -26
- package/examples/example_usage.md +0 -124
- package/examples/python_debug_patch.txt +0 -27
- package/examples/sample-transform.js +0 -76
- package/examples/typescript/advanced-ai-prompts.ts +0 -447
- package/examples/typescript/basic-usage.ts +0 -174
- package/examples/typescript/claude-desktop-config.json +0 -29
- package/examples/windsurf_mcp_config.json +0 -17
- package/types/ai-prompts.d.ts +0 -321
- package/types/airtable-mcp-server.d.ts +0 -52
- package/types/index.d.ts +0 -357
- package/types/tools.d.ts +0 -514
- /package/types/{test-suite.d.ts → typescript/test-suite.d.ts} +0 -0
|
@@ -0,0 +1,141 @@
|
|
|
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.loadConfig = loadConfig;
|
|
7
|
+
const node_fs_1 = __importDefault(require("node:fs"));
|
|
8
|
+
const node_path_1 = __importDefault(require("node:path"));
|
|
9
|
+
const node_crypto_1 = require("node:crypto");
|
|
10
|
+
const dotenv_1 = require("dotenv");
|
|
11
|
+
const types_1 = require("./types");
|
|
12
|
+
const errors_1 = require("../errors");
|
|
13
|
+
(0, dotenv_1.config)();
|
|
14
|
+
const DEFAULT_EXCEPTION_QUEUE_SIZE = 500;
|
|
15
|
+
function parseCsv(value) {
|
|
16
|
+
if (!value) {
|
|
17
|
+
return [];
|
|
18
|
+
}
|
|
19
|
+
return value
|
|
20
|
+
.split(',')
|
|
21
|
+
.map((entry) => entry.trim())
|
|
22
|
+
.filter((entry) => entry.length > 0);
|
|
23
|
+
}
|
|
24
|
+
function hashSecret(secret) {
|
|
25
|
+
return (0, node_crypto_1.createHash)('sha256').update(secret).digest('hex').slice(0, 12);
|
|
26
|
+
}
|
|
27
|
+
function resolveLogLevel() {
|
|
28
|
+
const raw = (process.env.LOG_LEVEL || 'info').toLowerCase();
|
|
29
|
+
if (raw === 'error' || raw === 'warn' || raw === 'info' || raw === 'debug') {
|
|
30
|
+
return raw;
|
|
31
|
+
}
|
|
32
|
+
return 'info';
|
|
33
|
+
}
|
|
34
|
+
function determineAllowedBases(defaultBaseId) {
|
|
35
|
+
const fromEnv = parseCsv(process.env.AIRTABLE_ALLOWED_BASES || process.env.AIRTABLE_BASE_ALLOWLIST);
|
|
36
|
+
const baseSet = new Set();
|
|
37
|
+
if (defaultBaseId) {
|
|
38
|
+
baseSet.add(defaultBaseId);
|
|
39
|
+
}
|
|
40
|
+
fromEnv.forEach((base) => baseSet.add(base));
|
|
41
|
+
// Allow empty base list - users can use list_bases tool to discover bases
|
|
42
|
+
// and then specify them dynamically in tool calls
|
|
43
|
+
return Array.from(baseSet);
|
|
44
|
+
}
|
|
45
|
+
function parseAllowedTables(raw) {
|
|
46
|
+
if (!raw) {
|
|
47
|
+
return [];
|
|
48
|
+
}
|
|
49
|
+
const tables = [];
|
|
50
|
+
for (const entry of raw.split(',')) {
|
|
51
|
+
const trimmed = entry.trim();
|
|
52
|
+
if (!trimmed)
|
|
53
|
+
continue;
|
|
54
|
+
const [baseId, table] = trimmed.split(':');
|
|
55
|
+
if (!baseId || !table) {
|
|
56
|
+
throw new errors_1.GovernanceError(`Invalid AIRTABLE_ALLOWED_TABLES entry "${trimmed}". Expected format baseId:tableName.`);
|
|
57
|
+
}
|
|
58
|
+
tables.push({ baseId: baseId.trim(), table: table.trim() });
|
|
59
|
+
}
|
|
60
|
+
return tables;
|
|
61
|
+
}
|
|
62
|
+
function readGovernanceFile() {
|
|
63
|
+
const explicitPath = process.env.AIRTABLE_GOVERNANCE_PATH;
|
|
64
|
+
const fallbackPath = node_path_1.default.resolve(process.cwd(), 'config', 'governance.json');
|
|
65
|
+
const filePath = explicitPath || fallbackPath;
|
|
66
|
+
if (!node_fs_1.default.existsSync(filePath)) {
|
|
67
|
+
return undefined;
|
|
68
|
+
}
|
|
69
|
+
try {
|
|
70
|
+
const raw = node_fs_1.default.readFileSync(filePath, 'utf8');
|
|
71
|
+
const parsed = JSON.parse(raw);
|
|
72
|
+
const partialSchema = types_1.governanceOutputSchema.partial();
|
|
73
|
+
const result = partialSchema.parse(parsed);
|
|
74
|
+
return result;
|
|
75
|
+
}
|
|
76
|
+
catch (error) {
|
|
77
|
+
throw new errors_1.GovernanceError(`Failed to parse governance configuration at ${filePath}: ${error instanceof Error ? error.message : String(error)}`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
function buildGovernanceSnapshot(allowedBases) {
|
|
81
|
+
const baseSnapshot = {
|
|
82
|
+
allowedBases,
|
|
83
|
+
allowedTables: [],
|
|
84
|
+
allowedOperations: ['describe', 'query', 'create', 'update', 'upsert'],
|
|
85
|
+
piiFields: [],
|
|
86
|
+
redactionPolicy: 'mask_on_inline',
|
|
87
|
+
loggingPolicy: 'minimal',
|
|
88
|
+
retentionDays: 7
|
|
89
|
+
};
|
|
90
|
+
const overrides = readGovernanceFile();
|
|
91
|
+
const envAllowedTables = parseAllowedTables(process.env.AIRTABLE_ALLOWED_TABLES);
|
|
92
|
+
const merged = {
|
|
93
|
+
...baseSnapshot,
|
|
94
|
+
...(overrides ?? {})
|
|
95
|
+
};
|
|
96
|
+
// Ensure allow-lists include env tables/bases.
|
|
97
|
+
const bases = new Set(merged.allowedBases);
|
|
98
|
+
allowedBases.forEach((base) => bases.add(base));
|
|
99
|
+
merged.allowedBases = Array.from(bases);
|
|
100
|
+
if (overrides?.allowedTables || envAllowedTables.length > 0) {
|
|
101
|
+
const tableSet = new Map();
|
|
102
|
+
(overrides?.allowedTables ?? []).forEach((table) => {
|
|
103
|
+
tableSet.set(`${table.baseId}:${table.table}`, table);
|
|
104
|
+
});
|
|
105
|
+
envAllowedTables.forEach((table) => {
|
|
106
|
+
tableSet.set(`${table.baseId}:${table.table}`, table);
|
|
107
|
+
});
|
|
108
|
+
merged.allowedTables = Array.from(tableSet.values());
|
|
109
|
+
}
|
|
110
|
+
return types_1.governanceOutputSchema.parse(merged);
|
|
111
|
+
}
|
|
112
|
+
function loadConfig() {
|
|
113
|
+
const personalAccessToken = process.env.AIRTABLE_PAT ||
|
|
114
|
+
process.env.AIRTABLE_TOKEN ||
|
|
115
|
+
process.env.AIRTABLE_API_TOKEN ||
|
|
116
|
+
process.env.AIRTABLE_API_KEY;
|
|
117
|
+
if (!personalAccessToken) {
|
|
118
|
+
throw new errors_1.GovernanceError('Missing Airtable credentials. Set AIRTABLE_PAT (preferred) or AIRTABLE_TOKEN.');
|
|
119
|
+
}
|
|
120
|
+
const defaultBaseId = process.env.AIRTABLE_DEFAULT_BASE ?? process.env.AIRTABLE_BASE_ID ?? process.env.AIRTABLE_BASE;
|
|
121
|
+
const allowedBases = determineAllowedBases(defaultBaseId);
|
|
122
|
+
const governance = buildGovernanceSnapshot(allowedBases);
|
|
123
|
+
const auth = {
|
|
124
|
+
personalAccessToken,
|
|
125
|
+
patHash: hashSecret(personalAccessToken),
|
|
126
|
+
allowedBases
|
|
127
|
+
};
|
|
128
|
+
if (defaultBaseId) {
|
|
129
|
+
auth.defaultBaseId = defaultBaseId;
|
|
130
|
+
}
|
|
131
|
+
return {
|
|
132
|
+
version: process.env.npm_package_version || '0.0.0',
|
|
133
|
+
auth,
|
|
134
|
+
governance,
|
|
135
|
+
logLevel: resolveLogLevel(),
|
|
136
|
+
exceptionQueueSize: Number.parseInt(process.env.EXCEPTION_QUEUE_SIZE || '', 10) > 0
|
|
137
|
+
? Number.parseInt(process.env.EXCEPTION_QUEUE_SIZE, 10)
|
|
138
|
+
: DEFAULT_EXCEPTION_QUEUE_SIZE
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../src/typescript/app/config.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.js","sourceRoot":"","sources":["../../../src/typescript/app/context.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ExceptionStore = void 0;
|
|
4
|
+
const node_crypto_1 = require("node:crypto");
|
|
5
|
+
function mapCategory(code) {
|
|
6
|
+
switch (code) {
|
|
7
|
+
case 'RateLimited':
|
|
8
|
+
return 'rate_limit';
|
|
9
|
+
case 'ValidationError':
|
|
10
|
+
return 'validation';
|
|
11
|
+
case 'AuthError':
|
|
12
|
+
return 'auth';
|
|
13
|
+
case 'ConflictError':
|
|
14
|
+
return 'conflict';
|
|
15
|
+
case 'GovernanceError':
|
|
16
|
+
return 'schema_drift';
|
|
17
|
+
default:
|
|
18
|
+
return 'other';
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
function mapSeverity(code) {
|
|
22
|
+
switch (code) {
|
|
23
|
+
case 'RateLimited':
|
|
24
|
+
case 'AuthError':
|
|
25
|
+
case 'ConflictError':
|
|
26
|
+
case 'GovernanceError':
|
|
27
|
+
return 'error';
|
|
28
|
+
case 'ValidationError':
|
|
29
|
+
return 'warning';
|
|
30
|
+
default:
|
|
31
|
+
return 'error';
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
class ExceptionStore {
|
|
35
|
+
constructor(capacity, logger) {
|
|
36
|
+
this.items = [];
|
|
37
|
+
this.capacity = capacity;
|
|
38
|
+
this.logger = logger.child({ component: 'exception_store' });
|
|
39
|
+
}
|
|
40
|
+
record(error, summary, details, proposedFix) {
|
|
41
|
+
const item = {
|
|
42
|
+
id: (0, node_crypto_1.randomUUID)(),
|
|
43
|
+
timestamp: new Date().toISOString(),
|
|
44
|
+
severity: mapSeverity(error.code),
|
|
45
|
+
category: mapCategory(error.code),
|
|
46
|
+
summary,
|
|
47
|
+
details,
|
|
48
|
+
proposedFix
|
|
49
|
+
};
|
|
50
|
+
this.items.unshift(item);
|
|
51
|
+
if (this.items.length > this.capacity) {
|
|
52
|
+
this.items.pop();
|
|
53
|
+
}
|
|
54
|
+
this.logger.debug('Recorded exception', { code: error.code });
|
|
55
|
+
}
|
|
56
|
+
list(params) {
|
|
57
|
+
const limit = params.limit ?? 100;
|
|
58
|
+
const cursorIndex = this.parseCursor(params.cursor);
|
|
59
|
+
let filtered = this.items;
|
|
60
|
+
if (params.since) {
|
|
61
|
+
filtered = filtered.filter((item) => item.timestamp > params.since);
|
|
62
|
+
}
|
|
63
|
+
if (params.severity) {
|
|
64
|
+
filtered = filtered.filter((item) => item.severity === params.severity);
|
|
65
|
+
}
|
|
66
|
+
const slice = filtered.slice(cursorIndex, cursorIndex + limit);
|
|
67
|
+
const nextCursor = cursorIndex + limit < filtered.length ? String(cursorIndex + limit) : undefined;
|
|
68
|
+
return {
|
|
69
|
+
items: slice,
|
|
70
|
+
cursor: nextCursor
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
parseCursor(cursor) {
|
|
74
|
+
if (!cursor) {
|
|
75
|
+
return 0;
|
|
76
|
+
}
|
|
77
|
+
const parsed = Number.parseInt(cursor, 10);
|
|
78
|
+
if (Number.isNaN(parsed) || parsed < 0) {
|
|
79
|
+
return 0;
|
|
80
|
+
}
|
|
81
|
+
return parsed;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
exports.ExceptionStore = ExceptionStore;
|
|
85
|
+
//# sourceMappingURL=exceptions.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"exceptions.js","sourceRoot":"","sources":["../../../src/typescript/app/exceptions.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GovernanceService = void 0;
|
|
4
|
+
const errors_1 = require("../errors");
|
|
5
|
+
class GovernanceService {
|
|
6
|
+
constructor(snapshot) {
|
|
7
|
+
this.snapshot = snapshot;
|
|
8
|
+
this.tablesByBase = this.buildTableIndex(snapshot);
|
|
9
|
+
}
|
|
10
|
+
ensureBaseAllowed(baseId) {
|
|
11
|
+
// If allowedBases is empty, allow all bases (user will use list_bases to discover)
|
|
12
|
+
if (this.snapshot.allowedBases.length > 0 && !this.snapshot.allowedBases.includes(baseId)) {
|
|
13
|
+
throw new errors_1.GovernanceError(`Base ${baseId} is not in the allow-list`, {
|
|
14
|
+
context: { baseId, governanceRule: 'allowedBases' }
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
ensureOperationAllowed(operation) {
|
|
19
|
+
if (!this.snapshot.allowedOperations.includes(operation)) {
|
|
20
|
+
throw new errors_1.GovernanceError(`Operation ${operation} is not permitted`, {
|
|
21
|
+
context: { governanceRule: 'allowedOperations' }
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
ensureTableAllowed(baseId, table) {
|
|
26
|
+
if (!this.isTableAllowed(baseId, table)) {
|
|
27
|
+
throw new errors_1.GovernanceError(`Table ${table} is not allowed in base ${baseId}`, {
|
|
28
|
+
context: { baseId, table, governanceRule: 'allowedTables' }
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
listPiiPolicies(baseId, table) {
|
|
33
|
+
return this.snapshot.piiFields
|
|
34
|
+
?.filter((field) => field.baseId === baseId && field.table === table)
|
|
35
|
+
.map((field) => ({ field: field.field, policy: field.policy })) ?? [];
|
|
36
|
+
}
|
|
37
|
+
getSnapshot() {
|
|
38
|
+
return this.snapshot;
|
|
39
|
+
}
|
|
40
|
+
isTableAllowed(baseId, table) {
|
|
41
|
+
const allowedTables = this.tablesByBase.get(baseId);
|
|
42
|
+
if (!allowedTables || allowedTables.size === 0) {
|
|
43
|
+
return true;
|
|
44
|
+
}
|
|
45
|
+
return allowedTables.has(table);
|
|
46
|
+
}
|
|
47
|
+
buildTableIndex(snapshot) {
|
|
48
|
+
const map = new Map();
|
|
49
|
+
for (const item of snapshot.allowedTables ?? []) {
|
|
50
|
+
const baseTables = map.get(item.baseId) ?? new Set();
|
|
51
|
+
baseTables.add(item.table);
|
|
52
|
+
map.set(item.baseId, baseTables);
|
|
53
|
+
}
|
|
54
|
+
return map;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
exports.GovernanceService = GovernanceService;
|
|
58
|
+
//# sourceMappingURL=governance.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"governance.js","sourceRoot":"","sources":["../../../src/typescript/app/governance.ts"],"names":[],"mappings":";;;AACA,sCAA4C;AAI5C,MAAa,iBAAiB;IAI5B,YAAY,QAA4B;QACtC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;IACrD,CAAC;IAED,iBAAiB,CAAC,MAAc;QAC9B,mFAAmF;QACnF,IAAI,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YAC1F,MAAM,IAAI,wBAAe,CAAC,QAAQ,MAAM,2BAA2B,EAAE;gBACnE,OAAO,EAAE,EAAE,MAAM,EAAE,cAAc,EAAE,cAAc,EAAE;aACpD,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,sBAAsB,CAAC,SAAoB;QACzC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,iBAAiB,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACzD,MAAM,IAAI,wBAAe,CAAC,aAAa,SAAS,mBAAmB,EAAE;gBACnE,OAAO,EAAE,EAAE,cAAc,EAAE,mBAAmB,EAAE;aACjD,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,kBAAkB,CAAC,MAAc,EAAE,KAAa;QAC9C,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,CAAC;YACxC,MAAM,IAAI,wBAAe,CAAC,SAAS,KAAK,2BAA2B,MAAM,EAAE,EAAE;gBAC3E,OAAO,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE;aAC5D,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,eAAe,CAAC,MAAc,EAAE,KAAa;QAC3C,OAAO,IAAI,CAAC,QAAQ,CAAC,SAAS;YAC5B,EAAE,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,MAAM,KAAK,MAAM,IAAI,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC;aACpE,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAC1E,CAAC;IAED,WAAW;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED,cAAc,CAAC,MAAc,EAAE,KAAa;QAC1C,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACpD,IAAI,CAAC,aAAa,IAAI,aAAa,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;YAC/C,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAClC,CAAC;IAEO,eAAe,CAAC,QAA4B;QAClD,MAAM,GAAG,GAAG,IAAI,GAAG,EAAuB,CAAC;QAC3C,KAAK,MAAM,IAAI,IAAI,QAAQ,CAAC,aAAa,IAAI,EAAE,EAAE,CAAC;YAChD,MAAM,UAAU,GAAG,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,GAAG,EAAU,CAAC;YAC7D,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;QACnC,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC;CACF;AA7DD,8CA6DC"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Logger = void 0;
|
|
4
|
+
const LEVEL_ORDER = {
|
|
5
|
+
error: 0,
|
|
6
|
+
warn: 1,
|
|
7
|
+
info: 2,
|
|
8
|
+
debug: 3
|
|
9
|
+
};
|
|
10
|
+
class Logger {
|
|
11
|
+
constructor(level, context = {}) {
|
|
12
|
+
this.level = level;
|
|
13
|
+
this.context = context;
|
|
14
|
+
}
|
|
15
|
+
child(context) {
|
|
16
|
+
return new Logger(this.level, { ...this.context, ...context });
|
|
17
|
+
}
|
|
18
|
+
error(message, metadata = {}) {
|
|
19
|
+
this.log('error', message, metadata);
|
|
20
|
+
}
|
|
21
|
+
warn(message, metadata = {}) {
|
|
22
|
+
this.log('warn', message, metadata);
|
|
23
|
+
}
|
|
24
|
+
info(message, metadata = {}) {
|
|
25
|
+
this.log('info', message, metadata);
|
|
26
|
+
}
|
|
27
|
+
debug(message, metadata = {}) {
|
|
28
|
+
this.log('debug', message, metadata);
|
|
29
|
+
}
|
|
30
|
+
log(level, message, metadata) {
|
|
31
|
+
if (LEVEL_ORDER[level] > LEVEL_ORDER[this.level]) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
const timestamp = new Date().toISOString();
|
|
35
|
+
const output = {
|
|
36
|
+
timestamp,
|
|
37
|
+
level,
|
|
38
|
+
message,
|
|
39
|
+
...this.context,
|
|
40
|
+
...(Object.keys(metadata).length > 0 ? { metadata } : {})
|
|
41
|
+
};
|
|
42
|
+
// eslint-disable-next-line no-console
|
|
43
|
+
console.log(JSON.stringify(output));
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
exports.Logger = Logger;
|
|
47
|
+
//# sourceMappingURL=logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../../src/typescript/app/logger.ts"],"names":[],"mappings":";;;AAEA,MAAM,WAAW,GAA6B;IAC5C,KAAK,EAAE,CAAC;IACR,IAAI,EAAE,CAAC;IACP,IAAI,EAAE,CAAC;IACP,KAAK,EAAE,CAAC;CACT,CAAC;AAIF,MAAa,MAAM;IAIjB,YAAY,KAAe,EAAE,UAAuB,EAAE;QACpD,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,OAAoB;QACxB,OAAO,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC,CAAC;IACjE,CAAC;IAED,KAAK,CAAC,OAAe,EAAE,WAAwB,EAAE;QAC/C,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACvC,CAAC;IAED,IAAI,CAAC,OAAe,EAAE,WAAwB,EAAE;QAC9C,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACtC,CAAC;IAED,IAAI,CAAC,OAAe,EAAE,WAAwB,EAAE;QAC9C,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,OAAe,EAAE,WAAwB,EAAE;QAC/C,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACvC,CAAC;IAEO,GAAG,CAAC,KAAe,EAAE,OAAe,EAAE,QAAqB;QACjE,IAAI,WAAW,CAAC,KAAK,CAAC,GAAG,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACjD,OAAO;QACT,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAC3C,MAAM,MAAM,GAAG;YACb,SAAS;YACT,KAAK;YACL,OAAO;YACP,GAAG,IAAI,CAAC,OAAO;YACf,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SAC1D,CAAC;QAEF,sCAAsC;QACtC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;IACtC,CAAC;CACF;AA9CD,wBA8CC"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.RateLimiter = void 0;
|
|
4
|
+
const promises_1 = require("node:timers/promises");
|
|
5
|
+
/**
|
|
6
|
+
* Lightweight token-based rate limiter to enforce Airtable quotas.
|
|
7
|
+
* Maintains per-key queues to preserve ordering and fairness.
|
|
8
|
+
*/
|
|
9
|
+
class RateLimiter {
|
|
10
|
+
constructor({ maxRequestsPerSecond }) {
|
|
11
|
+
this.lockByKey = new Map();
|
|
12
|
+
this.nextAvailableByKey = new Map();
|
|
13
|
+
if (maxRequestsPerSecond <= 0) {
|
|
14
|
+
throw new Error('maxRequestsPerSecond must be greater than zero');
|
|
15
|
+
}
|
|
16
|
+
this.minIntervalMs = Math.ceil(1000 / maxRequestsPerSecond);
|
|
17
|
+
}
|
|
18
|
+
async schedule(key) {
|
|
19
|
+
const previous = this.lockByKey.get(key) ?? Promise.resolve();
|
|
20
|
+
let release = () => undefined;
|
|
21
|
+
const current = new Promise((resolve) => {
|
|
22
|
+
release = resolve;
|
|
23
|
+
});
|
|
24
|
+
this.lockByKey.set(key, previous.then(() => current));
|
|
25
|
+
await previous;
|
|
26
|
+
const now = Date.now();
|
|
27
|
+
const availableAt = this.nextAvailableByKey.get(key) ?? now;
|
|
28
|
+
const waitMs = Math.max(availableAt - now, 0);
|
|
29
|
+
if (waitMs > 0) {
|
|
30
|
+
await (0, promises_1.setTimeout)(waitMs);
|
|
31
|
+
}
|
|
32
|
+
this.nextAvailableByKey.set(key, Date.now() + this.minIntervalMs);
|
|
33
|
+
release();
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
exports.RateLimiter = RateLimiter;
|
|
37
|
+
//# sourceMappingURL=rateLimiter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rateLimiter.js","sourceRoot":"","sources":["../../../src/typescript/app/rateLimiter.ts"],"names":[],"mappings":";;;AAAA,mDAA2D;AAE3D;;;GAGG;AACH,MAAa,WAAW;IAKtB,YAAY,EAAE,oBAAoB,EAAoC;QAHrD,cAAS,GAAG,IAAI,GAAG,EAAyB,CAAC;QAC7C,uBAAkB,GAAG,IAAI,GAAG,EAAkB,CAAC;QAG9D,IAAI,oBAAoB,IAAI,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,gDAAgD,CAAC,CAAC;QACpE,CAAC;QACD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC,CAAC;IAC9D,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,GAAW;QACxB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;QAC9D,IAAI,OAAO,GAAe,GAAG,EAAE,CAAC,SAAS,CAAC;QAC1C,MAAM,OAAO,GAAG,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;YAC5C,OAAO,GAAG,OAAO,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,CAAC,GAAG,CAChB,GAAG,EACH,QAAQ,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,CAC7B,CAAC;QAEF,MAAM,QAAQ,CAAC;QAEf,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC;QAC5D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,WAAW,GAAG,GAAG,EAAE,CAAC,CAAC,CAAC;QAC9C,IAAI,MAAM,GAAG,CAAC,EAAE,CAAC;YACf,MAAM,IAAA,qBAAK,EAAC,MAAM,CAAC,CAAC;QACtB,CAAC;QAED,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;QAClE,OAAO,EAAE,CAAC;IACZ,CAAC;CACF;AApCD,kCAoCC"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerCreateTool = registerCreateTool;
|
|
4
|
+
const types_1 = require("../types");
|
|
5
|
+
const handleError_1 = require("./handleError");
|
|
6
|
+
function chunk(arr, size) {
|
|
7
|
+
const out = [];
|
|
8
|
+
for (let i = 0; i < arr.length; i += size)
|
|
9
|
+
out.push(arr.slice(i, i + size));
|
|
10
|
+
return out;
|
|
11
|
+
}
|
|
12
|
+
function registerCreateTool(server, ctx) {
|
|
13
|
+
server.registerTool('create', {
|
|
14
|
+
description: 'Create Airtable records (requires diff-before-write via dryRun first).',
|
|
15
|
+
inputSchema: types_1.createInputSchema.shape,
|
|
16
|
+
outputSchema: types_1.createOutputSchema.shape
|
|
17
|
+
}, async (raw) => {
|
|
18
|
+
try {
|
|
19
|
+
const args = types_1.createInputSchema.parse(raw);
|
|
20
|
+
ctx.governance.ensureOperationAllowed('create');
|
|
21
|
+
ctx.governance.ensureBaseAllowed(args.baseId);
|
|
22
|
+
ctx.governance.ensureTableAllowed(args.baseId, args.table);
|
|
23
|
+
const logger = ctx.logger.child({ tool: 'create', baseId: args.baseId, table: args.table });
|
|
24
|
+
if (args.dryRun) {
|
|
25
|
+
const structuredContent = {
|
|
26
|
+
diff: { added: args.records.length, updated: 0, unchanged: 0 },
|
|
27
|
+
dryRun: true,
|
|
28
|
+
records: args.records.map((r) => ({ id: 'pending', fields: r.fields }))
|
|
29
|
+
};
|
|
30
|
+
return { structuredContent, content: [] };
|
|
31
|
+
}
|
|
32
|
+
const chunks = chunk(args.records, 10);
|
|
33
|
+
const aggregated = [];
|
|
34
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
35
|
+
const body = { records: chunks[i], typecast: args.typecast ?? false };
|
|
36
|
+
const headerKey = args.idempotencyKey ? `${args.idempotencyKey}:${i}` : undefined;
|
|
37
|
+
const response = await ctx.airtable.createRecords(args.baseId, args.table, body, headerKey);
|
|
38
|
+
if (Array.isArray(response?.records))
|
|
39
|
+
aggregated.push(...response.records);
|
|
40
|
+
}
|
|
41
|
+
const structuredContent = {
|
|
42
|
+
diff: { added: aggregated.length, updated: 0, unchanged: 0 },
|
|
43
|
+
records: aggregated.map((r) => ({ id: String(r.id), fields: r.fields || {} })),
|
|
44
|
+
dryRun: false
|
|
45
|
+
};
|
|
46
|
+
logger.info('Create completed', { added: aggregated.length });
|
|
47
|
+
return { structuredContent, content: [] };
|
|
48
|
+
}
|
|
49
|
+
catch (error) {
|
|
50
|
+
return (0, handleError_1.handleToolError)('create', error, ctx);
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=create.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"create.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/create.ts"],"names":[],"mappings":";;AAgBA,gDAiDC;AA/DD,oCAKkB;AAClB,+CAAgD;AAEhD,SAAS,KAAK,CAAI,GAAQ,EAAE,IAAY;IACtC,MAAM,GAAG,GAAU,EAAE,CAAC;IACtB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,IAAI,IAAI;QAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC;IAC5E,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAgB,kBAAkB,CAAC,MAAiB,EAAE,GAAe;IACnE,MAAM,CAAC,YAAY,CACjB,QAAQ,EACR;QACE,WAAW,EAAE,wEAAwE;QACrF,WAAW,EAAE,yBAAiB,CAAC,KAAK;QACpC,YAAY,EAAE,0BAAkB,CAAC,KAAK;KACvC,EACD,KAAK,EAAE,GAAgB,EAAE,EAAE;QACzB,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,yBAAiB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC1C,GAAG,CAAC,UAAU,CAAC,sBAAsB,CAAC,QAAQ,CAAC,CAAC;YAChD,GAAG,CAAC,UAAU,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC9C,GAAG,CAAC,UAAU,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YAE3D,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;YAE5F,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;gBAChB,MAAM,iBAAiB,GAAiB;oBACtC,IAAI,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE;oBAC9D,MAAM,EAAE,IAAI;oBACZ,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;iBACxE,CAAC;gBACF,OAAO,EAAE,iBAAiB,EAAE,OAAO,EAAE,EAAW,EAAE,CAAC;YACrD,CAAC;YAED,MAAM,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;YACvC,MAAM,UAAU,GAAU,EAAE,CAAC;YAE7B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACvC,MAAM,IAAI,GAAG,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,KAAK,EAAE,CAAC;gBACtE,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,cAAc,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;gBAClF,MAAM,QAAQ,GAAQ,MAAM,GAAG,CAAC,QAAQ,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;gBACjG,IAAI,KAAK,CAAC,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC;oBAAE,UAAU,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,OAAO,CAAC,CAAC;YAC7E,CAAC;YAED,MAAM,iBAAiB,GAAiB;gBACtC,IAAI,EAAE,EAAE,KAAK,EAAE,UAAU,CAAC,MAAM,EAAE,OAAO,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE;gBAC5D,OAAO,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,IAAI,EAAE,EAAE,CAAC,CAAC;gBAC9E,MAAM,EAAE,KAAK;aACd,CAAC;YAEF,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,EAAE,KAAK,EAAE,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC;YAC9D,OAAO,EAAE,iBAAiB,EAAE,OAAO,EAAE,EAAW,EAAE,CAAC;QACrD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,IAAA,6BAAe,EAAC,QAAQ,EAAE,KAAK,EAAE,GAAG,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerDescribeTool = registerDescribeTool;
|
|
4
|
+
const types_1 = require("../types");
|
|
5
|
+
const errors_1 = require("../../errors");
|
|
6
|
+
const handleError_1 = require("./handleError");
|
|
7
|
+
function normalizeField(raw) {
|
|
8
|
+
const source = raw;
|
|
9
|
+
const field = {
|
|
10
|
+
id: String(source?.id ?? ''),
|
|
11
|
+
name: String(source?.name ?? ''),
|
|
12
|
+
type: String(source?.type ?? '')
|
|
13
|
+
};
|
|
14
|
+
if (source?.description && typeof source.description === 'string') {
|
|
15
|
+
field.description = source.description;
|
|
16
|
+
}
|
|
17
|
+
if (source?.options && typeof source.options === 'object') {
|
|
18
|
+
field.options = source.options;
|
|
19
|
+
}
|
|
20
|
+
return field;
|
|
21
|
+
}
|
|
22
|
+
function normalizeView(raw) {
|
|
23
|
+
const source = raw;
|
|
24
|
+
const view = {
|
|
25
|
+
id: String(source?.id ?? ''),
|
|
26
|
+
name: String(source?.name ?? '')
|
|
27
|
+
};
|
|
28
|
+
if (source?.type && typeof source.type === 'string') {
|
|
29
|
+
view.type = source.type;
|
|
30
|
+
}
|
|
31
|
+
return view;
|
|
32
|
+
}
|
|
33
|
+
function normalizeTable(raw, { includeFields, includeViews }) {
|
|
34
|
+
const source = raw;
|
|
35
|
+
const table = {
|
|
36
|
+
id: String(source?.id ?? ''),
|
|
37
|
+
name: String(source?.name ?? '')
|
|
38
|
+
};
|
|
39
|
+
if (source?.primaryFieldId && typeof source.primaryFieldId === 'string') {
|
|
40
|
+
table.primaryFieldId = source.primaryFieldId;
|
|
41
|
+
}
|
|
42
|
+
if (includeFields && Array.isArray(source?.fields)) {
|
|
43
|
+
table.fields = source.fields.map((field) => normalizeField(field));
|
|
44
|
+
}
|
|
45
|
+
if (includeViews && Array.isArray(source?.views)) {
|
|
46
|
+
table.views = source.views.map((view) => normalizeView(view));
|
|
47
|
+
}
|
|
48
|
+
return table;
|
|
49
|
+
}
|
|
50
|
+
function registerDescribeTool(server, ctx) {
|
|
51
|
+
server.registerTool('describe', {
|
|
52
|
+
description: 'Describe Airtable base or table schema.',
|
|
53
|
+
inputSchema: types_1.describeInputShape,
|
|
54
|
+
outputSchema: types_1.describeOutputSchema.shape
|
|
55
|
+
}, async (args, _extra) => {
|
|
56
|
+
try {
|
|
57
|
+
const input = types_1.describeInputSchema.parse(args);
|
|
58
|
+
ctx.governance.ensureOperationAllowed('describe');
|
|
59
|
+
ctx.governance.ensureBaseAllowed(input.baseId);
|
|
60
|
+
const includeFields = input.includeFields ?? true;
|
|
61
|
+
const includeViews = input.includeViews ?? false;
|
|
62
|
+
const logger = ctx.logger.child({
|
|
63
|
+
tool: 'describe',
|
|
64
|
+
baseId: input.baseId,
|
|
65
|
+
scope: input.scope
|
|
66
|
+
});
|
|
67
|
+
const [baseInfo, tableInfo] = await Promise.all([
|
|
68
|
+
ctx.airtable.getBase(input.baseId),
|
|
69
|
+
ctx.airtable.listTables(input.baseId)
|
|
70
|
+
]);
|
|
71
|
+
const baseName = typeof baseInfo?.name === 'string'
|
|
72
|
+
? String(baseInfo.name)
|
|
73
|
+
: input.baseId;
|
|
74
|
+
const rawTables = Array.isArray(tableInfo?.tables)
|
|
75
|
+
? tableInfo.tables
|
|
76
|
+
: [];
|
|
77
|
+
const tables = rawTables
|
|
78
|
+
.filter((rawTable) => {
|
|
79
|
+
const record = rawTable;
|
|
80
|
+
const tableId = typeof record.id === 'string' ? record.id : '';
|
|
81
|
+
const tableName = typeof record.name === 'string' ? record.name : '';
|
|
82
|
+
const idAllowed = tableId
|
|
83
|
+
? ctx.governance.isTableAllowed(input.baseId, tableId)
|
|
84
|
+
: false;
|
|
85
|
+
const nameAllowed = tableName
|
|
86
|
+
? ctx.governance.isTableAllowed(input.baseId, tableName)
|
|
87
|
+
: false;
|
|
88
|
+
return idAllowed || nameAllowed;
|
|
89
|
+
})
|
|
90
|
+
.map((table) => normalizeTable(table, { includeFields, includeViews }));
|
|
91
|
+
let selectedTables = tables;
|
|
92
|
+
if (input.scope === 'table') {
|
|
93
|
+
const target = tables.find((tableRecord) => String(tableRecord.id) === input.table ||
|
|
94
|
+
String(tableRecord.name).toLowerCase() === input.table?.toLowerCase());
|
|
95
|
+
if (!target) {
|
|
96
|
+
const context = { baseId: input.baseId };
|
|
97
|
+
if (input.table) {
|
|
98
|
+
context.table = input.table;
|
|
99
|
+
}
|
|
100
|
+
throw new errors_1.NotFoundError(`Table ${input.table} not found in base ${input.baseId}`, {
|
|
101
|
+
context
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
const targetId = String(target.id);
|
|
105
|
+
const targetName = String(target.name);
|
|
106
|
+
if (!ctx.governance.isTableAllowed(input.baseId, targetId) &&
|
|
107
|
+
!ctx.governance.isTableAllowed(input.baseId, targetName)) {
|
|
108
|
+
const context = { baseId: input.baseId };
|
|
109
|
+
if (input.table) {
|
|
110
|
+
context.table = input.table;
|
|
111
|
+
}
|
|
112
|
+
throw new errors_1.GovernanceError(`Table ${input.table} is not allowed in base ${input.baseId}`, {
|
|
113
|
+
context
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
selectedTables = [target];
|
|
117
|
+
}
|
|
118
|
+
const structuredContent = {
|
|
119
|
+
base: {
|
|
120
|
+
id: input.baseId,
|
|
121
|
+
name: baseName
|
|
122
|
+
},
|
|
123
|
+
tables: selectedTables
|
|
124
|
+
};
|
|
125
|
+
if (input.scope === 'base' && includeViews) {
|
|
126
|
+
structuredContent.views = rawTables
|
|
127
|
+
.flatMap((table) => {
|
|
128
|
+
const record = table;
|
|
129
|
+
return Array.isArray(record.views) ? record.views : [];
|
|
130
|
+
})
|
|
131
|
+
.map((view) => normalizeView(view));
|
|
132
|
+
}
|
|
133
|
+
logger.debug('Describe completed', {
|
|
134
|
+
tableCount: selectedTables.length
|
|
135
|
+
});
|
|
136
|
+
return {
|
|
137
|
+
structuredContent,
|
|
138
|
+
content: []
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
catch (error) {
|
|
142
|
+
return (0, handleError_1.handleToolError)('describe', error, ctx);
|
|
143
|
+
}
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
//# sourceMappingURL=describe.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"describe.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/describe.ts"],"names":[],"mappings":"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"}
|