| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  | import re | 
					
						
							|  |  |  | import uuid | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  | from json import dumps as json_dumps | 
					
						
							| 
									
										
										
										
											2024-02-01 18:11:57 +08:00
										 |  |  | from json import loads as json_loads | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  | from json.decoder import JSONDecodeError | 
					
						
							| 
									
										
										
										
											2024-10-09 14:36:43 +08:00
										 |  |  | from typing import Optional | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-13 17:49:30 +08:00
										 |  |  | from flask import request | 
					
						
							| 
									
										
										
										
											2024-02-06 13:21:13 +08:00
										 |  |  | from requests import get | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  | from yaml import YAMLError, safe_load  # type: ignore | 
					
						
							| 
									
										
										
										
											2024-02-06 13:21:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-01 18:11:57 +08:00
										 |  |  | from core.tools.entities.common_entities import I18nObject | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  | from core.tools.entities.tool_bundle import ApiToolBundle | 
					
						
							| 
									
										
										
										
											2024-02-06 13:21:13 +08:00
										 |  |  | from core.tools.entities.tool_entities import ApiProviderSchemaType, ToolParameter | 
					
						
							| 
									
										
										
										
											2024-02-01 18:11:57 +08:00
										 |  |  | from core.tools.errors import ToolApiSchemaError, ToolNotSupportedError, ToolProviderNotFoundError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ApiBasedToolSchemaParser: | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     def parse_openapi_to_tool_bundle( | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         openapi: dict, extra_info: dict | None = None, warning: dict | None = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     ) -> list[ApiToolBundle]: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         warning = warning if warning is not None else {} | 
					
						
							|  |  |  |         extra_info = extra_info if extra_info is not None else {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # set description to extra_info | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         extra_info["description"] = openapi["info"].get("description", "") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         if len(openapi["servers"]) == 0: | 
					
						
							|  |  |  |             raise ToolProviderNotFoundError("No server found in the openapi yaml.") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         server_url = openapi["servers"][0]["url"] | 
					
						
							| 
									
										
										
										
											2025-01-13 17:49:30 +08:00
										 |  |  |         request_env = request.headers.get("X-Request-Env") | 
					
						
							|  |  |  |         if request_env: | 
					
						
							|  |  |  |             matched_servers = [server["url"] for server in openapi["servers"] if server["env"] == request_env] | 
					
						
							|  |  |  |             server_url = matched_servers[0] if matched_servers else server_url | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # list all interfaces | 
					
						
							|  |  |  |         interfaces = [] | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         for path, path_item in openapi["paths"].items(): | 
					
						
							|  |  |  |             methods = ["get", "post", "put", "delete", "patch", "head", "options", "trace"] | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |             for method in methods: | 
					
						
							|  |  |  |                 if method in path_item: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                     interfaces.append( | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             "path": path, | 
					
						
							|  |  |  |                             "method": method, | 
					
						
							|  |  |  |                             "operation": path_item[method], | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # get all parameters | 
					
						
							|  |  |  |         bundles = [] | 
					
						
							|  |  |  |         for interface in interfaces: | 
					
						
							|  |  |  |             # convert parameters | 
					
						
							|  |  |  |             parameters = [] | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             if "parameters" in interface["operation"]: | 
					
						
							| 
									
										
										
										
											2025-06-02 18:08:53 +08:00
										 |  |  |                 for i, parameter in enumerate(interface["operation"]["parameters"]): | 
					
						
							|  |  |  |                     if "$ref" in parameter: | 
					
						
							|  |  |  |                         root = openapi | 
					
						
							|  |  |  |                         reference = parameter["$ref"].split("/")[1:] | 
					
						
							|  |  |  |                         for ref in reference: | 
					
						
							|  |  |  |                             root = root[ref] | 
					
						
							|  |  |  |                         interface["operation"]["parameters"][i] = root | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                 for parameter in interface["operation"]["parameters"]: | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                     tool_parameter = ToolParameter( | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                         name=parameter["name"], | 
					
						
							|  |  |  |                         label=I18nObject(en_US=parameter["name"], zh_Hans=parameter["name"]), | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                         human_description=I18nObject( | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                             en_US=parameter.get("description", ""), zh_Hans=parameter.get("description", "") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                         ), | 
					
						
							| 
									
										
										
										
											2024-01-31 11:58:07 +08:00
										 |  |  |                         type=ToolParameter.ToolParameterType.STRING, | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                         required=parameter.get("required", False), | 
					
						
							| 
									
										
										
										
											2024-01-31 11:58:07 +08:00
										 |  |  |                         form=ToolParameter.ToolParameterForm.LLM, | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                         llm_description=parameter.get("description"), | 
					
						
							|  |  |  |                         default=parameter["schema"]["default"] | 
					
						
							|  |  |  |                         if "schema" in parameter and "default" in parameter["schema"] | 
					
						
							|  |  |  |                         else None, | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |                         placeholder=I18nObject( | 
					
						
							|  |  |  |                             en_US=parameter.get("description", ""), zh_Hans=parameter.get("description", "") | 
					
						
							|  |  |  |                         ), | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                     # check if there is a type | 
					
						
							|  |  |  |                     typ = ApiBasedToolSchemaParser._get_tool_parameter_type(parameter) | 
					
						
							|  |  |  |                     if typ: | 
					
						
							|  |  |  |                         tool_parameter.type = typ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     parameters.append(tool_parameter) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |             # create tool bundle | 
					
						
							|  |  |  |             # check if there is a request body | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             if "requestBody" in interface["operation"]: | 
					
						
							|  |  |  |                 request_body = interface["operation"]["requestBody"] | 
					
						
							|  |  |  |                 if "content" in request_body: | 
					
						
							|  |  |  |                     for content_type, content in request_body["content"].items(): | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                         # if there is a reference, get the reference and overwrite the content | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                         if "schema" not in content: | 
					
						
							| 
									
										
										
										
											2024-04-19 16:51:38 +08:00
										 |  |  |                             continue | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                         if "$ref" in content["schema"]: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                             # get the reference | 
					
						
							|  |  |  |                             root = openapi | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                             reference = content["schema"]["$ref"].split("/")[1:] | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                             for ref in reference: | 
					
						
							|  |  |  |                                 root = root[ref] | 
					
						
							|  |  |  |                             # overwrite the content | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                             interface["operation"]["requestBody"]["content"][content_type]["schema"] = root | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                     # parse body parameters | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                     if "schema" in interface["operation"]["requestBody"]["content"][content_type]: | 
					
						
							|  |  |  |                         body_schema = interface["operation"]["requestBody"]["content"][content_type]["schema"] | 
					
						
							|  |  |  |                         required = body_schema.get("required", []) | 
					
						
							|  |  |  |                         properties = body_schema.get("properties", {}) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                         for name, property in properties.items(): | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                             tool = ToolParameter( | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                                 name=name, | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                                 label=I18nObject(en_US=name, zh_Hans=name), | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                                 human_description=I18nObject( | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                                     en_US=property.get("description", ""), zh_Hans=property.get("description", "") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                                 ), | 
					
						
							| 
									
										
										
										
											2024-01-31 11:58:07 +08:00
										 |  |  |                                 type=ToolParameter.ToolParameterType.STRING, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                                 required=name in required, | 
					
						
							| 
									
										
										
										
											2024-01-31 11:58:07 +08:00
										 |  |  |                                 form=ToolParameter.ToolParameterForm.LLM, | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                                 llm_description=property.get("description", ""), | 
					
						
							|  |  |  |                                 default=property.get("default", None), | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |                                 placeholder=I18nObject( | 
					
						
							| 
									
										
										
										
											2025-01-09 10:30:43 +03:00
										 |  |  |                                     en_US=property.get("description", ""), zh_Hans=property.get("description", "") | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |                                 ), | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             # check if there is a type | 
					
						
							|  |  |  |                             typ = ApiBasedToolSchemaParser._get_tool_parameter_type(property) | 
					
						
							|  |  |  |                             if typ: | 
					
						
							|  |  |  |                                 tool.type = typ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             parameters.append(tool) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # check if parameters is duplicated | 
					
						
							|  |  |  |             parameters_count = {} | 
					
						
							|  |  |  |             for parameter in parameters: | 
					
						
							|  |  |  |                 if parameter.name not in parameters_count: | 
					
						
							|  |  |  |                     parameters_count[parameter.name] = 0 | 
					
						
							|  |  |  |                 parameters_count[parameter.name] += 1 | 
					
						
							|  |  |  |             for name, count in parameters_count.items(): | 
					
						
							|  |  |  |                 if count > 1: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                     warning["duplicated_parameter"] = f"Parameter {name} is duplicated." | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-30 22:22:58 +08:00
										 |  |  |             # check if there is a operation id, use $path_$method as operation id if not | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             if "operationId" not in interface["operation"]: | 
					
						
							| 
									
										
										
										
											2024-02-20 14:50:57 +08:00
										 |  |  |                 # remove special characters like / to ensure the operation id is valid ^[a-zA-Z0-9_-]{1,64}$ | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                 path = interface["path"] | 
					
						
							|  |  |  |                 if interface["path"].startswith("/"): | 
					
						
							|  |  |  |                     path = interface["path"][1:] | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                 # remove special characters like / to ensure the operation id is valid ^[a-zA-Z0-9_-]{1,64}$ | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                 path = re.sub(r"[^a-zA-Z0-9_-]", "", path) | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |                 if not path: | 
					
						
							|  |  |  |                     path = str(uuid.uuid4()) | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-21 10:12:29 +08:00
										 |  |  |                 interface["operation"]["operationId"] = f"{path}_{interface['method']}" | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             bundles.append( | 
					
						
							|  |  |  |                 ApiToolBundle( | 
					
						
							|  |  |  |                     server_url=server_url + interface["path"], | 
					
						
							|  |  |  |                     method=interface["method"], | 
					
						
							|  |  |  |                     summary=interface["operation"]["description"] | 
					
						
							|  |  |  |                     if "description" in interface["operation"] | 
					
						
							|  |  |  |                     else interface["operation"].get("summary", None), | 
					
						
							|  |  |  |                     operation_id=interface["operation"]["operationId"], | 
					
						
							|  |  |  |                     parameters=parameters, | 
					
						
							|  |  |  |                     author="", | 
					
						
							|  |  |  |                     icon=None, | 
					
						
							|  |  |  |                     openapi=interface["operation"], | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return bundles | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |     def _get_tool_parameter_type(parameter: dict) -> Optional[ToolParameter.ToolParameterType]: | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |         parameter = parameter or {} | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |         typ: Optional[str] = None | 
					
						
							| 
									
										
										
										
											2024-11-20 13:26:42 +08:00
										 |  |  |         if parameter.get("format") == "binary": | 
					
						
							|  |  |  |             return ToolParameter.ToolParameterType.FILE | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         if "type" in parameter: | 
					
						
							|  |  |  |             typ = parameter["type"] | 
					
						
							|  |  |  |         elif "schema" in parameter and "type" in parameter["schema"]: | 
					
						
							|  |  |  |             typ = parameter["schema"]["type"] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-13 22:42:08 +08:00
										 |  |  |         if typ in {"integer", "number"}: | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |             return ToolParameter.ToolParameterType.NUMBER | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         elif typ == "boolean": | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |             return ToolParameter.ToolParameterType.BOOLEAN | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         elif typ == "string": | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  |             return ToolParameter.ToolParameterType.STRING | 
					
						
							| 
									
										
										
										
											2025-03-27 11:47:35 +09:00
										 |  |  |         elif typ == "array": | 
					
						
							|  |  |  |             items = parameter.get("items") or parameter.get("schema", {}).get("items") | 
					
						
							|  |  |  |             return ToolParameter.ToolParameterType.FILES if items and items.get("format") == "binary" else None | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return None | 
					
						
							| 
									
										
										
										
											2024-02-27 15:59:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     def parse_openapi_yaml_to_tool_bundle( | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         yaml: str, extra_info: dict | None = None, warning: dict | None = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     ) -> list[ApiToolBundle]: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         parse openapi yaml to tool bundle | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :param yaml: the yaml string | 
					
						
							| 
									
										
										
										
											2025-03-31 13:19:15 +08:00
										 |  |  |         :param extra_info: the extra info | 
					
						
							|  |  |  |         :param warning: the warning message | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :return: the tool bundle | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         warning = warning if warning is not None else {} | 
					
						
							|  |  |  |         extra_info = extra_info if extra_info is not None else {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         openapi: dict = safe_load(yaml) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         if openapi is None: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             raise ToolApiSchemaError("Invalid openapi yaml.") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         return ApiBasedToolSchemaParser.parse_openapi_to_tool_bundle(openapi, extra_info=extra_info, warning=warning) | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |     def parse_swagger_to_openapi(swagger: dict, extra_info: dict | None = None, warning: dict | None = None) -> dict: | 
					
						
							|  |  |  |         warning = warning or {} | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         parse swagger to openapi | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :param swagger: the swagger dict | 
					
						
							|  |  |  |         :return: the openapi dict | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         # convert swagger to openapi | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         info = swagger.get("info", {"title": "Swagger", "description": "Swagger", "version": "1.0.0"}) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         servers = swagger.get("servers", []) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if len(servers) == 0: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             raise ToolApiSchemaError("No server found in the swagger yaml.") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         openapi = { | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             "openapi": "3.0.0", | 
					
						
							|  |  |  |             "info": { | 
					
						
							|  |  |  |                 "title": info.get("title", "Swagger"), | 
					
						
							|  |  |  |                 "description": info.get("description", "Swagger"), | 
					
						
							|  |  |  |                 "version": info.get("version", "1.0.0"), | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |             }, | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             "servers": swagger["servers"], | 
					
						
							|  |  |  |             "paths": {}, | 
					
						
							|  |  |  |             "components": {"schemas": {}}, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # check paths | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         if "paths" not in swagger or len(swagger["paths"]) == 0: | 
					
						
							|  |  |  |             raise ToolApiSchemaError("No paths found in the swagger yaml.") | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # convert paths | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         for path, path_item in swagger["paths"].items(): | 
					
						
							|  |  |  |             openapi["paths"][path] = {} | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |             for method, operation in path_item.items(): | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                 if "operationId" not in operation: | 
					
						
							|  |  |  |                     raise ToolApiSchemaError(f"No operationId found in operation {method} {path}.") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if ("summary" not in operation or len(operation["summary"]) == 0) and ( | 
					
						
							|  |  |  |                     "description" not in operation or len(operation["description"]) == 0 | 
					
						
							|  |  |  |                 ): | 
					
						
							| 
									
										
										
										
											2024-12-24 18:38:51 +08:00
										 |  |  |                     if warning is not None: | 
					
						
							|  |  |  |                         warning["missing_summary"] = f"No summary or description found in operation {method} {path}." | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 openapi["paths"][path][method] = { | 
					
						
							|  |  |  |                     "operationId": operation["operationId"], | 
					
						
							|  |  |  |                     "summary": operation.get("summary", ""), | 
					
						
							|  |  |  |                     "description": operation.get("description", ""), | 
					
						
							|  |  |  |                     "parameters": operation.get("parameters", []), | 
					
						
							|  |  |  |                     "responses": operation.get("responses", {}), | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |                 if "requestBody" in operation: | 
					
						
							|  |  |  |                     openapi["paths"][path][method]["requestBody"] = operation["requestBody"] | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # convert definitions | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         for name, definition in swagger["definitions"].items(): | 
					
						
							|  |  |  |             openapi["components"]["schemas"][name] = definition | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return openapi | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     def parse_openai_plugin_json_to_tool_bundle( | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         json: str, extra_info: dict | None = None, warning: dict | None = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     ) -> list[ApiToolBundle]: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         parse openapi plugin yaml to tool bundle | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :param json: the json string | 
					
						
							| 
									
										
										
										
											2025-03-31 13:19:15 +08:00
										 |  |  |         :param extra_info: the extra info | 
					
						
							|  |  |  |         :param warning: the warning message | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :return: the tool bundle | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         warning = warning if warning is not None else {} | 
					
						
							|  |  |  |         extra_info = extra_info if extra_info is not None else {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             openai_plugin = json_loads(json) | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             api = openai_plugin["api"] | 
					
						
							|  |  |  |             api_url = api["url"] | 
					
						
							|  |  |  |             api_type = api["type"] | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         except JSONDecodeError: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             raise ToolProviderNotFoundError("Invalid openai plugin json.") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if api_type != "openapi": | 
					
						
							|  |  |  |             raise ToolNotSupportedError("Only openapi is supported now.") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         # get openapi yaml | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         response = get(api_url, headers={"User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) "}, timeout=5) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if response.status_code != 200: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             raise ToolProviderNotFoundError("cannot get openapi yaml from url.") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ApiBasedToolSchemaParser.parse_openapi_yaml_to_tool_bundle( | 
					
						
							|  |  |  |             response.text, extra_info=extra_info, warning=warning | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     def auto_parse_to_tool_bundle( | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         content: str, extra_info: dict | None = None, warning: dict | None = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |     ) -> tuple[list[ApiToolBundle], str]: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         auto parse to tool bundle | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :param content: the content | 
					
						
							| 
									
										
										
										
											2025-03-31 13:19:15 +08:00
										 |  |  |         :param extra_info: the extra info | 
					
						
							|  |  |  |         :param warning: the warning message | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         :return: tools bundle, schema_type | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         warning = warning if warning is not None else {} | 
					
						
							|  |  |  |         extra_info = extra_info if extra_info is not None else {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         content = content.strip() | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         loaded_content = None | 
					
						
							|  |  |  |         json_error = None | 
					
						
							|  |  |  |         yaml_error = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             loaded_content = json_loads(content) | 
					
						
							|  |  |  |         except JSONDecodeError as e: | 
					
						
							|  |  |  |             json_error = e | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         if loaded_content is None: | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |                 loaded_content = safe_load(content) | 
					
						
							|  |  |  |             except YAMLError as e: | 
					
						
							|  |  |  |                 yaml_error = e | 
					
						
							|  |  |  |         if loaded_content is None: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             raise ToolApiSchemaError( | 
					
						
							| 
									
										
										
										
											2024-09-12 14:00:36 +08:00
										 |  |  |                 f"Invalid api schema, schema is neither json nor yaml. json error: {str(json_error)}," | 
					
						
							|  |  |  |                 f" yaml error: {str(yaml_error)}" | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         swagger_error = None | 
					
						
							|  |  |  |         openapi_error = None | 
					
						
							|  |  |  |         openapi_plugin_error = None | 
					
						
							|  |  |  |         schema_type = None | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             openapi = ApiBasedToolSchemaParser.parse_openapi_to_tool_bundle( | 
					
						
							|  |  |  |                 loaded_content, extra_info=extra_info, warning=warning | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |             schema_type = ApiProviderSchemaType.OPENAPI.value | 
					
						
							|  |  |  |             return openapi, schema_type | 
					
						
							|  |  |  |         except ToolApiSchemaError as e: | 
					
						
							|  |  |  |             openapi_error = e | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         # openai parse error, fallback to swagger | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             converted_swagger = ApiBasedToolSchemaParser.parse_swagger_to_openapi( | 
					
						
							|  |  |  |                 loaded_content, extra_info=extra_info, warning=warning | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |             schema_type = ApiProviderSchemaType.SWAGGER.value | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             return ApiBasedToolSchemaParser.parse_openapi_to_tool_bundle( | 
					
						
							|  |  |  |                 converted_swagger, extra_info=extra_info, warning=warning | 
					
						
							|  |  |  |             ), schema_type | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         except ToolApiSchemaError as e: | 
					
						
							|  |  |  |             swagger_error = e | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |         # swagger parse error, fallback to openai plugin | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |             openapi_plugin = ApiBasedToolSchemaParser.parse_openai_plugin_json_to_tool_bundle( | 
					
						
							|  |  |  |                 json_dumps(loaded_content), extra_info=extra_info, warning=warning | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-03-30 14:44:50 +08:00
										 |  |  |             return openapi_plugin, ApiProviderSchemaType.OPENAI_PLUGIN.value | 
					
						
							|  |  |  |         except ToolNotSupportedError as e: | 
					
						
							|  |  |  |             # maybe it's not plugin at all | 
					
						
							|  |  |  |             openapi_plugin_error = e | 
					
						
							| 
									
										
										
										
											2024-01-23 19:58:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         raise ToolApiSchemaError( | 
					
						
							| 
									
										
										
										
											2024-09-12 14:00:36 +08:00
										 |  |  |             f"Invalid api schema, openapi error: {str(openapi_error)}, swagger error: {str(swagger_error)}," | 
					
						
							|  |  |  |             f" openapi plugin error: {str(openapi_plugin_error)}" | 
					
						
							| 
									
										
										
										
											2024-09-10 17:00:20 +08:00
										 |  |  |         ) |