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.
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
+ ]
@@ -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