mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-09 03:11:58 +00:00
feat(google-workspace): Drive write ops + Docs/Sheets create/append (#21895)
Expand the google-workspace skill beyond read-only access to Drive and Docs. Sheets already had full scope — just adds the missing create verb. New subcommands: - drive get : metadata for a single file - drive upload : upload a local file (auto MIME detection) - drive download : download or export (Docs/Sheets/Slides export to pdf/csv/pdf by default) - drive create-folder - drive share : user/group/domain/anyone + reader/writer/etc. - drive delete : default trashes (reversible); --permanent skips the trash - sheets create : new spreadsheet with optional first-tab name - docs create : new doc, optional initial body - docs append : append text at end of an existing doc Scope changes: - drive.readonly -> drive - documents.readonly -> documents Existing users with old tokens will hit the existing partial-scope warning path (AUTHENTICATED (partial) ...) — the troubleshooting table now points them at $GSETUP --revoke + redo steps 3-5 to pick up the write scopes.
This commit is contained in:
parent
674fad1483
commit
e43d2fe520
3 changed files with 415 additions and 6 deletions
|
|
@ -1,7 +1,7 @@
|
|||
---
|
||||
name: google-workspace
|
||||
description: "Gmail, Calendar, Drive, Docs, Sheets via gws CLI or Python."
|
||||
version: 1.0.1
|
||||
version: 1.1.0
|
||||
author: Nous Research
|
||||
license: MIT
|
||||
required_credential_files:
|
||||
|
|
@ -216,8 +216,36 @@ $GAPI calendar delete EVENT_ID
|
|||
### Drive
|
||||
|
||||
```bash
|
||||
# Search existing files
|
||||
$GAPI drive search "quarterly report" --max 10
|
||||
$GAPI drive search "mimeType='application/pdf'" --raw-query --max 5
|
||||
|
||||
# Get metadata for a single file
|
||||
$GAPI drive get FILE_ID
|
||||
|
||||
# Upload a local file (auto-detects MIME type)
|
||||
$GAPI drive upload /path/to/report.pdf
|
||||
$GAPI drive upload /path/to/image.png --name "Logo.png" --parent FOLDER_ID
|
||||
|
||||
# Download (binary files download as-is; Google-native files export to a
|
||||
# sensible default — Docs→pdf, Sheets→csv, Slides→pdf, Drawings→png)
|
||||
$GAPI drive download FILE_ID
|
||||
$GAPI drive download DOC_ID --output ~/doc.pdf
|
||||
$GAPI drive download DOC_ID --export-mime text/plain --output ~/doc.txt
|
||||
|
||||
# Create a folder
|
||||
$GAPI drive create-folder "Reports"
|
||||
$GAPI drive create-folder "Q4" --parent FOLDER_ID
|
||||
|
||||
# Share
|
||||
$GAPI drive share FILE_ID --email alice@example.com --role reader
|
||||
$GAPI drive share FILE_ID --email alice@example.com --role writer --notify
|
||||
$GAPI drive share FILE_ID --type anyone --role reader # anyone with link
|
||||
$GAPI drive share FILE_ID --type domain --domain example.com --role reader
|
||||
|
||||
# Delete — defaults to trash (reversible). Use --permanent to skip the trash.
|
||||
$GAPI drive delete FILE_ID
|
||||
$GAPI drive delete FILE_ID --permanent
|
||||
```
|
||||
|
||||
### Contacts
|
||||
|
|
@ -229,6 +257,10 @@ $GAPI contacts list --max 20
|
|||
### Sheets
|
||||
|
||||
```bash
|
||||
# Create a new spreadsheet
|
||||
$GAPI sheets create --title "Q4 Budget"
|
||||
$GAPI sheets create --title "Inventory" --sheet-name "Stock"
|
||||
|
||||
# Read
|
||||
$GAPI sheets get SHEET_ID "Sheet1!A1:D10"
|
||||
|
||||
|
|
@ -242,7 +274,15 @@ $GAPI sheets append SHEET_ID "Sheet1!A:C" --values '[["new","row","data"]]'
|
|||
### Docs
|
||||
|
||||
```bash
|
||||
# Read
|
||||
$GAPI docs get DOC_ID
|
||||
|
||||
# Create a new Doc (optionally seeded with body text)
|
||||
$GAPI docs create --title "Meeting Notes"
|
||||
$GAPI docs create --title "Draft" --body "First paragraph..."
|
||||
|
||||
# Append text to the end of an existing Doc
|
||||
$GAPI docs append DOC_ID --text "Additional content to append"
|
||||
```
|
||||
|
||||
## Output Format
|
||||
|
|
@ -255,12 +295,21 @@ All commands return JSON. Parse with `jq` or read directly. Key fields:
|
|||
- **Calendar list**: `[{id, summary, start, end, location, description, htmlLink}]`
|
||||
- **Calendar create**: `{status: "created", id, summary, htmlLink}`
|
||||
- **Drive search**: `[{id, name, mimeType, modifiedTime, webViewLink}]`
|
||||
- **Drive get**: `{id, name, mimeType, modifiedTime, size, webViewLink, parents, owners}`
|
||||
- **Drive upload**: `{status: "uploaded", id, name, mimeType, webViewLink}`
|
||||
- **Drive download**: `{status: "downloaded", id, name, path, mimeType}`
|
||||
- **Drive create-folder**: `{status: "created", id, name, webViewLink}`
|
||||
- **Drive share**: `{status: "shared", permissionId, fileId, role, type}`
|
||||
- **Drive delete**: `{status: "trashed" | "deleted", fileId, permanent}`
|
||||
- **Contacts list**: `[{name, emails: [...], phones: [...]}]`
|
||||
- **Sheets get**: `[[cell, cell, ...], ...]`
|
||||
- **Sheets create**: `{status: "created", spreadsheetId, title, spreadsheetUrl}`
|
||||
- **Docs create**: `{status: "created", documentId, title, url}`
|
||||
- **Docs append**: `{status: "appended", documentId, inserted_at, characters}`
|
||||
|
||||
## Rules
|
||||
|
||||
1. **Never send email or create/delete events without confirming with the user first.** Show the draft content and ask for approval.
|
||||
1. **Never send email, create/delete calendar events, delete Drive files, share files, or modify Docs/Sheets without confirming with the user first.** Show what will be done (recipients, file IDs, content, share role) and ask for approval. For `drive delete`, prefer the default trash (reversible) over `--permanent`.
|
||||
2. **Check auth before first use** — run `setup.py --check`. If it fails, guide the user through setup.
|
||||
3. **Use the Gmail search syntax reference** for complex queries — load it with `skill_view("google-workspace", file_path="references/gmail-search-syntax.md")`.
|
||||
4. **Calendar times must include timezone** — always use ISO 8601 with offset (e.g., `2026-03-01T10:00:00-06:00`) or UTC (`Z`).
|
||||
|
|
@ -273,6 +322,7 @@ All commands return JSON. Parse with `jq` or read directly. Key fields:
|
|||
| `NOT_AUTHENTICATED` | Run setup Steps 2-5 above |
|
||||
| `REFRESH_FAILED` | Token revoked or expired — redo Steps 3-5 |
|
||||
| `HttpError 403: Insufficient Permission` | Missing API scope — `$GSETUP --revoke` then redo Steps 3-5 |
|
||||
| `AUTHENTICATED (partial)` or "Token missing scopes" | New write capabilities (Drive write/delete, Docs create/edit) require re-authorization. `$GSETUP --revoke` then redo Steps 3-5 to grant the upgraded scopes. |
|
||||
| `HttpError 403: Access Not Configured` | API not enabled — user needs to enable it in Google Cloud Console |
|
||||
| `ModuleNotFoundError` | Run `$GSETUP --install-deps` |
|
||||
| Advanced Protection blocks auth | Workspace admin must allowlist the OAuth client ID |
|
||||
|
|
|
|||
|
|
@ -47,10 +47,10 @@ SCOPES = [
|
|||
"https://www.googleapis.com/auth/gmail.send",
|
||||
"https://www.googleapis.com/auth/gmail.modify",
|
||||
"https://www.googleapis.com/auth/calendar",
|
||||
"https://www.googleapis.com/auth/drive.readonly",
|
||||
"https://www.googleapis.com/auth/drive",
|
||||
"https://www.googleapis.com/auth/contacts.readonly",
|
||||
"https://www.googleapis.com/auth/spreadsheets",
|
||||
"https://www.googleapis.com/auth/documents.readonly",
|
||||
"https://www.googleapis.com/auth/documents",
|
||||
]
|
||||
|
||||
|
||||
|
|
@ -587,6 +587,213 @@ def drive_search(args):
|
|||
print(json.dumps(files, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_get(args):
|
||||
"""Get metadata for a single Drive file by ID."""
|
||||
fields = "id, name, mimeType, modifiedTime, size, webViewLink, parents, owners(emailAddress)"
|
||||
if _gws_binary():
|
||||
result = _run_gws(
|
||||
["drive", "files", "get"],
|
||||
params={"fileId": args.file_id, "fields": fields},
|
||||
)
|
||||
print(json.dumps(result, indent=2, ensure_ascii=False))
|
||||
return
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
result = service.files().get(fileId=args.file_id, fields=fields).execute()
|
||||
print(json.dumps(result, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_upload(args):
|
||||
"""Upload a local file to Drive. Falls through to Python client even when gws
|
||||
is installed, because gws doesn't do multipart uploads."""
|
||||
import mimetypes
|
||||
from googleapiclient.http import MediaFileUpload
|
||||
|
||||
local_path = Path(args.path).expanduser()
|
||||
if not local_path.exists():
|
||||
print(f"ERROR: file not found: {local_path}", file=sys.stderr)
|
||||
sys.exit(1)
|
||||
|
||||
mime = args.mime_type or mimetypes.guess_type(str(local_path))[0] or "application/octet-stream"
|
||||
metadata = {"name": args.name or local_path.name}
|
||||
if args.parent:
|
||||
metadata["parents"] = [args.parent]
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
media = MediaFileUpload(str(local_path), mimetype=mime, resumable=True)
|
||||
result = service.files().create(
|
||||
body=metadata,
|
||||
media_body=media,
|
||||
fields="id, name, mimeType, webViewLink",
|
||||
).execute()
|
||||
print(json.dumps({
|
||||
"status": "uploaded",
|
||||
"id": result["id"],
|
||||
"name": result.get("name", ""),
|
||||
"mimeType": result.get("mimeType", ""),
|
||||
"webViewLink": result.get("webViewLink", ""),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_download(args):
|
||||
"""Download a Drive file to a local path. Google-native files (Docs/Sheets/Slides)
|
||||
must be exported; binary files are downloaded as-is."""
|
||||
import io
|
||||
from googleapiclient.http import MediaIoBaseDownload
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
|
||||
# Look up the file to decide download vs export.
|
||||
meta = service.files().get(fileId=args.file_id, fields="id, name, mimeType").execute()
|
||||
mime = meta.get("mimeType", "")
|
||||
name = meta.get("name", args.file_id)
|
||||
|
||||
# Map Google-native MIME types to a sensible export default.
|
||||
native_export_map = {
|
||||
"application/vnd.google-apps.document": ("application/pdf", ".pdf"),
|
||||
"application/vnd.google-apps.spreadsheet": ("text/csv", ".csv"),
|
||||
"application/vnd.google-apps.presentation": ("application/pdf", ".pdf"),
|
||||
"application/vnd.google-apps.drawing": ("image/png", ".png"),
|
||||
}
|
||||
|
||||
out_path = Path(args.output).expanduser() if args.output else Path.cwd() / name
|
||||
|
||||
if mime in native_export_map:
|
||||
export_mime = args.export_mime or native_export_map[mime][0]
|
||||
default_ext = native_export_map[mime][1]
|
||||
if not args.output and not out_path.suffix:
|
||||
out_path = out_path.with_suffix(default_ext)
|
||||
request = service.files().export_media(fileId=args.file_id, mimeType=export_mime)
|
||||
else:
|
||||
request = service.files().get_media(fileId=args.file_id)
|
||||
|
||||
out_path.parent.mkdir(parents=True, exist_ok=True)
|
||||
fh = io.FileIO(str(out_path), "wb")
|
||||
downloader = MediaIoBaseDownload(fh, request)
|
||||
done = False
|
||||
while not done:
|
||||
_, done = downloader.next_chunk()
|
||||
fh.close()
|
||||
|
||||
print(json.dumps({
|
||||
"status": "downloaded",
|
||||
"id": args.file_id,
|
||||
"name": name,
|
||||
"path": str(out_path),
|
||||
"mimeType": mime,
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_create_folder(args):
|
||||
body = {
|
||||
"name": args.name,
|
||||
"mimeType": "application/vnd.google-apps.folder",
|
||||
}
|
||||
if args.parent:
|
||||
body["parents"] = [args.parent]
|
||||
|
||||
if _gws_binary():
|
||||
result = _run_gws(
|
||||
["drive", "files", "create"],
|
||||
params={"fields": "id, name, webViewLink"},
|
||||
body=body,
|
||||
)
|
||||
print(json.dumps({
|
||||
"status": "created",
|
||||
"id": result["id"],
|
||||
"name": result.get("name", ""),
|
||||
"webViewLink": result.get("webViewLink", ""),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
return
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
result = service.files().create(body=body, fields="id, name, webViewLink").execute()
|
||||
print(json.dumps({
|
||||
"status": "created",
|
||||
"id": result["id"],
|
||||
"name": result.get("name", ""),
|
||||
"webViewLink": result.get("webViewLink", ""),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_share(args):
|
||||
permission = {
|
||||
"type": args.type,
|
||||
"role": args.role,
|
||||
}
|
||||
if args.type in ("user", "group"):
|
||||
if not args.email:
|
||||
print("ERROR: --email is required for type=user or type=group", file=sys.stderr)
|
||||
sys.exit(1)
|
||||
permission["emailAddress"] = args.email
|
||||
elif args.type == "domain":
|
||||
if not args.domain:
|
||||
print("ERROR: --domain is required for type=domain", file=sys.stderr)
|
||||
sys.exit(1)
|
||||
permission["domain"] = args.domain
|
||||
|
||||
if _gws_binary():
|
||||
result = _run_gws(
|
||||
["drive", "permissions", "create"],
|
||||
params={
|
||||
"fileId": args.file_id,
|
||||
"sendNotificationEmail": args.notify,
|
||||
},
|
||||
body=permission,
|
||||
)
|
||||
print(json.dumps({
|
||||
"status": "shared",
|
||||
"permissionId": result.get("id", ""),
|
||||
"fileId": args.file_id,
|
||||
"role": permission["role"],
|
||||
"type": permission["type"],
|
||||
}, indent=2, ensure_ascii=False))
|
||||
return
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
result = service.permissions().create(
|
||||
fileId=args.file_id,
|
||||
body=permission,
|
||||
sendNotificationEmail=args.notify,
|
||||
fields="id",
|
||||
).execute()
|
||||
print(json.dumps({
|
||||
"status": "shared",
|
||||
"permissionId": result.get("id", ""),
|
||||
"fileId": args.file_id,
|
||||
"role": permission["role"],
|
||||
"type": permission["type"],
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def drive_delete(args):
|
||||
"""Trash or permanently delete a Drive file. Defaults to trash (reversible)."""
|
||||
if args.permanent:
|
||||
if _gws_binary():
|
||||
_run_gws(["drive", "files", "delete"], params={"fileId": args.file_id})
|
||||
print(json.dumps({"status": "deleted", "fileId": args.file_id, "permanent": True}))
|
||||
return
|
||||
service = build_service("drive", "v3")
|
||||
service.files().delete(fileId=args.file_id).execute()
|
||||
print(json.dumps({"status": "deleted", "fileId": args.file_id, "permanent": True}))
|
||||
return
|
||||
|
||||
# Trash (reversible). Use files.update with trashed=True.
|
||||
body = {"trashed": True}
|
||||
if _gws_binary():
|
||||
_run_gws(
|
||||
["drive", "files", "update"],
|
||||
params={"fileId": args.file_id},
|
||||
body=body,
|
||||
)
|
||||
print(json.dumps({"status": "trashed", "fileId": args.file_id, "permanent": False}))
|
||||
return
|
||||
|
||||
service = build_service("drive", "v3")
|
||||
service.files().update(fileId=args.file_id, body=body).execute()
|
||||
print(json.dumps({"status": "trashed", "fileId": args.file_id, "permanent": False}))
|
||||
|
||||
|
||||
# =========================================================================
|
||||
# Contacts
|
||||
# =========================================================================
|
||||
|
|
@ -708,6 +915,34 @@ def sheets_append(args):
|
|||
print(json.dumps({"updatedCells": result.get("updates", {}).get("updatedCells", 0)}, indent=2))
|
||||
|
||||
|
||||
def sheets_create(args):
|
||||
"""Create a new spreadsheet. Returns the new spreadsheet ID and URL."""
|
||||
body = {"properties": {"title": args.title}}
|
||||
if args.sheet_name:
|
||||
body["sheets"] = [{"properties": {"title": args.sheet_name}}]
|
||||
|
||||
if _gws_binary():
|
||||
result = _run_gws(["sheets", "spreadsheets", "create"], body=body)
|
||||
print(json.dumps({
|
||||
"status": "created",
|
||||
"spreadsheetId": result.get("spreadsheetId", ""),
|
||||
"title": result.get("properties", {}).get("title", ""),
|
||||
"spreadsheetUrl": result.get("spreadsheetUrl", ""),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
return
|
||||
|
||||
service = build_service("sheets", "v4")
|
||||
result = service.spreadsheets().create(
|
||||
body=body, fields="spreadsheetId,properties,spreadsheetUrl",
|
||||
).execute()
|
||||
print(json.dumps({
|
||||
"status": "created",
|
||||
"spreadsheetId": result.get("spreadsheetId", ""),
|
||||
"title": result.get("properties", {}).get("title", ""),
|
||||
"spreadsheetUrl": result.get("spreadsheetUrl", ""),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
# =========================================================================
|
||||
# Docs
|
||||
# =========================================================================
|
||||
|
|
@ -734,6 +969,79 @@ def docs_get(args):
|
|||
print(json.dumps(result, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def docs_create(args):
|
||||
"""Create a new Doc. Optionally seed it with initial body text."""
|
||||
body = {"title": args.title}
|
||||
|
||||
if _gws_binary():
|
||||
doc = _run_gws(["docs", "documents", "create"], body=body)
|
||||
else:
|
||||
service = build_service("docs", "v1")
|
||||
doc = service.documents().create(body=body).execute()
|
||||
|
||||
doc_id = doc.get("documentId", "")
|
||||
|
||||
if args.body and doc_id:
|
||||
_docs_insert_text(doc_id, args.body, index=1)
|
||||
|
||||
print(json.dumps({
|
||||
"status": "created",
|
||||
"documentId": doc_id,
|
||||
"title": doc.get("title", ""),
|
||||
"url": f"https://docs.google.com/document/d/{doc_id}/edit" if doc_id else "",
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def docs_append(args):
|
||||
"""Append text to the end of an existing Doc."""
|
||||
if _gws_binary():
|
||||
doc = _run_gws(["docs", "documents", "get"], params={"documentId": args.doc_id})
|
||||
else:
|
||||
service = build_service("docs", "v1")
|
||||
doc = service.documents().get(documentId=args.doc_id).execute()
|
||||
|
||||
# The end-of-body index is one less than the segment endIndex of the body
|
||||
# (trailing newline is always at length-1). Docs indexes are 1-based; use
|
||||
# endIndex - 1 to insert before the final newline.
|
||||
content = doc.get("body", {}).get("content", [])
|
||||
end_index = 1
|
||||
for element in content:
|
||||
ei = element.get("endIndex")
|
||||
if isinstance(ei, int) and ei > end_index:
|
||||
end_index = ei
|
||||
insert_index = max(end_index - 1, 1)
|
||||
|
||||
text = args.text if args.text.endswith("\n") else args.text + "\n"
|
||||
_docs_insert_text(args.doc_id, text, index=insert_index)
|
||||
|
||||
print(json.dumps({
|
||||
"status": "appended",
|
||||
"documentId": args.doc_id,
|
||||
"inserted_at": insert_index,
|
||||
"characters": len(text),
|
||||
}, indent=2, ensure_ascii=False))
|
||||
|
||||
|
||||
def _docs_insert_text(doc_id: str, text: str, index: int) -> None:
|
||||
"""Send a batchUpdate with a single insertText request."""
|
||||
requests = [{
|
||||
"insertText": {
|
||||
"location": {"index": index},
|
||||
"text": text,
|
||||
}
|
||||
}]
|
||||
if _gws_binary():
|
||||
_run_gws(
|
||||
["docs", "documents", "batchUpdate"],
|
||||
params={"documentId": doc_id},
|
||||
body={"requests": requests},
|
||||
)
|
||||
return
|
||||
|
||||
service = build_service("docs", "v1")
|
||||
service.documents().batchUpdate(documentId=doc_id, body={"requests": requests}).execute()
|
||||
|
||||
|
||||
# =========================================================================
|
||||
# CLI parser
|
||||
# =========================================================================
|
||||
|
|
@ -817,6 +1125,42 @@ def main():
|
|||
p.add_argument("--raw-query", action="store_true", help="Use query as raw Drive API query")
|
||||
p.set_defaults(func=drive_search)
|
||||
|
||||
p = drv_sub.add_parser("get")
|
||||
p.add_argument("file_id")
|
||||
p.set_defaults(func=drive_get)
|
||||
|
||||
p = drv_sub.add_parser("upload")
|
||||
p.add_argument("path", help="Local file path to upload")
|
||||
p.add_argument("--name", default="", help="Override file name in Drive (defaults to local filename)")
|
||||
p.add_argument("--parent", default="", help="Parent folder ID")
|
||||
p.add_argument("--mime-type", default="", help="Override MIME type (auto-detected if omitted)")
|
||||
p.set_defaults(func=drive_upload)
|
||||
|
||||
p = drv_sub.add_parser("download")
|
||||
p.add_argument("file_id")
|
||||
p.add_argument("--output", default="", help="Local output path (defaults to ./<name> in cwd)")
|
||||
p.add_argument("--export-mime", default="", help="Export MIME for Google-native files (overrides defaults: pdf for Docs/Slides, csv for Sheets, png for Drawings)")
|
||||
p.set_defaults(func=drive_download)
|
||||
|
||||
p = drv_sub.add_parser("create-folder")
|
||||
p.add_argument("name")
|
||||
p.add_argument("--parent", default="", help="Parent folder ID (defaults to root)")
|
||||
p.set_defaults(func=drive_create_folder)
|
||||
|
||||
p = drv_sub.add_parser("share")
|
||||
p.add_argument("file_id")
|
||||
p.add_argument("--role", default="reader", choices=["reader", "commenter", "writer", "fileOrganizer", "organizer", "owner"])
|
||||
p.add_argument("--type", default="user", choices=["user", "group", "domain", "anyone"])
|
||||
p.add_argument("--email", default="", help="Email address (required for type=user or type=group)")
|
||||
p.add_argument("--domain", default="", help="Domain (required for type=domain)")
|
||||
p.add_argument("--notify", action="store_true", help="Send notification email")
|
||||
p.set_defaults(func=drive_share)
|
||||
|
||||
p = drv_sub.add_parser("delete")
|
||||
p.add_argument("file_id")
|
||||
p.add_argument("--permanent", action="store_true", help="Permanently delete (default is trash, which is reversible)")
|
||||
p.set_defaults(func=drive_delete)
|
||||
|
||||
# --- Contacts ---
|
||||
con = sub.add_parser("contacts")
|
||||
con_sub = con.add_subparsers(dest="action", required=True)
|
||||
|
|
@ -846,6 +1190,11 @@ def main():
|
|||
p.add_argument("--values", required=True, help="JSON array of arrays")
|
||||
p.set_defaults(func=sheets_append)
|
||||
|
||||
p = sh_sub.add_parser("create")
|
||||
p.add_argument("--title", required=True, help="Spreadsheet title")
|
||||
p.add_argument("--sheet-name", default="", help="Name of the first tab (defaults to 'Sheet1')")
|
||||
p.set_defaults(func=sheets_create)
|
||||
|
||||
# --- Docs ---
|
||||
docs = sub.add_parser("docs")
|
||||
docs_sub = docs.add_subparsers(dest="action", required=True)
|
||||
|
|
@ -854,6 +1203,16 @@ def main():
|
|||
p.add_argument("doc_id")
|
||||
p.set_defaults(func=docs_get)
|
||||
|
||||
p = docs_sub.add_parser("create")
|
||||
p.add_argument("--title", required=True, help="Document title")
|
||||
p.add_argument("--body", default="", help="Initial body text (optional)")
|
||||
p.set_defaults(func=docs_create)
|
||||
|
||||
p = docs_sub.add_parser("append")
|
||||
p.add_argument("doc_id")
|
||||
p.add_argument("--text", required=True, help="Text to append to the end of the document")
|
||||
p.set_defaults(func=docs_append)
|
||||
|
||||
args = parser.parse_args()
|
||||
args.func(args)
|
||||
|
||||
|
|
|
|||
|
|
@ -47,10 +47,10 @@ SCOPES = [
|
|||
"https://www.googleapis.com/auth/gmail.send",
|
||||
"https://www.googleapis.com/auth/gmail.modify",
|
||||
"https://www.googleapis.com/auth/calendar",
|
||||
"https://www.googleapis.com/auth/drive.readonly",
|
||||
"https://www.googleapis.com/auth/drive",
|
||||
"https://www.googleapis.com/auth/contacts.readonly",
|
||||
"https://www.googleapis.com/auth/spreadsheets",
|
||||
"https://www.googleapis.com/auth/documents.readonly",
|
||||
"https://www.googleapis.com/auth/documents",
|
||||
]
|
||||
|
||||
REQUIRED_PACKAGES = ["google-api-python-client", "google-auth-oauthlib", "google-auth-httplib2"]
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue