workspace-mcp 0.2.0__py3-none-any.whl
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.
- auth/__init__.py +1 -0
- auth/google_auth.py +549 -0
- auth/oauth_callback_server.py +241 -0
- auth/oauth_responses.py +223 -0
- auth/scopes.py +108 -0
- auth/service_decorator.py +404 -0
- core/__init__.py +1 -0
- core/server.py +214 -0
- core/utils.py +162 -0
- gcalendar/__init__.py +1 -0
- gcalendar/calendar_tools.py +496 -0
- gchat/__init__.py +6 -0
- gchat/chat_tools.py +254 -0
- gdocs/__init__.py +0 -0
- gdocs/docs_tools.py +244 -0
- gdrive/__init__.py +0 -0
- gdrive/drive_tools.py +362 -0
- gforms/__init__.py +3 -0
- gforms/forms_tools.py +318 -0
- gmail/__init__.py +1 -0
- gmail/gmail_tools.py +807 -0
- gsheets/__init__.py +23 -0
- gsheets/sheets_tools.py +393 -0
- gslides/__init__.py +0 -0
- gslides/slides_tools.py +316 -0
- main.py +160 -0
- workspace_mcp-0.2.0.dist-info/METADATA +29 -0
- workspace_mcp-0.2.0.dist-info/RECORD +32 -0
- workspace_mcp-0.2.0.dist-info/WHEEL +5 -0
- workspace_mcp-0.2.0.dist-info/entry_points.txt +2 -0
- workspace_mcp-0.2.0.dist-info/licenses/LICENSE +21 -0
- workspace_mcp-0.2.0.dist-info/top_level.txt +11 -0
gsheets/__init__.py
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
"""
|
2
|
+
Google Sheets MCP Integration
|
3
|
+
|
4
|
+
This module provides MCP tools for interacting with Google Sheets API.
|
5
|
+
"""
|
6
|
+
|
7
|
+
from .sheets_tools import (
|
8
|
+
list_spreadsheets,
|
9
|
+
get_spreadsheet_info,
|
10
|
+
read_sheet_values,
|
11
|
+
modify_sheet_values,
|
12
|
+
create_spreadsheet,
|
13
|
+
create_sheet,
|
14
|
+
)
|
15
|
+
|
16
|
+
__all__ = [
|
17
|
+
"list_spreadsheets",
|
18
|
+
"get_spreadsheet_info",
|
19
|
+
"read_sheet_values",
|
20
|
+
"modify_sheet_values",
|
21
|
+
"create_spreadsheet",
|
22
|
+
"create_sheet",
|
23
|
+
]
|
gsheets/sheets_tools.py
ADDED
@@ -0,0 +1,393 @@
|
|
1
|
+
"""
|
2
|
+
Google Sheets MCP Tools
|
3
|
+
|
4
|
+
This module provides MCP tools for interacting with Google Sheets API.
|
5
|
+
"""
|
6
|
+
|
7
|
+
import logging
|
8
|
+
import asyncio
|
9
|
+
from typing import List, Optional
|
10
|
+
|
11
|
+
from mcp import types
|
12
|
+
from googleapiclient.errors import HttpError
|
13
|
+
|
14
|
+
from auth.service_decorator import require_google_service
|
15
|
+
from core.server import server
|
16
|
+
|
17
|
+
# Configure module logger
|
18
|
+
logger = logging.getLogger(__name__)
|
19
|
+
|
20
|
+
|
21
|
+
@server.tool()
|
22
|
+
@require_google_service("drive", "drive_read")
|
23
|
+
async def list_spreadsheets(
|
24
|
+
service,
|
25
|
+
user_google_email: str,
|
26
|
+
max_results: int = 25,
|
27
|
+
) -> str:
|
28
|
+
"""
|
29
|
+
Lists spreadsheets from Google Drive that the user has access to.
|
30
|
+
|
31
|
+
Args:
|
32
|
+
user_google_email (str): The user's Google email address. Required.
|
33
|
+
max_results (int): Maximum number of spreadsheets to return. Defaults to 25.
|
34
|
+
|
35
|
+
Returns:
|
36
|
+
str: A formatted list of spreadsheet files (name, ID, modified time).
|
37
|
+
"""
|
38
|
+
logger.info(f"[list_spreadsheets] Invoked. Email: '{user_google_email}'")
|
39
|
+
|
40
|
+
try:
|
41
|
+
files_response = await asyncio.to_thread(
|
42
|
+
service.files()
|
43
|
+
.list(
|
44
|
+
q="mimeType='application/vnd.google-apps.spreadsheet'",
|
45
|
+
pageSize=max_results,
|
46
|
+
fields="files(id,name,modifiedTime,webViewLink)",
|
47
|
+
orderBy="modifiedTime desc",
|
48
|
+
)
|
49
|
+
.execute
|
50
|
+
)
|
51
|
+
|
52
|
+
files = files_response.get("files", [])
|
53
|
+
if not files:
|
54
|
+
return f"No spreadsheets found for {user_google_email}."
|
55
|
+
|
56
|
+
spreadsheets_list = [
|
57
|
+
f"- \"{file['name']}\" (ID: {file['id']}) | Modified: {file.get('modifiedTime', 'Unknown')} | Link: {file.get('webViewLink', 'No link')}"
|
58
|
+
for file in files
|
59
|
+
]
|
60
|
+
|
61
|
+
text_output = (
|
62
|
+
f"Successfully listed {len(files)} spreadsheets for {user_google_email}:\n"
|
63
|
+
+ "\n".join(spreadsheets_list)
|
64
|
+
)
|
65
|
+
|
66
|
+
logger.info(f"Successfully listed {len(files)} spreadsheets for {user_google_email}.")
|
67
|
+
return text_output
|
68
|
+
|
69
|
+
except HttpError as error:
|
70
|
+
message = f"API error listing spreadsheets: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
71
|
+
logger.error(message, exc_info=True)
|
72
|
+
raise Exception(message)
|
73
|
+
except Exception as e:
|
74
|
+
message = f"Unexpected error listing spreadsheets: {e}."
|
75
|
+
logger.exception(message)
|
76
|
+
raise Exception(message)
|
77
|
+
|
78
|
+
|
79
|
+
@server.tool()
|
80
|
+
@require_google_service("sheets", "sheets_read")
|
81
|
+
async def get_spreadsheet_info(
|
82
|
+
service,
|
83
|
+
user_google_email: str,
|
84
|
+
spreadsheet_id: str,
|
85
|
+
) -> str:
|
86
|
+
"""
|
87
|
+
Gets information about a specific spreadsheet including its sheets.
|
88
|
+
|
89
|
+
Args:
|
90
|
+
user_google_email (str): The user's Google email address. Required.
|
91
|
+
spreadsheet_id (str): The ID of the spreadsheet to get info for. Required.
|
92
|
+
|
93
|
+
Returns:
|
94
|
+
str: Formatted spreadsheet information including title and sheets list.
|
95
|
+
"""
|
96
|
+
logger.info(f"[get_spreadsheet_info] Invoked. Email: '{user_google_email}', Spreadsheet ID: {spreadsheet_id}")
|
97
|
+
|
98
|
+
try:
|
99
|
+
spreadsheet = await asyncio.to_thread(
|
100
|
+
service.spreadsheets().get(spreadsheetId=spreadsheet_id).execute
|
101
|
+
)
|
102
|
+
|
103
|
+
title = spreadsheet.get("properties", {}).get("title", "Unknown")
|
104
|
+
sheets = spreadsheet.get("sheets", [])
|
105
|
+
|
106
|
+
sheets_info = []
|
107
|
+
for sheet in sheets:
|
108
|
+
sheet_props = sheet.get("properties", {})
|
109
|
+
sheet_name = sheet_props.get("title", "Unknown")
|
110
|
+
sheet_id = sheet_props.get("sheetId", "Unknown")
|
111
|
+
grid_props = sheet_props.get("gridProperties", {})
|
112
|
+
rows = grid_props.get("rowCount", "Unknown")
|
113
|
+
cols = grid_props.get("columnCount", "Unknown")
|
114
|
+
|
115
|
+
sheets_info.append(
|
116
|
+
f" - \"{sheet_name}\" (ID: {sheet_id}) | Size: {rows}x{cols}"
|
117
|
+
)
|
118
|
+
|
119
|
+
text_output = (
|
120
|
+
f"Spreadsheet: \"{title}\" (ID: {spreadsheet_id})\n"
|
121
|
+
f"Sheets ({len(sheets)}):\n"
|
122
|
+
+ "\n".join(sheets_info) if sheets_info else " No sheets found"
|
123
|
+
)
|
124
|
+
|
125
|
+
logger.info(f"Successfully retrieved info for spreadsheet {spreadsheet_id} for {user_google_email}.")
|
126
|
+
return text_output
|
127
|
+
|
128
|
+
except HttpError as error:
|
129
|
+
message = f"API error getting spreadsheet info: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
130
|
+
logger.error(message, exc_info=True)
|
131
|
+
raise Exception(message)
|
132
|
+
except Exception as e:
|
133
|
+
message = f"Unexpected error getting spreadsheet info: {e}."
|
134
|
+
logger.exception(message)
|
135
|
+
raise Exception(message)
|
136
|
+
|
137
|
+
|
138
|
+
@server.tool()
|
139
|
+
@require_google_service("sheets", "sheets_read")
|
140
|
+
async def read_sheet_values(
|
141
|
+
service,
|
142
|
+
user_google_email: str,
|
143
|
+
spreadsheet_id: str,
|
144
|
+
range_name: str = "A1:Z1000",
|
145
|
+
) -> str:
|
146
|
+
"""
|
147
|
+
Reads values from a specific range in a Google Sheet.
|
148
|
+
|
149
|
+
Args:
|
150
|
+
user_google_email (str): The user's Google email address. Required.
|
151
|
+
spreadsheet_id (str): The ID of the spreadsheet. Required.
|
152
|
+
range_name (str): The range to read (e.g., "Sheet1!A1:D10", "A1:D10"). Defaults to "A1:Z1000".
|
153
|
+
|
154
|
+
Returns:
|
155
|
+
str: The formatted values from the specified range.
|
156
|
+
"""
|
157
|
+
logger.info(f"[read_sheet_values] Invoked. Email: '{user_google_email}', Spreadsheet: {spreadsheet_id}, Range: {range_name}")
|
158
|
+
|
159
|
+
try:
|
160
|
+
result = await asyncio.to_thread(
|
161
|
+
service.spreadsheets()
|
162
|
+
.values()
|
163
|
+
.get(spreadsheetId=spreadsheet_id, range=range_name)
|
164
|
+
.execute
|
165
|
+
)
|
166
|
+
|
167
|
+
values = result.get("values", [])
|
168
|
+
if not values:
|
169
|
+
return f"No data found in range '{range_name}' for {user_google_email}."
|
170
|
+
|
171
|
+
# Format the output as a readable table
|
172
|
+
formatted_rows = []
|
173
|
+
for i, row in enumerate(values, 1):
|
174
|
+
# Pad row with empty strings to show structure
|
175
|
+
padded_row = row + [""] * max(0, len(values[0]) - len(row)) if values else row
|
176
|
+
formatted_rows.append(f"Row {i:2d}: {padded_row}")
|
177
|
+
|
178
|
+
text_output = (
|
179
|
+
f"Successfully read {len(values)} rows from range '{range_name}' in spreadsheet {spreadsheet_id} for {user_google_email}:\n"
|
180
|
+
+ "\n".join(formatted_rows[:50]) # Limit to first 50 rows for readability
|
181
|
+
+ (f"\n... and {len(values) - 50} more rows" if len(values) > 50 else "")
|
182
|
+
)
|
183
|
+
|
184
|
+
logger.info(f"Successfully read {len(values)} rows for {user_google_email}.")
|
185
|
+
return text_output
|
186
|
+
|
187
|
+
except HttpError as error:
|
188
|
+
message = f"API error reading sheet values: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
189
|
+
logger.error(message, exc_info=True)
|
190
|
+
raise Exception(message)
|
191
|
+
except Exception as e:
|
192
|
+
message = f"Unexpected error reading sheet values: {e}."
|
193
|
+
logger.exception(message)
|
194
|
+
raise Exception(message)
|
195
|
+
|
196
|
+
|
197
|
+
@server.tool()
|
198
|
+
@require_google_service("sheets", "sheets_write")
|
199
|
+
async def modify_sheet_values(
|
200
|
+
service,
|
201
|
+
user_google_email: str,
|
202
|
+
spreadsheet_id: str,
|
203
|
+
range_name: str,
|
204
|
+
values: Optional[List[List[str]]] = None,
|
205
|
+
value_input_option: str = "USER_ENTERED",
|
206
|
+
clear_values: bool = False,
|
207
|
+
) -> str:
|
208
|
+
"""
|
209
|
+
Modifies values in a specific range of a Google Sheet - can write, update, or clear values.
|
210
|
+
|
211
|
+
Args:
|
212
|
+
user_google_email (str): The user's Google email address. Required.
|
213
|
+
spreadsheet_id (str): The ID of the spreadsheet. Required.
|
214
|
+
range_name (str): The range to modify (e.g., "Sheet1!A1:D10", "A1:D10"). Required.
|
215
|
+
values (Optional[List[List[str]]]): 2D array of values to write/update. Required unless clear_values=True.
|
216
|
+
value_input_option (str): How to interpret input values ("RAW" or "USER_ENTERED"). Defaults to "USER_ENTERED".
|
217
|
+
clear_values (bool): If True, clears the range instead of writing values. Defaults to False.
|
218
|
+
|
219
|
+
Returns:
|
220
|
+
str: Confirmation message of the successful modification operation.
|
221
|
+
"""
|
222
|
+
operation = "clear" if clear_values else "write"
|
223
|
+
logger.info(f"[modify_sheet_values] Invoked. Operation: {operation}, Email: '{user_google_email}', Spreadsheet: {spreadsheet_id}, Range: {range_name}")
|
224
|
+
|
225
|
+
if not clear_values and not values:
|
226
|
+
raise Exception("Either 'values' must be provided or 'clear_values' must be True.")
|
227
|
+
|
228
|
+
try:
|
229
|
+
if clear_values:
|
230
|
+
result = await asyncio.to_thread(
|
231
|
+
service.spreadsheets()
|
232
|
+
.values()
|
233
|
+
.clear(spreadsheetId=spreadsheet_id, range=range_name)
|
234
|
+
.execute
|
235
|
+
)
|
236
|
+
|
237
|
+
cleared_range = result.get("clearedRange", range_name)
|
238
|
+
text_output = f"Successfully cleared range '{cleared_range}' in spreadsheet {spreadsheet_id} for {user_google_email}."
|
239
|
+
logger.info(f"Successfully cleared range '{cleared_range}' for {user_google_email}.")
|
240
|
+
else:
|
241
|
+
body = {"values": values}
|
242
|
+
|
243
|
+
result = await asyncio.to_thread(
|
244
|
+
service.spreadsheets()
|
245
|
+
.values()
|
246
|
+
.update(
|
247
|
+
spreadsheetId=spreadsheet_id,
|
248
|
+
range=range_name,
|
249
|
+
valueInputOption=value_input_option,
|
250
|
+
body=body,
|
251
|
+
)
|
252
|
+
.execute
|
253
|
+
)
|
254
|
+
|
255
|
+
updated_cells = result.get("updatedCells", 0)
|
256
|
+
updated_rows = result.get("updatedRows", 0)
|
257
|
+
updated_columns = result.get("updatedColumns", 0)
|
258
|
+
|
259
|
+
text_output = (
|
260
|
+
f"Successfully updated range '{range_name}' in spreadsheet {spreadsheet_id} for {user_google_email}. "
|
261
|
+
f"Updated: {updated_cells} cells, {updated_rows} rows, {updated_columns} columns."
|
262
|
+
)
|
263
|
+
logger.info(f"Successfully updated {updated_cells} cells for {user_google_email}.")
|
264
|
+
|
265
|
+
return text_output
|
266
|
+
|
267
|
+
except HttpError as error:
|
268
|
+
message = f"API error modifying sheet values: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
269
|
+
logger.error(message, exc_info=True)
|
270
|
+
raise Exception(message)
|
271
|
+
except Exception as e:
|
272
|
+
message = f"Unexpected error modifying sheet values: {e}."
|
273
|
+
logger.exception(message)
|
274
|
+
raise Exception(message)
|
275
|
+
|
276
|
+
|
277
|
+
@server.tool()
|
278
|
+
@require_google_service("sheets", "sheets_write")
|
279
|
+
async def create_spreadsheet(
|
280
|
+
service,
|
281
|
+
user_google_email: str,
|
282
|
+
title: str,
|
283
|
+
sheet_names: Optional[List[str]] = None,
|
284
|
+
) -> str:
|
285
|
+
"""
|
286
|
+
Creates a new Google Spreadsheet.
|
287
|
+
|
288
|
+
Args:
|
289
|
+
user_google_email (str): The user's Google email address. Required.
|
290
|
+
title (str): The title of the new spreadsheet. Required.
|
291
|
+
sheet_names (Optional[List[str]]): List of sheet names to create. If not provided, creates one sheet with default name.
|
292
|
+
|
293
|
+
Returns:
|
294
|
+
str: Information about the newly created spreadsheet including ID and URL.
|
295
|
+
"""
|
296
|
+
logger.info(f"[create_spreadsheet] Invoked. Email: '{user_google_email}', Title: {title}")
|
297
|
+
|
298
|
+
try:
|
299
|
+
spreadsheet_body = {
|
300
|
+
"properties": {
|
301
|
+
"title": title
|
302
|
+
}
|
303
|
+
}
|
304
|
+
|
305
|
+
if sheet_names:
|
306
|
+
spreadsheet_body["sheets"] = [
|
307
|
+
{"properties": {"title": sheet_name}} for sheet_name in sheet_names
|
308
|
+
]
|
309
|
+
|
310
|
+
spreadsheet = await asyncio.to_thread(
|
311
|
+
service.spreadsheets().create(body=spreadsheet_body).execute
|
312
|
+
)
|
313
|
+
|
314
|
+
spreadsheet_id = spreadsheet.get("spreadsheetId")
|
315
|
+
spreadsheet_url = spreadsheet.get("spreadsheetUrl")
|
316
|
+
|
317
|
+
text_output = (
|
318
|
+
f"Successfully created spreadsheet '{title}' for {user_google_email}. "
|
319
|
+
f"ID: {spreadsheet_id} | URL: {spreadsheet_url}"
|
320
|
+
)
|
321
|
+
|
322
|
+
logger.info(f"Successfully created spreadsheet for {user_google_email}. ID: {spreadsheet_id}")
|
323
|
+
return text_output
|
324
|
+
|
325
|
+
except HttpError as error:
|
326
|
+
message = f"API error creating spreadsheet: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
327
|
+
logger.error(message, exc_info=True)
|
328
|
+
raise Exception(message)
|
329
|
+
except Exception as e:
|
330
|
+
message = f"Unexpected error creating spreadsheet: {e}."
|
331
|
+
logger.exception(message)
|
332
|
+
raise Exception(message)
|
333
|
+
|
334
|
+
|
335
|
+
@server.tool()
|
336
|
+
@require_google_service("sheets", "sheets_write")
|
337
|
+
async def create_sheet(
|
338
|
+
service,
|
339
|
+
user_google_email: str,
|
340
|
+
spreadsheet_id: str,
|
341
|
+
sheet_name: str,
|
342
|
+
) -> str:
|
343
|
+
"""
|
344
|
+
Creates a new sheet within an existing spreadsheet.
|
345
|
+
|
346
|
+
Args:
|
347
|
+
user_google_email (str): The user's Google email address. Required.
|
348
|
+
spreadsheet_id (str): The ID of the spreadsheet. Required.
|
349
|
+
sheet_name (str): The name of the new sheet. Required.
|
350
|
+
|
351
|
+
Returns:
|
352
|
+
str: Confirmation message of the successful sheet creation.
|
353
|
+
"""
|
354
|
+
logger.info(f"[create_sheet] Invoked. Email: '{user_google_email}', Spreadsheet: {spreadsheet_id}, Sheet: {sheet_name}")
|
355
|
+
|
356
|
+
try:
|
357
|
+
request_body = {
|
358
|
+
"requests": [
|
359
|
+
{
|
360
|
+
"addSheet": {
|
361
|
+
"properties": {
|
362
|
+
"title": sheet_name
|
363
|
+
}
|
364
|
+
}
|
365
|
+
}
|
366
|
+
]
|
367
|
+
}
|
368
|
+
|
369
|
+
response = await asyncio.to_thread(
|
370
|
+
service.spreadsheets()
|
371
|
+
.batchUpdate(spreadsheetId=spreadsheet_id, body=request_body)
|
372
|
+
.execute
|
373
|
+
)
|
374
|
+
|
375
|
+
sheet_id = response["replies"][0]["addSheet"]["properties"]["sheetId"]
|
376
|
+
|
377
|
+
text_output = (
|
378
|
+
f"Successfully created sheet '{sheet_name}' (ID: {sheet_id}) in spreadsheet {spreadsheet_id} for {user_google_email}."
|
379
|
+
)
|
380
|
+
|
381
|
+
logger.info(f"Successfully created sheet for {user_google_email}. Sheet ID: {sheet_id}")
|
382
|
+
return text_output
|
383
|
+
|
384
|
+
except HttpError as error:
|
385
|
+
message = f"API error creating sheet: {error}. You might need to re-authenticate. LLM: Try 'start_google_auth' with user's email and service_name='Google Sheets'."
|
386
|
+
logger.error(message, exc_info=True)
|
387
|
+
raise Exception(message)
|
388
|
+
except Exception as e:
|
389
|
+
message = f"Unexpected error creating sheet: {e}."
|
390
|
+
logger.exception(message)
|
391
|
+
raise Exception(message)
|
392
|
+
|
393
|
+
|
gslides/__init__.py
ADDED
File without changes
|