Documentation Index
Fetch the complete documentation index at: https://ibm-llm-runtime-aaf3a78b.mintlify.app/llms.txt
Use this file to discover all available pages before exploring further.
Prerequisites: Quick Start complete,
pip install mellea faiss-cpu sentence-transformers, Ollama running locally.
Step 5 (groundedness checking) additionally requires pip install "mellea[hf]".
Retrieval-augmented generation (RAG) reduces hallucination by grounding the
model’s answer in documents you supply. Mellea adds two things a plain RAG loop
lacks: an LLM-based relevance filter before generation, and optional
groundedness checking after.
The pipeline
Query
|
v
Embedding model → vector search → top-k candidates
|
v
@generative relevance filter
|
v
m.instruct() with grounding_context
|
v
Final answer
(optional: guardian_check groundedness)
Step 1: Index your documents
Use any embedding model and vector store. This example uses
sentence-transformers and a FAISS flat inner-product index:
# Requires: faiss-cpu, sentence-transformers
# Returns: IndexFlatIP
from faiss import IndexFlatIP
from sentence_transformers import SentenceTransformer
def build_index(docs: list[str], model: SentenceTransformer) -> IndexFlatIP:
embeddings = model.encode(docs)
index = IndexFlatIP(embeddings.shape[1])
index.add(embeddings) # type: ignore
return index
def search(
query: str,
docs: list[str],
index: IndexFlatIP,
model: SentenceTransformer,
k: int = 5,
) -> list[str]:
query_vec = model.encode([query])
_, indices = index.search(query_vec, k)
return [docs[i] for i in indices[0]]
IndexFlatIP scores by inner product, which is equivalent to cosine similarity
for L2-normalised embeddings — the default output of sentence-transformers.
Choosing k: start with 5. Too small risks missing the relevant document;
too large floods the filter step and the context window. Tune after measuring
filter acceptance rates.
Step 2: Filter candidates with @generative
Vector similarity finds topically related documents but cannot determine
whether a document actually answers the question. Add an @generative LLM filter:
# Requires: mellea
# Returns: bool
from mellea import generative
@generative
def is_relevant(document: str, question: str) -> bool:
"""Determine whether the document contains information that would help answer the question."""
Apply it after retrieval:
# Requires: mellea, sentence-transformers, faiss-cpu
# Returns: str
from mellea import start_session
embedding_model = SentenceTransformer("all-MiniLM-L6-v2")
index = build_index(docs, embedding_model)
candidates = search(query, docs, index, embedding_model)
del embedding_model # free memory before loading the LLM
m = start_session()
relevant = [
doc for doc in candidates
if is_relevant(m, document=doc, question=query)
]
del embedding_model before starting the Mellea session avoids having both
models resident simultaneously — important on memory-constrained machines.
If all candidates are filtered out, fall back gracefully rather than calling
m.instruct() with an empty context:
if not relevant:
print("No relevant documents found.")
else:
# proceed to generation
...
Step 3: Generate with grounding_context
Pass the surviving documents as named entries in grounding_context. Mellea
injects them into the prompt and tracks them as separate context components:
# Requires: mellea
# Returns: str
answer = m.instruct(
"Using the provided documents, answer the following question: {{question}}",
user_variables={"question": query},
grounding_context={f"doc{i}": doc for i, doc in enumerate(relevant)},
)
print(str(answer))
grounding_context is separate from user_variables so each component is
rendered and traced independently. Without it, m.instruct() generates from
the model’s parametric knowledge — no grounding.
Step 4: Add requirements to the answer (optional)
Use requirements to enforce answer format, length, or citation style:
# Requires: mellea
# Returns: str
from mellea.stdlib.requirements import req, simple_validate
answer = m.instruct(
"Using the provided documents, answer the following question: {{question}}",
user_variables={"question": query},
grounding_context={f"doc{i}": doc for i, doc in enumerate(relevant)},
requirements=[
req("The answer must be based only on the provided documents."),
req(
"The answer must be 100 words or fewer.",
validation_fn=simple_validate(
lambda x: (
len(x.split()) <= 100,
f"Answer is {len(x.split())} words; must be 100 or fewer.",
)
),
),
],
)
Step 5: Check groundedness (optional)
After generation, use guardian_check() with
criteria="groundedness" to verify the answer does not hallucinate beyond the
retrieved documents:
# Requires: mellea[hf]
# Returns: float (0.0–1.0 risk score)
from mellea.backends.huggingface import LocalHFBackend
from mellea.stdlib.components import Document, Message
from mellea.stdlib.components.intrinsic import guardian
from mellea.stdlib.context import ChatContext
guardian_backend = LocalHFBackend(model_id="ibm-granite/granite-4.1-3b")
docs = [Document(text=doc, doc_id=str(i)) for i, doc in enumerate(relevant)]
eval_ctx = (
ChatContext()
.add(Message("user", query))
.add(Message("assistant", str(answer), documents=docs))
)
score = guardian.guardian_check(eval_ctx, guardian_backend, criteria="groundedness")
if score < 0.5:
print(f"Grounded answer (score: {score:.4f}):", str(answer))
else:
print(f"Groundedness risk detected (score: {score:.4f})")
Include the same documents in the evaluation context that you passed to
grounding_context — this ensures the groundedness model evaluates the answer
against exactly what the generator was given.
Putting it together
# Requires: mellea[hf], faiss-cpu, sentence-transformers
# Returns: str
from faiss import IndexFlatIP
from sentence_transformers import SentenceTransformer
from mellea import generative, start_session
from mellea.backends.huggingface import LocalHFBackend
from mellea.stdlib.components import Document, Message
from mellea.stdlib.components.intrinsic import guardian
from mellea.stdlib.context import ChatContext
from mellea.stdlib.requirements import req, simple_validate
@generative
def is_relevant(document: str, question: str) -> bool:
"""Determine whether the document contains information that would help answer the question."""
def build_index(docs: list[str], model: SentenceTransformer) -> IndexFlatIP:
embeddings = model.encode(docs)
index = IndexFlatIP(embeddings.shape[1])
index.add(embeddings) # type: ignore
return index
def search(query: str, docs: list[str], index: IndexFlatIP,
model: SentenceTransformer, k: int = 5) -> list[str]:
_, indices = index.search(model.encode([query]), k)
return [docs[i] for i in indices[0]]
# Loaded at module scope so weights are downloaded once and the model stays
# resident across calls. First import triggers a multi-GB Granite download.
guardian_backend = LocalHFBackend(model_id="ibm-granite/granite-4.1-3b")
def rag(docs: list[str], query: str, *, check_groundedness: bool = True) -> str | None:
embedding_model = SentenceTransformer("all-MiniLM-L6-v2")
index = build_index(docs, embedding_model)
candidates = search(query, docs, index, embedding_model)
del embedding_model
m = start_session()
relevant = [doc for doc in candidates if is_relevant(m, document=doc, question=query)]
if not relevant:
return None
answer = m.instruct(
"Using the provided documents, answer this question: {{question}}",
user_variables={"question": query},
grounding_context={f"doc{i}": doc for i, doc in enumerate(relevant)},
requirements=[req("Answer only from the provided documents.")],
)
# Optional: groundedness check. Adds Guardian latency on every call;
# disable with `check_groundedness=False` when latency matters more
# than fact-verification (e.g. low-stakes summaries).
if check_groundedness:
docs_for_eval = [Document(text=doc, doc_id=str(i)) for i, doc in enumerate(relevant)]
eval_ctx = (
ChatContext()
.add(Message("user", query))
.add(Message("assistant", str(answer), documents=docs_for_eval))
)
score = guardian.guardian_check(eval_ctx, guardian_backend, criteria="groundedness")
if score >= 0.5:
print(f"Warning: groundedness risk detected (score: {score:.4f})")
return str(answer)
What to tune
| Parameter | Effect | Starting point |
|---|
k in search() | Candidates passed to the filter | 5 |
is_relevant docstring | How strictly the filter interprets relevance | Adjust phrasing to match your domain |
grounding_context key names | Tracing and debugging in spans | Use descriptive names in production |
requirements on m.instruct() | Answer length, citation, tone | Add after baseline quality is good |
guardian_check document context | What the groundedness model checks against | Match exactly what you pass to grounding_context |
See also: Resilient RAG with Fallback Filtering | Making Agents Reliable | The Requirements System