# PR Description ## Split toolkits This PR splits the `Microsoft`, `Google`, and `Search` toolkits into multiple toolkits each. * `Microsoft` --> `OutlookCalendar`, `OutlookMail`. * `Google` -----> `GoogleCalendar`, `GoogleContacts`, `GoogleDocs`, `GoogleDrive`, `Gmail`, `GoogleSheets` * `Search` -----> `GoogleFinance`, `GoogleFlights`, `GoogleHotels`, `GoogleJobs`, `GoogleMaps`, `GoogleNews`, `GoogleSearch`, `GoogleShopping`, `Walmart`, `Youtube` > The original monolithic toolkits (`Microsoft`, `Google`, `Search`) are not removed in this PR. The plan is to keep those toolkits around while we > 1. Stop documenting the toolkits, > 2. Stop displaying the toolkits in the dashboard, and > 3. Help customers migrate over to the new split toolkits. ## Rename toolkits This PR renames the following toolkits * `Web` ------------> `Firecrawl` * `CodeSandbox` ---> `E2B` > The `Web` and `CodeSandbox` toolkits are not removed in this PR. The plan is to keep them around while we > 1. Stop documenting the toolkits, > 2. Stop displaying the toolkits in the dashboard, and > 3. Help customers migrate over to the new renamed toolkits. ## Rename tools Since toolkit names were changed, this called for some tools to be renamed as well. * `GoogleSearch.SearchGoogle` ----------------> `GoogleSearch.Search` * `GoogleShopping.SearchShoppingProducts` ---> `GoogleShopping.SearchProducts` * `Walmart.SearchWalmartProducts` ------------> `Walmart.SearchProducts` * `Walmart.GetWalmartProductDetails` ---------> `Walmart.GetProductDetails` * `Youtube.SearchYoutubeVideos` --------------> `Youtube.SearchForVideos` ## Google File Picker Improvements to the Google File Picker experience were also added in this PR. The following tools will ALWAYS provide llm_instructions in their response to "let the end-user know that they have the option to select more files via the file picker url if they want to": * `GoogleDocs.SearchDocuments` * `GoogleDocs.SearchAndRetrieveDocuments` * `GoogleDrive.GetFileTreeStructure` The following tools will only provide the file picker URL if a 404 or 403 from the Google API: * `GoogleDocs.InsertTextAtEndOfDocument` * `GoogleDocs.GetDocumentById` * `GoogleSheets.GetSpreadsheet` * `GoogleSheets.WriteToCell` Also, a standalone `GoogleDrive.GenerateGoogleFilePickerUrl` tool exists. ## Other * The `SearchDocuments` and `SearchAndRetrieveDocuments` tools used to be organized within the Drive portion of the Google toolkit, but I moved these into the new GoogleDocs toolkit because they are specific to Docs. # Progress - [x] `OutlookCalendar` - [x] `OutlookMail` - [x] `GoogleFinance` - [x] `GoogleFlights` - [x] `GoogleHotels` - [x] `GoogleJobs` - [x] `GoogleMaps` - [x] `GoogleNews` - [x] `GoogleSearch` - [x] `GoogleShopping` - [x] `Walmart` - [x] `Youtube` - [x] `GoogleCalendar` - [x] `GoogleContacts` - [x] `GoogleDocs` - [x] `GoogleDrive` - [x] `Gmail` - [x] `GoogleSheets` - [x] `Firecrawl` - [x] `E2B` - [x] File picker # Discussion * Repeated code is a consequence of splitting toolkits that use the same provider. I am open to any ideas that would allow multiple toolkits to reference common code. Comment your ideas in this PR.
82 lines
2.7 KiB
Python
82 lines
2.7 KiB
Python
from typing import Annotated
|
|
|
|
from arcade_tdk import ToolContext, tool
|
|
from arcade_tdk.auth import Google
|
|
|
|
from arcade_google_docs.utils import build_docs_service
|
|
|
|
|
|
# Uses https://developers.google.com/docs/api/reference/rest/v1/documents/create
|
|
# Example `arcade chat` query: `create blank document with title "My New Document"`
|
|
@tool(
|
|
requires_auth=Google(
|
|
scopes=[
|
|
"https://www.googleapis.com/auth/drive.file",
|
|
],
|
|
)
|
|
)
|
|
async def create_blank_document(
|
|
context: ToolContext, title: Annotated[str, "The title of the blank document to create"]
|
|
) -> Annotated[dict, "The created document's title, documentId, and documentUrl in a dictionary"]:
|
|
"""
|
|
Create a blank Google Docs document with the specified title.
|
|
"""
|
|
service = build_docs_service(context.get_auth_token_or_empty())
|
|
|
|
body = {"title": title}
|
|
|
|
# Execute the documents().create() method. Returns a Document object https://developers.google.com/docs/api/reference/rest/v1/documents#Document
|
|
request = service.documents().create(body=body)
|
|
response = request.execute()
|
|
|
|
return {
|
|
"title": response["title"],
|
|
"documentId": response["documentId"],
|
|
"documentUrl": f"https://docs.google.com/document/d/{response['documentId']}/edit",
|
|
}
|
|
|
|
|
|
# Uses https://developers.google.com/docs/api/reference/rest/v1/documents/batchUpdate
|
|
# Example `arcade chat` query:
|
|
# `create document with title "My New Document" and text content "Hello, World!"`
|
|
@tool(
|
|
requires_auth=Google(
|
|
scopes=[
|
|
"https://www.googleapis.com/auth/drive.file",
|
|
],
|
|
)
|
|
)
|
|
async def create_document_from_text(
|
|
context: ToolContext,
|
|
title: Annotated[str, "The title of the document to create"],
|
|
text_content: Annotated[str, "The text content to insert into the document"],
|
|
) -> Annotated[dict, "The created document's title, documentId, and documentUrl in a dictionary"]:
|
|
"""
|
|
Create a Google Docs document with the specified title and text content.
|
|
"""
|
|
# First, create a blank document
|
|
document = await create_blank_document(context, title)
|
|
|
|
service = build_docs_service(context.get_auth_token_or_empty())
|
|
|
|
requests = [
|
|
{
|
|
"insertText": {
|
|
"location": {
|
|
"index": 1,
|
|
},
|
|
"text": text_content,
|
|
}
|
|
}
|
|
]
|
|
|
|
# Execute the batchUpdate method to insert text
|
|
service.documents().batchUpdate(
|
|
documentId=document["documentId"], body={"requests": requests}
|
|
).execute()
|
|
|
|
return {
|
|
"title": document["title"],
|
|
"documentId": document["documentId"],
|
|
"documentUrl": f"https://docs.google.com/document/d/{document['documentId']}/edit",
|
|
}
|