-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathinspector_server.py
399 lines (325 loc) · 14.2 KB
/
inspector_server.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
#!/usr/bin/env python3
"""
Airtable MCP Inspector Server
-----------------------------
A simple MCP server that implements the Airtable tools
"""
import os
import sys
import json
import logging
import requests
import argparse
import traceback
from typing import Optional, Dict, Any, List
try:
from mcp.server.fastmcp import FastMCP
except ImportError:
print("Error: MCP SDK not found. Please install with 'pip install mcp'")
sys.exit(1)
# Parse command line arguments
def parse_args():
parser = argparse.ArgumentParser(description="Airtable MCP Server")
parser.add_argument("--token", dest="api_token", help="Airtable Personal Access Token")
parser.add_argument("--base", dest="base_id", help="Airtable Base ID")
parser.add_argument("--config", dest="config_json", help="Configuration as JSON (for Smithery integration)")
return parser.parse_args()
# Set up logging
logging.basicConfig(level=logging.INFO)
logger = logging.getLogger("airtable-mcp")
# Parse arguments
args = parse_args()
# Handle config JSON from Smithery if provided
config = {}
if args.config_json:
try:
# Strip any trailing quotes or backslashes that might be present
config_str = args.config_json.rstrip('\\"')
# Additional sanitization for JSON format
config_str = config_str.strip()
# Handle escaped quotes
if config_str.startswith('"') and config_str.endswith('"'):
config_str = config_str[1:-1]
# Fix escaped quotes within JSON
config_str = config_str.replace('\\"', '"')
# Replace escaped backslashes
config_str = config_str.replace('\\\\', '\\')
logger.info(f"Parsing sanitized config: {config_str}")
config = json.loads(config_str)
logger.info(f"Successfully parsed config: {config}")
except json.JSONDecodeError as e:
logger.error(f"Failed to parse config JSON: {e}")
logger.error(f"Raw config string: {args.config_json}")
# Try one more approach - sometimes config is double-quoted JSON
try:
# Try to interpret as Python string literal
import ast
literal_str = ast.literal_eval(f"'''{args.config_json}'''")
config = json.loads(literal_str)
logger.info(f"Successfully parsed config using ast: {config}")
except Exception as ast_error:
logger.error(f"Failed alternate parsing method: {ast_error}")
# Create MCP server
app = FastMCP("Airtable Tools")
# Add error handling wrapper for all MCP methods
def handle_exceptions(func):
"""Decorator to properly handle and format exceptions in MCP functions"""
async def wrapper(*args, **kwargs):
try:
return await func(*args, **kwargs)
except Exception as e:
error_trace = traceback.format_exc()
logger.error(f"Error in MCP handler: {str(e)}\n{error_trace}")
sys.stderr.write(f"Error in MCP handler: {str(e)}\n{error_trace}\n")
# For tool functions that return strings, return a formatted error message
if hasattr(func, "__annotations__") and func.__annotations__.get("return") == str:
return f"Error: {str(e)}"
# For RPC methods that return dicts, return a properly formatted JSON error
return {"error": {"code": -32000, "message": str(e)}}
return wrapper
# Patch the tool method to automatically apply error handling
original_tool = app.tool
def patched_tool(*args, **kwargs):
def decorator(func):
wrapped_func = handle_exceptions(func)
return original_tool(*args, **kwargs)(wrapped_func)
return decorator
# Replace app.tool with our patched version
app.tool = patched_tool
# Also patch rpc_method
original_rpc_method = app.rpc_method
def patched_rpc_method(*args, **kwargs):
def decorator(func):
wrapped_func = handle_exceptions(func)
return original_rpc_method(*args, **kwargs)(wrapped_func)
return decorator
# Replace app.rpc_method with our patched version
app.rpc_method = patched_rpc_method
# Get token from arguments, config, or environment
token = args.api_token or config.get("airtable_token", "") or os.environ.get("AIRTABLE_PERSONAL_ACCESS_TOKEN", "")
# Clean up token if it has trailing quote
if token and token.endswith('"'):
token = token[:-1]
base_id = args.base_id or config.get("base_id", "") or os.environ.get("AIRTABLE_BASE_ID", "")
if not token:
logger.warning("No Airtable API token provided. Use --token, --config, or set AIRTABLE_PERSONAL_ACCESS_TOKEN environment variable.")
else:
logger.info(f"Using Airtable token: {token[:5]}...{token[-5:]}")
if base_id:
logger.info(f"Using base ID: {base_id}")
else:
logger.warning("No base ID provided. Use --base, --config, or set AIRTABLE_BASE_ID environment variable.")
# Helper functions for Airtable API calls
async def api_call(endpoint, method="GET", data=None, params=None):
"""Make an Airtable API call"""
if not token:
return {"error": "No Airtable API token provided. Use --token, --config, or set AIRTABLE_PERSONAL_ACCESS_TOKEN environment variable."}
headers = {
"Authorization": f"Bearer {token}",
"Content-Type": "application/json"
}
url = f"https://api.airtable.com/v0/{endpoint}"
try:
if method == "GET":
response = requests.get(url, headers=headers, params=params)
elif method == "POST":
response = requests.post(url, headers=headers, json=data)
elif method == "PATCH":
response = requests.patch(url, headers=headers, json=data)
elif method == "DELETE":
response = requests.delete(url, headers=headers, params=params)
else:
raise ValueError(f"Unsupported method: {method}")
response.raise_for_status()
return response.json()
except Exception as e:
logger.error(f"API call error: {str(e)}")
return {"error": str(e)}
# Define MCP tool functions
@app.tool()
async def list_bases() -> str:
"""List all accessible Airtable bases"""
if not token:
return "Please provide an Airtable API token to list your bases."
result = await api_call("meta/bases")
if "error" in result:
return f"Error: {result['error']}"
bases = result.get("bases", [])
if not bases:
return "No bases found accessible with your token."
base_list = [f"{i+1}. {base['name']} (ID: {base['id']})" for i, base in enumerate(bases)]
return "Available bases:\n" + "\n".join(base_list)
@app.tool()
async def list_tables(base_id_param: Optional[str] = None) -> str:
"""List all tables in the specified base or the default base"""
global base_id
current_base = base_id_param or base_id
if not token:
return "Please provide an Airtable API token to list tables."
if not current_base:
return "Error: No base ID provided. Please specify a base_id or set AIRTABLE_BASE_ID environment variable."
result = await api_call(f"meta/bases/{current_base}/tables")
if "error" in result:
return f"Error: {result['error']}"
tables = result.get("tables", [])
if not tables:
return "No tables found in this base."
table_list = [f"{i+1}. {table['name']} (ID: {table['id']}, Fields: {len(table.get('fields', []))})"
for i, table in enumerate(tables)]
return "Tables in this base:\n" + "\n".join(table_list)
@app.tool()
async def list_records(table_name: str, max_records: Optional[int] = 100, filter_formula: Optional[str] = None) -> str:
"""List records from a table with optional filtering"""
if not token:
return "Please provide an Airtable API token to list records."
if not base_id:
return "Error: No base ID set. Please use --base or set AIRTABLE_BASE_ID environment variable."
params = {"maxRecords": max_records}
if filter_formula:
params["filterByFormula"] = filter_formula
result = await api_call(f"{base_id}/{table_name}", params=params)
if "error" in result:
return f"Error: {result['error']}"
records = result.get("records", [])
if not records:
return "No records found in this table."
# Format the records for display
formatted_records = []
for i, record in enumerate(records):
record_id = record.get("id", "unknown")
fields = record.get("fields", {})
field_text = ", ".join([f"{k}: {v}" for k, v in fields.items()])
formatted_records.append(f"{i+1}. ID: {record_id} - {field_text}")
return "Records:\n" + "\n".join(formatted_records)
@app.tool()
async def get_record(table_name: str, record_id: str) -> str:
"""Get a specific record from a table"""
if not token:
return "Please provide an Airtable API token to get records."
if not base_id:
return "Error: No base ID set. Please set AIRTABLE_BASE_ID environment variable."
result = await api_call(f"{base_id}/{table_name}/{record_id}")
if "error" in result:
return f"Error: {result['error']}"
fields = result.get("fields", {})
if not fields:
return f"Record {record_id} found but contains no fields."
# Format the fields for display
formatted_fields = []
for key, value in fields.items():
formatted_fields.append(f"{key}: {value}")
return f"Record ID: {record_id}\n" + "\n".join(formatted_fields)
@app.tool()
async def create_records(table_name: str, records_json: str) -> str:
"""Create records in a table from JSON string"""
if not token:
return "Please provide an Airtable API token to create records."
if not base_id:
return "Error: No base ID set. Please set AIRTABLE_BASE_ID environment variable."
try:
records_data = json.loads(records_json)
# Format the records for Airtable API
if not isinstance(records_data, list):
records_data = [records_data]
records = [{"fields": record} for record in records_data]
data = {"records": records}
result = await api_call(f"{base_id}/{table_name}", method="POST", data=data)
if "error" in result:
return f"Error: {result['error']}"
created_records = result.get("records", [])
return f"Successfully created {len(created_records)} records."
except json.JSONDecodeError:
return "Error: Invalid JSON format in records_json parameter."
except Exception as e:
return f"Error creating records: {str(e)}"
@app.tool()
async def update_records(table_name: str, records_json: str) -> str:
"""Update records in a table from JSON string"""
if not token:
return "Please provide an Airtable API token to update records."
if not base_id:
return "Error: No base ID set. Please set AIRTABLE_BASE_ID environment variable."
try:
records_data = json.loads(records_json)
# Format the records for Airtable API
if not isinstance(records_data, list):
records_data = [records_data]
records = []
for record in records_data:
if "id" not in record:
return "Error: Each record must have an 'id' field."
rec_id = record.pop("id")
fields = record.get("fields", record) # Support both {id, fields} format and direct fields
records.append({"id": rec_id, "fields": fields})
data = {"records": records}
result = await api_call(f"{base_id}/{table_name}", method="PATCH", data=data)
if "error" in result:
return f"Error: {result['error']}"
updated_records = result.get("records", [])
return f"Successfully updated {len(updated_records)} records."
except json.JSONDecodeError:
return "Error: Invalid JSON format in records_json parameter."
except Exception as e:
return f"Error updating records: {str(e)}"
@app.tool()
async def set_base_id(base_id_param: str) -> str:
"""Set the current Airtable base ID"""
global base_id
base_id = base_id_param
return f"Base ID set to: {base_id}"
# Add Claude-specific methods
@app.rpc_method("resources/list")
async def resources_list(params: Dict = None) -> Dict:
"""List available Airtable resources for Claude"""
try:
resources = [
{
"id": "airtable_bases",
"name": "Airtable Bases",
"description": "The Airtable bases accessible with your API token"
},
{
"id": "airtable_tables",
"name": "Airtable Tables",
"description": "Tables in your current Airtable base"
},
{
"id": "airtable_records",
"name": "Airtable Records",
"description": "Records in your Airtable tables"
}
]
return {"resources": resources}
except Exception as e:
error_trace = traceback.format_exc()
logger.error(f"Error in resources/list: {str(e)}\n{error_trace}")
return {"error": {"code": -32000, "message": str(e)}}
@app.rpc_method("prompts/list")
async def prompts_list(params: Dict = None) -> Dict:
"""List available prompts for Claude"""
try:
prompts = [
{
"id": "list_tables_prompt",
"name": "List Tables",
"description": "List all tables in your Airtable base"
},
{
"id": "list_records_prompt",
"name": "List Records",
"description": "List records from a specific Airtable table"
},
{
"id": "create_record_prompt",
"name": "Create Record",
"description": "Create a new record in an Airtable table"
}
]
return {"prompts": prompts}
except Exception as e:
error_trace = traceback.format_exc()
logger.error(f"Error in prompts/list: {str(e)}\n{error_trace}")
return {"error": {"code": -32000, "message": str(e)}}
# Start the server
if __name__ == "__main__":
app.start()