import streamlit as st import json import time import faiss from sentence_transformers import SentenceTransformer from sentence_transformers.cross_encoder import CrossEncoder class DocumentSearch: ''' This class is dedicated to perform semantic document search based on previously trained: faiss: index sbert: encoder sbert: cross_encoder ''' # we mention pass to every file that needed to run models # and search over our data enc_path = "ivan-savchuk/msmarco-distilbert-dot-v5-tuned-full-v1" idx_path = "idx_vectors.index" cross_enc_path = "ivan-savchuk/cross-encoder-ms-marco-MiniLM-L-12-v2-tuned_mediqa-v1" docs_path = "docs.json" def __init__(self): # loading docs and corresponding urls with open(DocumentSearch.docs_path, 'r') as json_file: self.docs = json.load(json_file) # loading sbert encoder model self.encoder = SentenceTransformer(DocumentSearch.enc_path) # loading faiss index self.index = faiss.read_index(DocumentSearch.idx_path) # loading sbert cross_encoder self.cross_encoder = CrossEncoder(DocumentSearch.cross_enc_path) def search(self, query: str, k: int) -> list: # get vector representation of text query query_vector = self.encoder.encode([query]) # perform search via faiss FlatIP index _, indeces = self.index.search(query_vector, k*10) # get answers by index answers = [self.docs[i] for i in indeces[0]] # prepare inputs for cross encoder model_inputs = [[query, pairs[0]] for pairs in answers] urls = [pairs[1] for pairs in answers] # get similarity score between query and documents scores = self.cross_encoder.predict(model_inputs, batch_size=1) # compose results into list of dicts results = [{'doc': doc[1], 'url': url, 'score': score} for doc, url, score in zip(model_inputs, urls, scores)] # return results sorteed by similarity scores return sorted(results, key=lambda x: x['score'], reverse=True)[:k] if __name__ == "__main__": # get instance of DocumentSearch class surfer = DocumentSearch() # streamlit part starts here with title title = """