| from utils.check_pydantic_version import use_pydantic_v1 |
| use_pydantic_v1() |
|
|
|
|
| from operator import index |
| import streamlit as st |
| import logging |
| import os |
|
|
| from annotated_text import annotation |
| from json import JSONDecodeError |
| from markdown import markdown |
| from utils.config import parser |
| from utils.haystack import start_document_store, query, initialize_pipeline, start_preprocessor_node, start_retriever, start_reader |
| from utils.ui import reset_results, set_initial_state |
| import pandas as pd |
| import haystack |
|
|
| from datetime import datetime |
| import streamlit.components.v1 as components |
| import streamlit_authenticator as stauth |
| import pickle |
|
|
| from streamlit_modal import Modal |
| import numpy as np |
|
|
|
|
|
|
| names = ['mlreply'] |
| usernames = ['mlreply'] |
| with open('hashed_password.pkl','rb') as f: |
| hashed_passwords = pickle.load(f) |
|
|
|
|
|
|
| |
| DISABLE_FILE_UPLOAD = bool(os.getenv("DISABLE_FILE_UPLOAD")) |
|
|
|
|
| def show_documents_list(retrieved_documents): |
| data = [] |
| for i, document in enumerate(retrieved_documents): |
| data.append([document.meta['name']]) |
| df = pd.DataFrame(data, columns=['Uploaded Document Name']) |
| df.drop_duplicates(subset=['Uploaded Document Name'], inplace=True) |
| df.index = np.arange(1, len(df) + 1) |
| return df |
|
|
| |
| def upload_files(): |
| uploaded_files = upload_container.file_uploader( |
| "upload", type=["pdf", "txt", "docx"], accept_multiple_files=True, label_visibility="hidden", key=1 |
| ) |
| return uploaded_files |
|
|
|
|
| |
| def process_file(data_file, preprocesor, document_store): |
| |
| file_contents = data_file.read().decode("utf-8") |
| docs = [{ |
| 'content': str(file_contents), |
| 'meta': {'name': str(data_file.name)} |
| }] |
| try: |
| names = [item.meta.get('name') for item in document_store.get_all_documents()] |
| |
| |
| if data_file.name in names: |
| print(f"{data_file.name} already processed") |
| else: |
| print(f'preprocessing uploaded doc {data_file.name}.......') |
| |
| preprocessed_docs = preprocesor.process(docs) |
| print('writing to document store.......') |
| document_store.write_documents(preprocessed_docs) |
| print('updating emebdding.......') |
| document_store.update_embeddings(retriever) |
| except Exception as e: |
| print(e) |
|
|
|
|
| |
| def upload_document(): |
| if data_files is not None: |
| for data_file in data_files: |
| |
| if data_file: |
| try: |
| |
| |
| if args.store == 'inmemory': |
| processed_data = process_file(data_file, preprocesor, document_store) |
| |
| except Exception as e: |
| upload_container.write(str(data_file.name) + " β ") |
| upload_container.write("_This file could not be parsed, see the logs for more information._") |
|
|
| |
| def reset_documents(): |
| print('\nReseting documents list at ' + str(datetime.now()) + '\n') |
| st.session_state.data_files = None |
| document_store.delete_documents() |
|
|
| try: |
| args = parser.parse_args() |
| preprocesor = start_preprocessor_node() |
| document_store = start_document_store(type=args.store) |
| document_store.get_all_documents() |
| retriever = start_retriever(document_store) |
| reader = start_reader() |
| st.set_page_config( |
| page_title="MLReplySearch", |
| layout="centered", |
| page_icon=":shark:", |
| menu_items={ |
| 'Get Help': 'https://www.extremelycoolapp.com/help', |
| 'Report a bug': "https://www.extremelycoolapp.com/bug", |
| 'About': "# This is a header. This is an *extremely* cool app!" |
| } |
| ) |
| st.sidebar.image("ml_logo.png", use_column_width=True) |
|
|
| authenticator = stauth.Authenticate(names, usernames, hashed_passwords, "document_search", "random_text", cookie_expiry_days=1) |
|
|
| name, authentication_status, username = authenticator.login("Login", "main") |
|
|
| if authentication_status == False: |
| st.error("Username/Password is incorrect") |
|
|
| if authentication_status == None: |
| st.warning("Please enter your username and password") |
|
|
| if authentication_status: |
|
|
| |
| st.sidebar.header('Options:') |
|
|
| |
| openai_key = st.sidebar.text_input("Enter LLM-authorization Key:", type="password") |
|
|
| if openai_key: |
| task_options = ['Extractive', 'Generative'] |
| else: |
| task_options = ['Extractive'] |
|
|
| task_selection = st.sidebar.radio('Select the task:', task_options) |
|
|
| |
| if task_selection == 'Extractive': |
| pipeline_extractive = initialize_pipeline("extractive", document_store, retriever, reader) |
| elif task_selection == 'Generative' and openai_key: |
| pipeline_rag = initialize_pipeline("rag", document_store, retriever, reader, openai_key=openai_key) |
|
|
|
|
| set_initial_state() |
|
|
| modal = Modal("Manage Files", key="demo-modal") |
| open_modal = st.sidebar.button("Manage Files", use_container_width=True) |
| if open_modal: |
| modal.open() |
|
|
| st.write('# ' + args.name) |
| if modal.is_open(): |
| with modal.container(): |
| if not DISABLE_FILE_UPLOAD: |
| upload_container = st.container() |
| data_files = upload_files() |
| upload_document() |
| st.session_state.sidebar_state = 'collapsed' |
| st.table(show_documents_list(document_store.get_all_documents())) |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| |
| st.sidebar.button("Reset documents", on_click=reset_documents, args=(), use_container_width=True) |
|
|
| if "question" not in st.session_state: |
| st.session_state.question = "" |
| |
| question = st.text_input("Question", value=st.session_state.question, max_chars=100, on_change=reset_results, label_visibility="hidden") |
|
|
| run_pressed = st.button("Run") |
|
|
| run_query = ( |
| run_pressed or question != st.session_state.question |
| ) |
|
|
| |
| if run_query and question: |
| if task_selection == 'Extractive': |
| reset_results() |
| st.session_state.question = question |
| with st.spinner("π Running your pipeline"): |
| try: |
| st.session_state.results_extractive = query(pipeline_extractive, question) |
| st.session_state.task = task_selection |
| except JSONDecodeError as je: |
| st.error( |
| "π An error occurred reading the results. Is the document store working?" |
| ) |
| except Exception as e: |
| logging.exception(e) |
| st.error("π An error occurred during the request.") |
|
|
| elif task_selection == 'Generative': |
| reset_results() |
| st.session_state.question = question |
| with st.spinner("π Running your pipeline"): |
| try: |
| st.session_state.results_generative = query(pipeline_rag, question) |
| st.session_state.task = task_selection |
| except JSONDecodeError as je: |
| st.error( |
| "π An error occurred reading the results. Is the document store working?" |
| ) |
| except Exception as e: |
| if "API key is invalid" in str(e): |
| logging.exception(e) |
| st.error("π incorrect API key provided. You can find your API key at https://platform.openai.com/account/api-keys.") |
| else: |
| logging.exception(e) |
| st.error("π An error occurred during the request.") |
| |
| if (st.session_state.results_extractive or st.session_state.results_generative) and run_query: |
|
|
| |
| if task_selection == 'Extractive': |
| results = st.session_state.results_extractive |
|
|
| st.subheader("Extracted Answers:") |
|
|
| if 'answers' in results: |
| answers = results['answers'] |
| treshold = 0.2 |
| higher_then_treshold = any(ans.score > treshold for ans in answers) |
| if not higher_then_treshold: |
| st.markdown(f"<span style='color:red'>Please note none of the answers achieved a score higher then {int(treshold) * 100}%. Which probably means that the desired answer is not in the searched documents.</span>", unsafe_allow_html=True) |
| for count, answer in enumerate(answers): |
| if answer.answer: |
| text, context = answer.answer, answer.context |
| start_idx = context.find(text) |
| end_idx = start_idx + len(text) |
| score = round(answer.score, 3) |
| st.markdown(f"**Answer {count + 1}:**") |
| st.markdown( |
| context[:start_idx] + str(annotation(body=text, label=f'SCORE {score}', background='#964448', color='#ffffff')) + context[end_idx:], |
| unsafe_allow_html=True, |
| ) |
| else: |
| st.info( |
| "π€ Haystack is unsure whether any of the documents contain an answer to your question. Try to reformulate it!" |
| ) |
|
|
| |
| elif task_selection == 'Generative': |
| results = st.session_state.results_generative |
| st.subheader("Generated Answer:") |
| if 'results' in results: |
| st.markdown("**Answer:**") |
| st.write(results['results'][0]) |
|
|
| |
| if 'documents' in results: |
| retrieved_documents = results['documents'] |
| st.subheader("Retriever Results:") |
|
|
| data = [] |
| for i, document in enumerate(retrieved_documents): |
| |
| truncated_content = (document.content[:150] + '...') if len(document.content) > 150 else document.content |
| data.append([i + 1, document.meta['name'], truncated_content]) |
|
|
| |
| df = pd.DataFrame(data, columns=['Ranked Context', 'Document Name', 'Content']) |
| st.table(df) |
| except SystemExit as e: |
| os._exit(e.code) |