classZapierToolSpec(BaseToolSpec):"""Zapier tool spec."""spec_functions=[]def__init__(self,api_key:Optional[str]=None,oauth_access_token:Optional[str]=None)->None:"""Initialize with parameters."""ifapi_key:self._headers={"x-api-key":api_key}elifoauth_access_token:self._headers={"Authorization":f"Bearer {oauth_access_token}"}else:raiseValueError("Must provide either api_key or oauth_access_token")# Get the exposed actions from Zapieractions=json.loads(self.list_actions())if"results"notinactions:raiseValueError("No Zapier actions exposed, visit https://nla.zapier.com/dev/actions/"" to expose actions.")results=actions["results"]# Register the actions as Toolsforactioninresults:params=action["params"]deffunction_action(id=action["id"],**kwargs):returnself.natural_language_query(id,**kwargs)action_name=action["description"].split(": ")[1].replace(" ","_")function_action.__name__=action_namefunction_action.__doc__=f""" This is a Zapier Natural Language Action function wrapper. The 'instructions' key is REQUIRED for all function calls. The instructions key is a natural language string describing the action to be taken The following are all of the valid arguments you can provide: {params} Ignore the id field, it is provided for you. If the returned error field is not null, interpret the error and try to fix it. Otherwise, inform the user of how they might fix it. """setattr(self,action_name,function_action)self.spec_functions.append(action_name)deflist_actions(self):response=requests.get("https://nla.zapier.com/api/v1/dynamic/exposed/",headers=self._headers)returnresponse.textdefnatural_language_query(self,id:str,**kwargs):response=requests.post(ACTION_URL_TMPL.format(action_id=id),headers=self._headers,data=json.dumps(kwargs),)returnresponse.text