from langchain.prompts import StringPromptTemplate import re from document_scrapped import get_data from qa_txt import conversation_chain # from key_extract import chain from langchain_core.prompts import PromptTemplate from bs4 import BeautifulSoup import requests from data_process import * from langchain.tools.base import StructuredTool from langchain.agents import initialize_agent from qa_txt import llm from trans import trans import pathlib from pydantic import BaseModel from fastapi import FastAPI from fastapi.middleware.cors import CORSMiddleware # import threading, time from langchain.agents import ( create_react_agent, AgentExecutor, tool, ) from langchain import hub import os app = FastAPI(title="Data Gov MA API", description="API for Data Gov MA", version="1.0.0") class Generate(BaseModel): text:str file_text = pathlib.Path('intents_v2.txt').read_text() prompt = hub.pull("hwchase17/react") def faq(query: str) -> str: reponse = conversation_chain.invoke({"input": query, 'document': file_text}) return reponse qa_faq = StructuredTool.from_function( func = faq , description=""" Respond to general questions about the website like the documentation, contact, utility, support... Don't use it when the user request data about a subject (economie, justice, water, or any type of public dataset) only for contact or useful links data. Parameters : - query (string) : the same input as the user input no more no less and dont translate it even if it is in another language. Returns : - string : the output as returned from the function in french. """, ) analyze_data = StructuredTool.from_function( func=get_data, description = """ Analyze and summarize data from a given url asked by the user to summarize or analyze dont use it with the tool request_data and the user must write analyze before this function get executed. Parameters : - url (string) : the url given by the user. Returns : - string : Once the data is outputed from the function it get summarized and returned to the user and it must be in french. Example: >>> analyse this data of this link : https://data.gov.ma/data/fr/dataset/evolution-des-affaires-au-cours-du-quinquennat-2018-2022 >>> input : https://data.gov.ma/data/fr/dataset/evolution-des-affaires-au-cours-du-quinquennat-2018-2022 """, return_direct = True, ) def request_data(query: str) -> str: mot_cle = nettoyer_string(query) mots = mot_cle.split() ui = mots[0] rg = chercher_data(ui) if len(rg[0]): reponse_final = format_reponse(rg) return reponse_final else: return query fetch_data = StructuredTool.from_function( func=request_data, description=""" Request and fetch data using a search keyword. Parameters : - query (string) : A keyword in french about the subject of what to user is looking for, it must be always be in french and a noun if not convert it.For example if the user inputed "I want data about water" you need to input water in french which is "eau" same for other languages and the words translatted must be nouns not adjectives or verbs also the user may request data about an organization where you need to take just the main subject for example "Je veux les données de l'agence de développement digitale" you take just "développement". Returns : - string : the output as returned from the function in french , includes the link to all the data about the keyword along with an example. """, return_direct = True, ) # def request_data(query: str) -> str: # request = chain.invoke({"input": query}) # mot_cle = nettoyer_string(request) # mots = mot_cle.split() # ui = mots[0] # rg = chercher_data(ui) # if len(rg[0]): # reponse_final = format_reponse(rg) # return reponse_final # else: # return "Désolé, il semble que nous n'ayons pas de données correspondant à votre demande pour le moment. Avez-vous une autre question ou avez-vous besoin d'aide sur quelque chose d'autre?" # fetch_data = StructuredTool.from_function( # func=request_data, # description=""" # Request and fetch data using a search keyword. # Parameters : # - query (string) : the same input as the user input no more no less and always it must be in french if it isn't already. For example : "give me data about health" the input is health in french which is santé, same for other languages and the words translatted must be nouns not adjectives or verbs also the user may request data about an organization where you need to take just the main subject for example "Je veux les données de l'agence de développement digitale" you take just "développement". # Returns : # - string : the output as returned from the function in french , includes the link to all the data about the keyword along with an example. # """, # ) def translate(query: str) -> str: translated = trans.invoke({"input": query}) return translated translate_text = StructuredTool.from_function( func=translate, description= """ Translate from any language to french. Don't use it if the text is already in french. Use it only with the function request_data don't use it with the other tools Parameters : - query (string) : the same input as the user input no more no less only if it isnt in french already. Returns : - string : isolate just the translated text in french with no other useless words. """, ) # template = '''Answer the following questions as best you can. You have access to the following tools: # {tools} # Use the following format: # Question: the input question you must answer and must be in french if not translate it in french # Thought: you should always think about what to do # Action: the action to take, should be one of [{tool_names}] # Action Input: the input to the action # Observation: the result of the action, don't include /nObservation in the end of each observation just what you observed # ... (this Thought/Action/Action Input/Observation can repeat N times) # Thought: I now know the final answer # Final Answer: the final answer to the original input question and must always be in french no matter what. # Begin! # Question: {input} # Thought:{agent_scratchpad}''' # prompt_2 = PromptTemplate.from_template(template) tools_add = [ qa_faq, fetch_data, analyze_data, translate_text, ] agent = create_react_agent(llm=llm, tools=tools_add, prompt=prompt) agent_executor = AgentExecutor( agent=agent, tools=tools_add, verbose=True, max_iterations = 10, handle_parsing_errors=True, #max_execution_time = 45, optionel mais useful dans le deployement ) def data_gov_ma(message, history = []): try: response = agent_executor.invoke({"input": message}) final_response = response['output'] timeout_iteration_error = 'Agent stopped due to iteration limit or time limit.' if final_response == timeout_iteration_error: return {"text":"Je suis désolé, je n'ai pas compris votre question.Pourriez-vous la reformuler s'il vous plaît ?"} else: return {"text":final_response} except ValueError as e: return {"text":"Je suis désolé, je n'ai pas compris votre question.Pourriez-vous la reformuler s'il vous plaît ?"} app.add_middleware( CORSMiddleware, allow_origins=["*"], allow_credentials=True, allow_methods=["*"], allow_headers=["*"], ) @app.get("/", tags=["Home"]) def api_home(): return {'detail': 'Welcome to FastAPI TextGen Tutorial!'} @app.post("/api/generate", summary="Generate text from prompt", tags=["Generate"], response_model=Generate) def inference(input_prompt: str): return data_gov_ma(message=input_prompt)