import os import logging from llama_index import GPTSimpleVectorIndex from llama_index import download_loader from llama_index import ( Document, LLMPredictor, PromptHelper, QuestionAnswerPrompt, RefinePrompt, ) from langchain.llms import OpenAI import colorama from modules.presets import * from modules.utils import * def get_index_name(file_src): index_name = [] for file in file_src: index_name.append(os.path.basename(file.name)) index_name = sorted(index_name) index_name = "".join(index_name) index_name = sha1sum(index_name) return index_name def get_documents(file_src): documents = [] logging.debug("Loading documents...") logging.debug(f"file_src: {file_src}") for file in file_src: logging.info(f"loading file: {file.name}") if os.path.splitext(file.name)[1] == ".pdf": logging.debug("Loading PDF...") CJKPDFReader = download_loader("CJKPDFReader") loader = CJKPDFReader() text_raw = loader.load_data(file=file.name)[0].text elif os.path.splitext(file.name)[1] == ".docx": logging.debug("Loading DOCX...") DocxReader = download_loader("DocxReader") loader = DocxReader() text_raw = loader.load_data(file=file.name)[0].text elif os.path.splitext(file.name)[1] == ".epub": logging.debug("Loading EPUB...") EpubReader = download_loader("EpubReader") loader = EpubReader() text_raw = loader.load_data(file=file.name)[0].text else: logging.debug("Loading text file...") with open(file.name, "r", encoding="utf-8") as f: text_raw = f.read() text = add_space(text_raw) documents += [Document(text)] return documents def construct_index( api_key, file_src, max_input_size=4096, num_outputs=1, max_chunk_overlap=20, chunk_size_limit=600, embedding_limit=None, separator=" ", num_children=10, max_keywords_per_chunk=10, ): os.environ["OPENAI_API_KEY"] = api_key chunk_size_limit = None if chunk_size_limit == 0 else chunk_size_limit embedding_limit = None if embedding_limit == 0 else embedding_limit separator = " " if separator == "" else separator llm_predictor = LLMPredictor( llm=OpenAI(model_name="gpt-3.5-turbo-0301", openai_api_key=api_key) ) prompt_helper = PromptHelper( max_input_size, num_outputs, max_chunk_overlap, embedding_limit, chunk_size_limit, separator=separator, ) index_name = get_index_name(file_src) if os.path.exists(f"./index/{index_name}.json"): logging.info("找到了缓存的索引文件,加载中……") return GPTSimpleVectorIndex.load_from_disk(f"./index/{index_name}.json") else: try: documents = get_documents(file_src) logging.debug("构建索引中……") index = GPTSimpleVectorIndex( documents, llm_predictor=llm_predictor, prompt_helper=prompt_helper ) os.makedirs("./index", exist_ok=True) index.save_to_disk(f"./index/{index_name}.json") return index except Exception as e: print(e) return None def chat_ai( api_key, index, question, context, chatbot, reply_language, ): os.environ["OPENAI_API_KEY"] = api_key logging.info(f"Question: {question}") response, chatbot_display, status_text = ask_ai( api_key, index, question, replace_today(PROMPT_TEMPLATE), REFINE_TEMPLATE, SIM_K, INDEX_QUERY_TEMPRATURE, context, reply_language, ) if response is None: status_text = "查询失败,请换个问法试试" return context, chatbot response = response context.append({"role": "user", "content": question}) context.append({"role": "assistant", "content": response}) chatbot.append((question, chatbot_display)) os.environ["OPENAI_API_KEY"] = "" return context, chatbot, status_text def ask_ai( api_key, index, question, prompt_tmpl, refine_tmpl, sim_k=1, temprature=0, prefix_messages=[], reply_language="中文", ): os.environ["OPENAI_API_KEY"] = api_key logging.debug("Index file found") logging.debug("Querying index...") llm_predictor = LLMPredictor( llm=OpenAI( temperature=temprature, model_name="gpt-3.5-turbo-0301", prefix_messages=prefix_messages, ) ) response = None # Initialize response variable to avoid UnboundLocalError qa_prompt = QuestionAnswerPrompt(prompt_tmpl.replace("{reply_language}", reply_language)) rf_prompt = RefinePrompt(refine_tmpl.replace("{reply_language}", reply_language)) response = index.query( question, llm_predictor=llm_predictor, similarity_top_k=sim_k, text_qa_template=qa_prompt, refine_template=rf_prompt, response_mode="compact", ) if response is not None: logging.info(f"Response: {response}") ret_text = response.response nodes = [] for index, node in enumerate(response.source_nodes): brief = node.source_text[:25].replace("\n", "") nodes.append( f"<details><summary>[{index + 1}]\t{brief}...</summary><p>{node.source_text}</p></details>" ) new_response = ret_text + "\n----------\n" + "\n\n".join(nodes) logging.info( f"Response: {colorama.Fore.BLUE}{ret_text}{colorama.Style.RESET_ALL}" ) os.environ["OPENAI_API_KEY"] = "" return ret_text, new_response, f"查询消耗了{llm_predictor.last_token_usage} tokens" else: logging.warning("No response found, returning None") os.environ["OPENAI_API_KEY"] = "" return None def add_space(text): punctuations = {",": ", ", "。": "。 ", "?": "? ", "!": "! ", ":": ": ", ";": "; "} for cn_punc, en_punc in punctuations.items(): text = text.replace(cn_punc, en_punc) return text