VectorStore/QA, MMR support¶
NOTE: this uses Cassandra's "Vector Search" capability. Make sure you are connecting to a vector-enabled database for this demo.
Cassandra's VectorStore
allows for Vector Search with the Maximal Marginal Relevance (MMR) algorithm.
This is a search criterion that instead of just selecting the k stored documents most relevant to the provided query, first identifies a larger pool of relevant results, and then singles out k of them so that they carry as diverse information between them as possible.
In this way, when the stored text fragments are likely to be redundant, you can optimize token usage and help the models give more comprehensive answers.
This is very useful, for instance, if you are building a QA chatbot on past Support chat recorded interactions.
First prepare a connection to a vector-search-capable Cassandra and initialize the required LLM and embeddings:
from langchain.indexes.vectorstore import VectorStoreIndexWrapper
from langchain.vectorstores import Cassandra
from cqlsession import getCQLSession, getCQLKeyspace
cqlMode = 'astra_db' # 'astra_db'/'local'
session = getCQLSession(mode=cqlMode)
keyspace = getCQLKeyspace(mode=cqlMode)
Below is the logic to instantiate the LLM and embeddings of choice. We chose to leave it in the notebooks for clarity.
import os
from llm_choice import suggestLLMProvider
llmProvider = suggestLLMProvider()
# (Alternatively set llmProvider to 'GCP_VertexAI', 'OpenAI', 'Azure_OpenAI' ... manually if you have credentials)
if llmProvider == 'GCP_VertexAI':
from langchain.llms import VertexAI
from langchain.embeddings import VertexAIEmbeddings
llm = VertexAI()
myEmbedding = VertexAIEmbeddings()
print('LLM+embeddings from Vertex AI')
elif llmProvider == 'OpenAI':
os.environ['OPENAI_API_TYPE'] = 'open_ai'
from langchain.llms import OpenAI
from langchain.embeddings import OpenAIEmbeddings
llm = OpenAI(temperature=0)
myEmbedding = OpenAIEmbeddings()
print('LLM+embeddings from OpenAI')
elif llmProvider == 'Azure_OpenAI':
os.environ['OPENAI_API_TYPE'] = 'azure'
os.environ['OPENAI_API_VERSION'] = os.environ['AZURE_OPENAI_API_VERSION']
os.environ['OPENAI_API_BASE'] = os.environ['AZURE_OPENAI_API_BASE']
os.environ['OPENAI_API_KEY'] = os.environ['AZURE_OPENAI_API_KEY']
from langchain.llms import AzureOpenAI
from langchain.embeddings import OpenAIEmbeddings
llm = AzureOpenAI(temperature=0, model_name=os.environ['AZURE_OPENAI_LLM_MODEL'],
engine=os.environ['AZURE_OPENAI_LLM_DEPLOYMENT'])
myEmbedding = OpenAIEmbeddings(model=os.environ['AZURE_OPENAI_EMBEDDINGS_MODEL'],
deployment=os.environ['AZURE_OPENAI_EMBEDDINGS_DEPLOYMENT'])
print('LLM+embeddings from Azure OpenAI')
else:
raise ValueError('Unknown LLM provider.')
LLM+embeddings from OpenAI
Create the store¶
Create a (Cassandra-backed) VectorStore
and the corresponding LangChain VectorStoreIndexWrapper
myCassandraVStore = Cassandra(
embedding=myEmbedding,
session=session,
keyspace=keyspace,
table_name='vs_test2_' + llmProvider,
)
index = VectorStoreIndexWrapper(vectorstore=myCassandraVStore)
This command simply resets the store in case you want to run this demo repeatedly:
myCassandraVStore.clear()
Populate the index¶
Notice that the first four sentences express the same concept, while the fifth adds a new detail:
BASE_SENTENCE_0 = ('The frogs and the toads were meeting in the night '
'for a party under the moon.')
BASE_SENTENCE_1 = ('There was a party under the moon, that all toads, '
'with the frogs, decided to throw that night.')
BASE_SENTENCE_2 = ('And the frogs and the toads said: "Let us have a party '
'tonight, as the moon is shining".')
BASE_SENTENCE_3 = ('I remember that night... toads, along with frogs, '
'were all busy planning a moonlit celebration.')
DIFFERENT_SENTENCE = ('For the party, frogs and toads set a rule: '
'everyone was to wear a purple hat.')
Insert the three into the index, specifying "sources" while you're at it (it will be useful later):
texts = [
BASE_SENTENCE_0,
BASE_SENTENCE_1,
BASE_SENTENCE_2,
BASE_SENTENCE_3,
DIFFERENT_SENTENCE,
]
metadatas = [
{'source': 'Barney\'s story at the pub'},
{'source': 'Barney\'s story at the pub'},
{'source': 'Barney\'s story at the pub'},
{'source': 'Barney\'s story at the pub'},
{'source': 'The chronicles at the village library'},
]
if llmProvider != 'Azure_OpenAI':
ids = myCassandraVStore.add_texts(
texts,
metadatas=metadatas,
)
print('\n'.join(ids))
else:
# Note: this is a temporary mitigation of an Azure OpenAI error with asking for
# multiple embedding in a single request, which would error with:
# "InvalidRequestError: Too many inputs. The max number of inputs is 1"
for text, metadata in zip(texts, metadatas):
thisId = myCassandraVStore.add_texts(
[text],
metadatas=[metadata],
)[0]
print(thisId)
95e21aa001bb4b4fbc93937ed88a6f25 2e96aefb71344f3390851d63e47cb5a7 941ba0c152a14c849f999787b1892652 7078275068fe4ab38a2d2dbe5cb443e2 a57a7b7ad92c47079ccdd976b0a4063c
Query the store¶
Here is the question you'll use to query the index:
QUESTION = 'Tell me about the party that night.'
Query with "similarity" search type¶
If you ask for two matches, you will get the two documents most related to the question. But in this case this is something of a waste of tokens:
matchesSim = myCassandraVStore.search(QUESTION, search_type='similarity', k=2)
for i, doc in enumerate(matchesSim):
print(f'[{i:2}]: "{doc.page_content}"')
[ 0]: "There was a party under the moon, that all toads, with the frogs, decided to throw that night." [ 1]: "I remember that night... toads, along with frogs, were all busy planning a moonlit celebration."
Query with MMR¶
Now, here's what happens with the MMR search type.
(Not shown here: you can tune the size of the results pool for the first step of the algorithm.)
matchesMMR = myCassandraVStore.search(QUESTION, search_type='mmr', k=2)
for i, doc in enumerate(matchesMMR):
print(f'[{i:2}]: "{doc.page_content}"')
[ 0]: "There was a party under the moon, that all toads, with the frogs, decided to throw that night." [ 1]: "For the party, frogs and toads set a rule: everyone was to wear a purple hat."
Query the index¶
Currently, LangChain's higher "index" abstraction does not allow to specify the search type, nor the number of matches subsequently used in creating the answer. So, by running this command you get an answer, all right.
# (implicitly) by similarity
print(index.query(QUESTION, llm=llm))
The frogs and toads were having a party under the moon that night. They were busy planning and celebrating together.
You can request the question-answering process to provide references (as long as you annotated all input documents with a source
metadata field):
responseSrc = index.query_with_sources(QUESTION, llm=llm)
print('Automatic chain (implicitly by similarity):')
print(f' ANSWER : {responseSrc["answer"].strip()}')
print(f' SOURCES: {responseSrc["sources"].strip()}')
Automatic chain (implicitly by similarity): ANSWER : The frogs and toads were planning a party under the moon that night. SOURCES: Barney's story at the pub
Here the default is to fetch four documents ... so that the only other text actually carrying additional information is left out!
The QA Process behind the scenes¶
In order to exploit the MMR search in end-to-end question-answering pipelines, you need to recreate and manually tweak the steps behind the query
or query_with_sources
methods. This takes just a few lines.
First you need a few additional modules:
from langchain.chains.retrieval_qa.base import RetrievalQA
from langchain.chains.qa_with_sources.retrieval import RetrievalQAWithSourcesChain
You are ready to run two QA chains, identical in all respects (especially in the number of results to fetch, two), except the search_type
:
Similarity-based QA¶
# manual creation of the "retriever" with the 'similarity' search type
retrieverSim = myCassandraVStore.as_retriever(
search_type='similarity',
search_kwargs={
'k': 2,
# ...
},
)
# Create a "RetrievalQA" chain
chainSim = RetrievalQA.from_chain_type(
llm=llm,
retriever=retrieverSim,
)
# Run it and print results
responseSim = chainSim.run(QUESTION)
print(responseSim)
The party was held under the moon and was planned by both toads and frogs.
MMR-based QA¶
# manual creation of the "retriever" with the 'MMR' search type
retrieverMMR = myCassandraVStore.as_retriever(
search_type='mmr',
search_kwargs={
'k': 2,
# ...
},
)
# Create a "RetrievalQA" chain
chainMMR = RetrievalQA.from_chain_type(
llm=llm,
retriever=retrieverMMR
)
# Run it and print results
responseMMR = chainMMR.run(QUESTION)
print(responseMMR)
The party was held under the moon and was attended by both frogs and toads. Everyone was required to wear a purple hat.
Answers with sources¶
You can run the variant of these chains that also returns the source for the documents used in preparing the answer, which makes it even more obvious:
chainSimSrc = RetrievalQAWithSourcesChain.from_chain_type(
llm,
retriever=retrieverSim,
)
#
responseSimSrc = chainSimSrc({chainSimSrc.question_key: QUESTION})
print('Similarity-based chain:')
print(f' ANSWER : {responseSimSrc["answer"].strip()}')
print(f' SOURCES: {responseSimSrc["sources"].strip()}')
Similarity-based chain: ANSWER : The toads and frogs were planning a moonlit celebration. SOURCES: Barney's story at the pub
chainMMRSrc = RetrievalQAWithSourcesChain.from_chain_type(
llm,
retriever=retrieverMMR,
)
#
responseMMRSrc = chainMMRSrc({chainMMRSrc.question_key: QUESTION})
print('MMR-based chain:')
print(f' ANSWER : {responseMMRSrc["answer"].strip()}')
print(f' SOURCES: {responseMMRSrc["sources"].strip()}')
MMR-based chain: ANSWER : The party that night was thrown by frogs and toads, and everyone was required to wear a purple hat. SOURCES: Barney's story at the pub, The chronicles at the village library