| from typing import List, Optional |
| import requests |
| import json |
| from mcp.server.fastmcp import FastMCP |
|
|
| mcp = FastMCP("organizedprogrammers-mcp-server") |
|
|
| @mcp.tool() |
| def search_arxiv_papers(keyword: str, limit: int = 5) -> str: |
| """ |
| Search papers from arXiv database with specified keywords [optional: a limit of papers the user wants] |
| Args: keyword: string, [optional: limit: integer, set limit to 5 if not specified] |
| """ |
| response = requests.post("https://om4r932-arxiv.hf.space/search", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps({ |
| "keyword": keyword, |
| "limit": limit |
| }), verify=False) |
|
|
| if response.status_code != 200: |
| return "Unable to find papers: error on post()" |
| |
| responseJson = response.json() |
|
|
| if responseJson.get("error") or not isinstance(responseJson['message'], dict): |
| return f"Unable to find papers: error on API -> {responseJson['message']}" |
|
|
| if len(responseJson["message"].keys()) == 0: |
| return "No papers has been found" |
| |
| return "\n".join([f"arXiv n°{paper_id} - {paper_meta['title']} by {paper_meta['authors']} : {paper_meta['abstract']}" for paper_id, paper_meta in responseJson['message'].items()]) |
|
|
| @mcp.tool() |
| def locate_3gpp_document(doc_id: str) -> str: |
| """ |
| Find 3GPP document location with the document's ID |
| Args: doc_id: string |
| """ |
| response = requests.post("https://organizedprogrammers-3gppdocfinder.hf.space/find", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps({ |
| "doc_id": doc_id |
| }), verify=False) |
|
|
| if response.status_code != 200: |
| return f"Unable to find document: {response.status_code} - {response.content}" |
| |
| responseJson = response.json() |
|
|
| if responseJson.get("detail"): |
| return responseJson['detail'] |
| |
| return f"Document ID {responseJson['doc_id']} version {responseJson['version']} is downloadable via this link: {responseJson['url']}.\n{responseJson['scope']}" |
|
|
| @mcp.tool() |
| def locate_multiple_3gpp_documents(doc_ids: List[str]) -> str: |
| """ |
| Find 3GPP document location with the document's ID |
| Args: doc_id: string |
| """ |
| response = requests.post("https://organizedprogrammers-3gppdocfinder.hf.space/batch", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps({ |
| "doc_ids": doc_ids |
| }), verify=False) |
|
|
| if response.status_code != 200: |
| return f"Unable to find document: {response.status_code} - {response.content}" |
| |
| responseJson = response.json() |
|
|
| if responseJson.get("detail"): |
| return responseJson['detail'] |
| |
| return "\n".join([f"The document {doc_id} is downloadable via this link: {url}" for doc_id, url in responseJson['results']] + [f"We can't find document {doc_id}" for doc_id in responseJson['missing']]) |
|
|
| @mcp.tool() |
| def locate_etsi_document(doc_id: str) -> str: |
| """ |
| Find ETSI document location with the document's ID (starts with SET or SCP) |
| Args: doc_id: string |
| """ |
| response = requests.post("https://organizedprogrammers-etsidocfinder.hf.space/find", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps({ |
| "doc_id": doc_id |
| }), verify=False) |
|
|
| if response.status_code != 200: |
| return f"Unable to find document: {response.status_code} - {response.content}" |
| |
| responseJson = response.json() |
|
|
| if responseJson.get("detail"): |
| return responseJson['detail'] |
| |
| return f"Document ID {responseJson['doc_id']} is downloadable via this link: {responseJson['url']}" |
|
|
| @mcp.tool() |
| def search_3gpp_specifications(keywords: str, threshold: int, release: Optional[str] = "", working_group: Optional[str] = "", spec_type: Optional[str] = "") -> str: |
| """ |
| Search 3GPP specifications with specified keywords and filters using BM25 |
| Args: keywords: string, threshold: integer 0-100 [default 60], release: optional filter, string [only the number Rel-19 -> '19'], working_group: optional filter, string [options: C1,C2,...,C6,CP or S1,S2,...,S6,SP], spec_type: optional filter, string [either TS (Technical Specification) or TR (Technical Report)] |
| For each non-used optional filters, leave a empty string |
| """ |
| body = {"keywords": keywords, "threshold": threshold} |
| if release: |
| body['release'] = release |
| if working_group: |
| body['working_group'] = working_group |
| if spec_type: |
| body['spec_type'] = spec_type |
|
|
| response = requests.post("https://organizedprogrammers-3gppdocfinder.hf.space/search-spec/experimental", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps(body), verify=False) |
|
|
| if response.status_code != 200: |
| return f"Unable to find document: {response.status_code} - {response.content}" |
| |
| responseJson = response.json() |
|
|
| if responseJson.get("detail"): |
| return responseJson['detail'] |
| |
| return "\n--\n".join([f"3GPP {spec['type']} {spec['id']} version {spec['version']} - {spec['title']} is downloadable via this link: {spec['url']}\n{spec['scope']}" for spec in responseJson['results']]) |
|
|
| @mcp.tool() |
| def ask_questions_to_3gpp_database(question: str, threshold: int = 65, release: Optional[str] = "", working_group: Optional[str] = "", spec_type: Optional[str] = "") -> str: |
| """ |
| Retrieve technical documents sections to help AI answer the user's technical question, if same topic already called, re-use the downloaded documents |
| 3GPP specifications are used as source documents, using BM25 to filter the documents |
| Args: question: string, threshold: integer 0-100 [default 60], release: optional filter, string [only the number Rel-19 -> '19'], working_group: optional filter, string [options: C1,C2,...,C6,CP or S1,S2,...,S6,SP], spec_type: optional filter, string [either TS (Technical Specification) or TR (Technical Report)] |
| For each non-used optional filters, leave a empty string |
| After extracting the documents, answer to the question with a complete and detailed paragraph with the sources cited |
| """ |
| body = {"question": question, "threshold": threshold} |
| if release: |
| body['release'] = release |
| if working_group: |
| body['working_group'] = working_group |
| if spec_type: |
| body['spec_type'] = spec_type |
|
|
| response = requests.post("https://organizedprogrammers-3gppdocfinder.hf.space/list-rag-docs", headers={ |
| "Content-Type": "application/json" |
| }, data=json.dumps(body), verify=False) |
|
|
| if response.status_code != 200: |
| return f"Unable to extract documents: {response.status_code}" |
|
|
| docs = response.json()['output'] |
| return docs |
|
|
| if __name__ == "__main__": |
| mcp.run(transport="stdio") |