@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,54 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.handleToolError = handleToolError;
|
|
4
|
+
const errors_1 = require("../../errors");
|
|
5
|
+
function toUserMessage(error) {
|
|
6
|
+
switch (error.code) {
|
|
7
|
+
case 'RateLimited':
|
|
8
|
+
return 'Airtable rate limit exceeded. Please retry after backoff.';
|
|
9
|
+
case 'ValidationError':
|
|
10
|
+
return 'Airtable rejected the request. Check field names and values.';
|
|
11
|
+
case 'AuthError':
|
|
12
|
+
return 'Authentication failed. Verify the Airtable token scopes and base access.';
|
|
13
|
+
case 'ConflictError':
|
|
14
|
+
return 'The record changed since it was fetched. Refresh data and review the diff.';
|
|
15
|
+
case 'NotFound':
|
|
16
|
+
return 'Requested Airtable resource was not found. Confirm the base and table identifiers.';
|
|
17
|
+
case 'GovernanceError':
|
|
18
|
+
return 'Operation blocked by governance allow-lists.';
|
|
19
|
+
default:
|
|
20
|
+
return 'Unexpected Airtable error. Please retry or check the exceptions queue.';
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
function handleToolError(toolName, error, ctx) {
|
|
24
|
+
if (error instanceof errors_1.AirtableBrainError) {
|
|
25
|
+
ctx.logger.error(`${toolName} failed`, {
|
|
26
|
+
code: error.code,
|
|
27
|
+
status: error.status,
|
|
28
|
+
retryAfterMs: error.retryAfterMs
|
|
29
|
+
});
|
|
30
|
+
ctx.exceptions.record(error, `${toolName} failed`, error.message);
|
|
31
|
+
return {
|
|
32
|
+
isError: true,
|
|
33
|
+
content: [
|
|
34
|
+
{
|
|
35
|
+
type: 'text',
|
|
36
|
+
text: toUserMessage(error)
|
|
37
|
+
}
|
|
38
|
+
]
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
ctx.logger.error(`${toolName} failed with unknown error`, {
|
|
42
|
+
error: error instanceof Error ? error.message : String(error)
|
|
43
|
+
});
|
|
44
|
+
return {
|
|
45
|
+
isError: true,
|
|
46
|
+
content: [
|
|
47
|
+
{
|
|
48
|
+
type: 'text',
|
|
49
|
+
text: 'Unexpected server error. Check logs for details.'
|
|
50
|
+
}
|
|
51
|
+
]
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=handleError.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"handleError.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/handleError.ts"],"names":[],"mappings":";;AAsBA,0CAgCC;AAtDD,yCAAkD;AAGlD,SAAS,aAAa,CAAC,KAAyB;IAC9C,QAAQ,KAAK,CAAC,IAAI,EAAE,CAAC;QACnB,KAAK,aAAa;YAChB,OAAO,2DAA2D,CAAC;QACrE,KAAK,iBAAiB;YACpB,OAAO,8DAA8D,CAAC;QACxE,KAAK,WAAW;YACd,OAAO,0EAA0E,CAAC;QACpF,KAAK,eAAe;YAClB,OAAO,4EAA4E,CAAC;QACtF,KAAK,UAAU;YACb,OAAO,oFAAoF,CAAC;QAC9F,KAAK,iBAAiB;YACpB,OAAO,8CAA8C,CAAC;QACxD;YACE,OAAO,wEAAwE,CAAC;IACpF,CAAC;AACH,CAAC;AAED,SAAgB,eAAe,CAAC,QAAgB,EAAE,KAAc,EAAE,GAAe;IAC/E,IAAI,KAAK,YAAY,2BAAkB,EAAE,CAAC;QACxC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,QAAQ,SAAS,EAAE;YACrC,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,YAAY,EAAE,KAAK,CAAC,YAAY;SACjC,CAAC,CAAC;QACH,GAAG,CAAC,UAAU,CAAC,MAAM,CAAC,KAAK,EAAE,GAAG,QAAQ,SAAS,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;QAClE,OAAO;YACL,OAAO,EAAE,IAAI;YACb,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAe;oBACrB,IAAI,EAAE,aAAa,CAAC,KAAK,CAAC;iBAC3B;aACF;SACF,CAAC;IACJ,CAAC;IAED,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,QAAQ,4BAA4B,EAAE;QACxD,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;KAC9D,CAAC,CAAC;IAEH,OAAO;QACL,OAAO,EAAE,IAAI;QACb,OAAO,EAAE;YACP;gBACE,IAAI,EAAE,MAAe;gBACrB,IAAI,EAAE,kDAAkD;aACzD;SACF;KACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerAllTools = registerAllTools;
|
|
4
|
+
const listBases_1 = require("./listBases");
|
|
5
|
+
const describe_1 = require("./describe");
|
|
6
|
+
const query_1 = require("./query");
|
|
7
|
+
const listGovernance_1 = require("./listGovernance");
|
|
8
|
+
const listExceptions_1 = require("./listExceptions");
|
|
9
|
+
const create_1 = require("./create");
|
|
10
|
+
const update_1 = require("./update");
|
|
11
|
+
const upsert_1 = require("./upsert");
|
|
12
|
+
const webhooks_1 = require("./webhooks");
|
|
13
|
+
function registerAllTools(server, ctx) {
|
|
14
|
+
(0, listBases_1.registerListBasesTool)(server, ctx);
|
|
15
|
+
(0, describe_1.registerDescribeTool)(server, ctx);
|
|
16
|
+
(0, query_1.registerQueryTool)(server, ctx);
|
|
17
|
+
(0, listGovernance_1.registerGovernanceTool)(server, ctx);
|
|
18
|
+
(0, listExceptions_1.registerExceptionsTool)(server, ctx);
|
|
19
|
+
(0, create_1.registerCreateTool)(server, ctx);
|
|
20
|
+
(0, update_1.registerUpdateTool)(server, ctx);
|
|
21
|
+
(0, upsert_1.registerUpsertTool)(server, ctx);
|
|
22
|
+
(0, webhooks_1.registerWebhookTools)(server, ctx);
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/index.ts"],"names":[],"mappings":";;AAYA,4CAUC;AApBD,2CAAoD;AACpD,yCAAkD;AAClD,mCAA4C;AAC5C,qDAA0D;AAC1D,qDAA0D;AAC1D,qCAA8C;AAC9C,qCAA8C;AAC9C,qCAA8C;AAC9C,yCAAkD;AAElD,SAAgB,gBAAgB,CAAC,MAAiB,EAAE,GAAe;IACjE,IAAA,iCAAqB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IACnC,IAAA,+BAAoB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAClC,IAAA,yBAAiB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC/B,IAAA,uCAAsB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IACpC,IAAA,uCAAsB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IACpC,IAAA,2BAAkB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAChC,IAAA,2BAAkB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAChC,IAAA,2BAAkB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAChC,IAAA,+BAAoB,EAAC,MAAM,EAAE,GAAG,CAAC,CAAC;AACpC,CAAC"}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerListBasesTool = registerListBasesTool;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const handleError_1 = require("./handleError");
|
|
6
|
+
// Schema for list_bases output
|
|
7
|
+
const listBasesOutputSchema = zod_1.z.object({
|
|
8
|
+
bases: zod_1.z.array(zod_1.z.object({
|
|
9
|
+
id: zod_1.z.string(),
|
|
10
|
+
name: zod_1.z.string(),
|
|
11
|
+
permissionLevel: zod_1.z.string().optional()
|
|
12
|
+
}))
|
|
13
|
+
});
|
|
14
|
+
function registerListBasesTool(server, ctx) {
|
|
15
|
+
server.registerTool('list_bases', {
|
|
16
|
+
description: 'List all accessible Airtable bases with their names, IDs, and permission levels',
|
|
17
|
+
inputSchema: {},
|
|
18
|
+
outputSchema: listBasesOutputSchema.shape
|
|
19
|
+
}, async (_args, _extra) => {
|
|
20
|
+
try {
|
|
21
|
+
ctx.logger.info('Listing accessible Airtable bases');
|
|
22
|
+
const response = await ctx.airtable.listBases();
|
|
23
|
+
const bases = response.bases;
|
|
24
|
+
if (!bases || bases.length === 0) {
|
|
25
|
+
const structuredContent = {
|
|
26
|
+
bases: []
|
|
27
|
+
};
|
|
28
|
+
return {
|
|
29
|
+
structuredContent,
|
|
30
|
+
content: []
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
const normalizedBases = bases.map((base) => ({
|
|
34
|
+
id: String(base.id ?? ''),
|
|
35
|
+
name: String(base.name ?? ''),
|
|
36
|
+
permissionLevel: base.permissionLevel ? String(base.permissionLevel) : undefined
|
|
37
|
+
}));
|
|
38
|
+
const structuredContent = {
|
|
39
|
+
bases: normalizedBases
|
|
40
|
+
};
|
|
41
|
+
ctx.logger.info('Successfully listed bases', { count: bases.length });
|
|
42
|
+
return {
|
|
43
|
+
structuredContent,
|
|
44
|
+
content: []
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
catch (error) {
|
|
48
|
+
return (0, handleError_1.handleToolError)('list_bases', error, ctx);
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
//# sourceMappingURL=listBases.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"listBases.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/listBases.ts"],"names":[],"mappings":";;AAkBA,sDA8CC;AA9DD,6BAAwB;AACxB,+CAAgD;AAEhD,+BAA+B;AAC/B,MAAM,qBAAqB,GAAG,OAAC,CAAC,MAAM,CAAC;IACrC,KAAK,EAAE,OAAC,CAAC,KAAK,CACZ,OAAC,CAAC,MAAM,CAAC;QACP,EAAE,EAAE,OAAC,CAAC,MAAM,EAAE;QACd,IAAI,EAAE,OAAC,CAAC,MAAM,EAAE;QAChB,eAAe,EAAE,OAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE;KACvC,CAAC,CACH;CACF,CAAC,CAAC;AAIH,SAAgB,qBAAqB,CAAC,MAAiB,EAAE,GAAe;IACtE,MAAM,CAAC,YAAY,CACjB,YAAY,EACZ;QACE,WAAW,EAAE,iFAAiF;QAC9F,WAAW,EAAE,EAAE;QACf,YAAY,EAAE,qBAAqB,CAAC,KAAK;KAC1C,EACD,KAAK,EAAE,KAAc,EAAE,MAAe,EAAE,EAAE;QACxC,IAAI,CAAC;YACH,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;YAErD,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC;YAChD,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC;YAE7B,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACjC,MAAM,iBAAiB,GAAoB;oBACzC,KAAK,EAAE,EAAE;iBACV,CAAC;gBACF,OAAO;oBACL,iBAAiB;oBACjB,OAAO,EAAE,EAAW;iBACrB,CAAC;YACJ,CAAC;YAED,MAAM,eAAe,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,CAAC;gBAChD,EAAE,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC;gBACzB,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC;gBAC7B,eAAe,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,SAAS;aACjF,CAAC,CAAC,CAAC;YAEJ,MAAM,iBAAiB,GAAoB;gBACzC,KAAK,EAAE,eAAe;aACvB,CAAC;YAEF,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,2BAA2B,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC;YAEtE,OAAO;gBACL,iBAAiB;gBACjB,OAAO,EAAE,EAAW;aACrB,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,IAAA,6BAAe,EAAC,YAAY,EAAE,KAAK,EAAE,GAAG,CAAC,CAAC;QACnD,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerExceptionsTool = registerExceptionsTool;
|
|
4
|
+
const types_1 = require("../types");
|
|
5
|
+
function registerExceptionsTool(server, ctx) {
|
|
6
|
+
server.registerTool('list_exceptions', {
|
|
7
|
+
description: 'List recent exceptions and remediation proposals.',
|
|
8
|
+
inputSchema: types_1.listExceptionsInputSchema.shape,
|
|
9
|
+
outputSchema: types_1.listExceptionsOutputSchema.shape
|
|
10
|
+
}, async (args) => {
|
|
11
|
+
const snapshot = ctx.exceptions.list(args);
|
|
12
|
+
return {
|
|
13
|
+
structuredContent: snapshot,
|
|
14
|
+
content: []
|
|
15
|
+
};
|
|
16
|
+
});
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=listExceptions.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"listExceptions.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/listExceptions.ts"],"names":[],"mappings":";;AAQA,wDAgBC;AAvBD,oCAIkB;AAGlB,SAAgB,sBAAsB,CAAC,MAAiB,EAAE,GAAe;IACvE,MAAM,CAAC,YAAY,CACjB,iBAAiB,EACjB;QACE,WAAW,EAAE,mDAAmD;QAChE,WAAW,EAAE,iCAAyB,CAAC,KAAK;QAC5C,YAAY,EAAE,kCAA0B,CAAC,KAAK;KAC/C,EACD,KAAK,EAAE,IAAyB,EAAE,EAAE;QAClC,MAAM,QAAQ,GAAG,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3C,OAAO;YACL,iBAAiB,EAAE,QAAQ;YAC3B,OAAO,EAAE,EAAW;SACrB,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerGovernanceTool = registerGovernanceTool;
|
|
4
|
+
const types_1 = require("../types");
|
|
5
|
+
function registerGovernanceTool(server, ctx) {
|
|
6
|
+
server.registerTool('list_governance', {
|
|
7
|
+
description: 'Return governance allow-lists and PII masking policies.',
|
|
8
|
+
outputSchema: types_1.governanceOutputSchema.shape
|
|
9
|
+
}, async () => {
|
|
10
|
+
const snapshot = ctx.governance.getSnapshot();
|
|
11
|
+
return {
|
|
12
|
+
structuredContent: snapshot,
|
|
13
|
+
content: []
|
|
14
|
+
};
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=listGovernance.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"listGovernance.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/listGovernance.ts"],"names":[],"mappings":";;AAIA,wDAeC;AAlBD,oCAAkD;AAGlD,SAAgB,sBAAsB,CAAC,MAAiB,EAAE,GAAe;IACvE,MAAM,CAAC,YAAY,CACjB,iBAAiB,EACjB;QACE,WAAW,EAAE,yDAAyD;QACtE,YAAY,EAAE,8BAAsB,CAAC,KAAK;KAC3C,EACD,KAAK,IAAI,EAAE;QACT,MAAM,QAAQ,GAAG,GAAG,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;QAC9C,OAAO;YACL,iBAAiB,EAAE,QAAQ;YAC3B,OAAO,EAAE,EAAW;SACrB,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerQueryTool = registerQueryTool;
|
|
4
|
+
const node_crypto_1 = require("node:crypto");
|
|
5
|
+
const types_1 = require("../types");
|
|
6
|
+
const handleError_1 = require("./handleError");
|
|
7
|
+
function maskValue(value) {
|
|
8
|
+
if (value === null || value === undefined) {
|
|
9
|
+
return value;
|
|
10
|
+
}
|
|
11
|
+
if (Array.isArray(value)) {
|
|
12
|
+
return value.map(() => '••••');
|
|
13
|
+
}
|
|
14
|
+
if (typeof value === 'object') {
|
|
15
|
+
return '[redacted]';
|
|
16
|
+
}
|
|
17
|
+
return '••••';
|
|
18
|
+
}
|
|
19
|
+
function hashValue(value) {
|
|
20
|
+
const serialized = typeof value === 'string' ? value : JSON.stringify(value ?? '');
|
|
21
|
+
return (0, node_crypto_1.createHash)('sha256').update(serialized).digest('hex');
|
|
22
|
+
}
|
|
23
|
+
function applyPiiPolicies(fields, policies) {
|
|
24
|
+
if (!policies.length) {
|
|
25
|
+
return fields;
|
|
26
|
+
}
|
|
27
|
+
const result = { ...fields };
|
|
28
|
+
for (const policy of policies) {
|
|
29
|
+
if (!(policy.field in result))
|
|
30
|
+
continue;
|
|
31
|
+
switch (policy.policy) {
|
|
32
|
+
case 'drop':
|
|
33
|
+
delete result[policy.field];
|
|
34
|
+
break;
|
|
35
|
+
case 'mask':
|
|
36
|
+
result[policy.field] = maskValue(result[policy.field]);
|
|
37
|
+
break;
|
|
38
|
+
case 'hash':
|
|
39
|
+
result[policy.field] = hashValue(result[policy.field]);
|
|
40
|
+
break;
|
|
41
|
+
default:
|
|
42
|
+
break;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return result;
|
|
46
|
+
}
|
|
47
|
+
function registerQueryTool(server, ctx) {
|
|
48
|
+
server.registerTool('query', {
|
|
49
|
+
description: 'Query Airtable records with filtering, sorting, and pagination.',
|
|
50
|
+
inputSchema: types_1.queryInputShape,
|
|
51
|
+
outputSchema: types_1.queryOutputSchema.shape
|
|
52
|
+
}, async (args, _extra) => {
|
|
53
|
+
try {
|
|
54
|
+
const input = types_1.queryInputSchema.parse(args);
|
|
55
|
+
ctx.governance.ensureOperationAllowed('query');
|
|
56
|
+
ctx.governance.ensureBaseAllowed(input.baseId);
|
|
57
|
+
ctx.governance.ensureTableAllowed(input.baseId, input.table);
|
|
58
|
+
const logger = ctx.logger.child({
|
|
59
|
+
tool: 'query',
|
|
60
|
+
baseId: input.baseId,
|
|
61
|
+
table: input.table
|
|
62
|
+
});
|
|
63
|
+
const queryParams = {};
|
|
64
|
+
if (input.fields) {
|
|
65
|
+
queryParams.fields = input.fields;
|
|
66
|
+
}
|
|
67
|
+
if (input.filterByFormula) {
|
|
68
|
+
queryParams.filterByFormula = input.filterByFormula;
|
|
69
|
+
}
|
|
70
|
+
if (input.view) {
|
|
71
|
+
queryParams.view = input.view;
|
|
72
|
+
}
|
|
73
|
+
if (input.pageSize) {
|
|
74
|
+
queryParams.pageSize = input.pageSize;
|
|
75
|
+
}
|
|
76
|
+
if (input.maxRecords) {
|
|
77
|
+
queryParams.maxRecords = input.maxRecords;
|
|
78
|
+
}
|
|
79
|
+
if (input.offset) {
|
|
80
|
+
queryParams.offset = input.offset;
|
|
81
|
+
}
|
|
82
|
+
if (typeof input.returnFieldsByFieldId === 'boolean') {
|
|
83
|
+
queryParams.returnFieldsByFieldId = input.returnFieldsByFieldId;
|
|
84
|
+
}
|
|
85
|
+
if (input.sorts) {
|
|
86
|
+
input.sorts.forEach((sort, index) => {
|
|
87
|
+
queryParams[`sort[${index}][field]`] = sort.field;
|
|
88
|
+
queryParams[`sort[${index}][direction]`] = sort.direction ?? 'asc';
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
const response = await ctx.airtable.queryRecords(input.baseId, input.table, queryParams);
|
|
92
|
+
const rawRecords = Array.isArray(response?.records)
|
|
93
|
+
? response.records
|
|
94
|
+
: [];
|
|
95
|
+
const piiPolicies = ctx.governance.listPiiPolicies(input.baseId, input.table);
|
|
96
|
+
const sanitizedRecords = rawRecords.map((record) => {
|
|
97
|
+
const fields = typeof record.fields === 'object' && record.fields !== null ? record.fields : {};
|
|
98
|
+
return {
|
|
99
|
+
id: String(record.id ?? ''),
|
|
100
|
+
createdTime: record.createdTime ? String(record.createdTime) : undefined,
|
|
101
|
+
fields: applyPiiPolicies(fields, piiPolicies)
|
|
102
|
+
};
|
|
103
|
+
});
|
|
104
|
+
const structuredContent = {
|
|
105
|
+
records: sanitizedRecords,
|
|
106
|
+
offset: typeof response?.offset === 'string' ? response.offset : undefined,
|
|
107
|
+
summary: {
|
|
108
|
+
returned: sanitizedRecords.length,
|
|
109
|
+
hasMore: Boolean(response?.offset)
|
|
110
|
+
}
|
|
111
|
+
};
|
|
112
|
+
logger.debug('Query completed', {
|
|
113
|
+
returned: sanitizedRecords.length,
|
|
114
|
+
hasMore: structuredContent.summary?.hasMore
|
|
115
|
+
});
|
|
116
|
+
return {
|
|
117
|
+
structuredContent,
|
|
118
|
+
content: []
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
catch (error) {
|
|
122
|
+
return (0, handleError_1.handleToolError)('query', error, ctx);
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
}
|
|
126
|
+
//# sourceMappingURL=query.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"query.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/query.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerUpdateTool = registerUpdateTool;
|
|
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 registerUpdateTool(server, ctx) {
|
|
13
|
+
server.registerTool('update', {
|
|
14
|
+
description: 'Update Airtable records (requires diff-before-write via dryRun first).',
|
|
15
|
+
inputSchema: types_1.updateInputSchema.shape,
|
|
16
|
+
outputSchema: types_1.updateOutputSchema.shape
|
|
17
|
+
}, async (raw) => {
|
|
18
|
+
try {
|
|
19
|
+
const args = types_1.updateInputSchema.parse(raw);
|
|
20
|
+
ctx.governance.ensureOperationAllowed('update');
|
|
21
|
+
ctx.governance.ensureBaseAllowed(args.baseId);
|
|
22
|
+
ctx.governance.ensureTableAllowed(args.baseId, args.table);
|
|
23
|
+
const logger = ctx.logger.child({ tool: 'update', baseId: args.baseId, table: args.table });
|
|
24
|
+
if (args.dryRun) {
|
|
25
|
+
const structuredContent = {
|
|
26
|
+
diff: { added: 0, updated: args.records.length, unchanged: 0, conflicts: 0 },
|
|
27
|
+
dryRun: true,
|
|
28
|
+
records: args.records.map((r) => ({ id: r.id, fields: r.fields })),
|
|
29
|
+
conflicts: []
|
|
30
|
+
};
|
|
31
|
+
return { structuredContent, content: [] };
|
|
32
|
+
}
|
|
33
|
+
const chunks = chunk(args.records, 10);
|
|
34
|
+
const aggregated = [];
|
|
35
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
36
|
+
const body = { records: chunks[i], typecast: args.typecast ?? false };
|
|
37
|
+
const headerKey = args.idempotencyKey ? `${args.idempotencyKey}:${i}` : undefined;
|
|
38
|
+
const response = await ctx.airtable.updateRecords(args.baseId, args.table, body, headerKey);
|
|
39
|
+
if (Array.isArray(response?.records))
|
|
40
|
+
aggregated.push(...response.records);
|
|
41
|
+
}
|
|
42
|
+
const structuredContent = {
|
|
43
|
+
diff: { added: 0, updated: aggregated.length, unchanged: 0, conflicts: 0 },
|
|
44
|
+
records: aggregated.map((r) => ({ id: String(r.id), fields: r.fields || {} })),
|
|
45
|
+
dryRun: false,
|
|
46
|
+
conflicts: []
|
|
47
|
+
};
|
|
48
|
+
logger.info('Update completed', { updated: aggregated.length });
|
|
49
|
+
return { structuredContent, content: [] };
|
|
50
|
+
}
|
|
51
|
+
catch (error) {
|
|
52
|
+
return (0, handleError_1.handleToolError)('update', error, ctx);
|
|
53
|
+
}
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
//# sourceMappingURL=update.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"update.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/update.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerUpsertTool = registerUpsertTool;
|
|
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 registerUpsertTool(server, ctx) {
|
|
13
|
+
server.registerTool('upsert', {
|
|
14
|
+
description: 'Upsert Airtable records using performUpsert.fieldsToMergeOn.',
|
|
15
|
+
inputSchema: types_1.upsertInputSchema.shape,
|
|
16
|
+
outputSchema: types_1.upsertOutputSchema.shape
|
|
17
|
+
}, async (raw) => {
|
|
18
|
+
try {
|
|
19
|
+
const args = types_1.upsertInputSchema.parse(raw);
|
|
20
|
+
ctx.governance.ensureOperationAllowed('upsert');
|
|
21
|
+
ctx.governance.ensureBaseAllowed(args.baseId);
|
|
22
|
+
ctx.governance.ensureTableAllowed(args.baseId, args.table);
|
|
23
|
+
const logger = ctx.logger.child({ tool: 'upsert', baseId: args.baseId, table: args.table });
|
|
24
|
+
const matchedBy = args.performUpsert.fieldsToMergeOn;
|
|
25
|
+
if (args.dryRun) {
|
|
26
|
+
const structuredContent = {
|
|
27
|
+
diff: { added: args.records.length, updated: 0, unchanged: 0, conflicts: 0 },
|
|
28
|
+
dryRun: true,
|
|
29
|
+
records: args.records.map((r) => ({ id: 'pending', fields: r.fields })),
|
|
30
|
+
conflicts: []
|
|
31
|
+
};
|
|
32
|
+
// Note: Upsert output in PRD expects 'matchedBy' array and no conflicts property; keep consistent with docs
|
|
33
|
+
// When using strict PRD output, we can omit conflicts and include matchedBy
|
|
34
|
+
structuredContent.matchedBy = matchedBy;
|
|
35
|
+
delete structuredContent.conflicts;
|
|
36
|
+
return { structuredContent, content: [] };
|
|
37
|
+
}
|
|
38
|
+
const chunks = chunk(args.records, 10);
|
|
39
|
+
const aggregated = [];
|
|
40
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
41
|
+
const body = {
|
|
42
|
+
records: chunks[i],
|
|
43
|
+
typecast: args.typecast ?? false,
|
|
44
|
+
performUpsert: { fieldsToMergeOn: matchedBy }
|
|
45
|
+
};
|
|
46
|
+
const headerKey = args.idempotencyKey ? `${args.idempotencyKey}:${i}` : undefined;
|
|
47
|
+
const response = await ctx.airtable.upsertRecords(args.baseId, args.table, body, headerKey);
|
|
48
|
+
if (Array.isArray(response?.records))
|
|
49
|
+
aggregated.push(...response.records);
|
|
50
|
+
}
|
|
51
|
+
const structuredContent = {
|
|
52
|
+
diff: { added: 0, updated: aggregated.length, unchanged: 0 },
|
|
53
|
+
matchedBy,
|
|
54
|
+
records: aggregated.map((r) => ({ id: String(r.id), fields: r.fields || {} })),
|
|
55
|
+
dryRun: false
|
|
56
|
+
};
|
|
57
|
+
logger.info('Upsert completed', { processed: aggregated.length, matchedBy });
|
|
58
|
+
return { structuredContent, content: [] };
|
|
59
|
+
}
|
|
60
|
+
catch (error) {
|
|
61
|
+
return (0, handleError_1.handleToolError)('upsert', error, ctx);
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=upsert.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"upsert.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/upsert.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerWebhookTools = registerWebhookTools;
|
|
4
|
+
const handleError_1 = require("./handleError");
|
|
5
|
+
function registerWebhookTools(server, ctx) {
|
|
6
|
+
server.registerTool('list_webhooks', { description: 'List Airtable webhooks for the default base.' }, async (_args) => {
|
|
7
|
+
try {
|
|
8
|
+
const baseId = ctx.config.auth.defaultBaseId || ctx.config.auth.allowedBases[0];
|
|
9
|
+
if (!baseId)
|
|
10
|
+
throw new Error('No base configured');
|
|
11
|
+
const body = await ctx.airtable.queryRecords(baseId, 'meta/webhooks');
|
|
12
|
+
return { structuredContent: { webhooks: body }, content: [] };
|
|
13
|
+
}
|
|
14
|
+
catch (error) {
|
|
15
|
+
return (0, handleError_1.handleToolError)('list_webhooks', error, ctx);
|
|
16
|
+
}
|
|
17
|
+
});
|
|
18
|
+
server.registerTool('create_webhook', { description: 'Create a new webhook for a base.' }, async (args) => {
|
|
19
|
+
try {
|
|
20
|
+
const baseId = args.baseId || ctx.config.auth.defaultBaseId || ctx.config.auth.allowedBases[0];
|
|
21
|
+
if (!baseId)
|
|
22
|
+
throw new Error('No base configured');
|
|
23
|
+
const payload = { notificationUrl: String(args.notificationUrl || '') };
|
|
24
|
+
const result = await ctx.airtable.createRecords(baseId, 'meta/webhooks', payload);
|
|
25
|
+
return { structuredContent: { webhook: result }, content: [] };
|
|
26
|
+
}
|
|
27
|
+
catch (error) {
|
|
28
|
+
return (0, handleError_1.handleToolError)('create_webhook', error, ctx);
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
server.registerTool('refresh_webhook', { description: 'Refresh webhook expiration.' }, async (args) => {
|
|
32
|
+
try {
|
|
33
|
+
const baseId = args.baseId || ctx.config.auth.defaultBaseId || ctx.config.auth.allowedBases[0];
|
|
34
|
+
if (!baseId)
|
|
35
|
+
throw new Error('No base configured');
|
|
36
|
+
const result = await ctx.airtable.updateRecords(baseId, `meta/webhooks/${String(args.webhookId)}/refresh`, {});
|
|
37
|
+
return { structuredContent: { webhook: result }, content: [] };
|
|
38
|
+
}
|
|
39
|
+
catch (error) {
|
|
40
|
+
return (0, handleError_1.handleToolError)('refresh_webhook', error, ctx);
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=webhooks.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhooks.js","sourceRoot":"","sources":["../../../../src/typescript/app/tools/webhooks.ts"],"names":[],"mappings":"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"}
|