rerank_example.py 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. """
  2. LightRAG Rerank Integration Example
  3. This example demonstrates how to use rerank functionality with LightRAG
  4. to improve retrieval quality across different query modes.
  5. Configuration Required:
  6. 1. Set your OpenAI LLM API key and base URL with env vars
  7. LLM_MODEL
  8. LLM_BINDING_HOST
  9. LLM_BINDING_API_KEY
  10. 2. Set your OpenAI embedding API key and base URL with env vars:
  11. EMBEDDING_MODEL
  12. EMBEDDING_DIM
  13. EMBEDDING_BINDING_HOST
  14. EMBEDDING_BINDING_API_KEY
  15. 3. Set your vLLM deployed AI rerank model setting with env vars:
  16. RERANK_BINDING=cohere
  17. RERANK_MODEL (e.g., answerai-colbert-small-v1 or rerank-v3.5)
  18. RERANK_BINDING_HOST (e.g., https://api.cohere.com/v2/rerank or LiteLLM proxy)
  19. RERANK_BINDING_API_KEY
  20. RERANK_ENABLE_CHUNKING=true (optional, for models with token limits)
  21. RERANK_MAX_TOKENS_PER_DOC=480 (optional, default 4096)
  22. Note: Rerank is controlled per query via the 'enable_rerank' parameter (default: True)
  23. """
  24. import asyncio
  25. import os
  26. import numpy as np
  27. from lightrag import LightRAG, QueryParam
  28. from lightrag.llm.openai import openai_complete_if_cache, openai_embed
  29. from lightrag.utils import EmbeddingFunc, setup_logger
  30. from functools import partial
  31. from lightrag.rerank import cohere_rerank
  32. # Set up your working directory
  33. WORKING_DIR = "./test_rerank"
  34. setup_logger("test_rerank")
  35. if not os.path.exists(WORKING_DIR):
  36. os.mkdir(WORKING_DIR)
  37. async def llm_model_func(
  38. prompt, system_prompt=None, history_messages=[], **kwargs
  39. ) -> str:
  40. return await openai_complete_if_cache(
  41. os.getenv("LLM_MODEL"),
  42. prompt,
  43. system_prompt=system_prompt,
  44. history_messages=history_messages,
  45. api_key=os.getenv("LLM_BINDING_API_KEY"),
  46. base_url=os.getenv("LLM_BINDING_HOST"),
  47. **kwargs,
  48. )
  49. async def embedding_func(texts: list[str]) -> np.ndarray:
  50. return await openai_embed(
  51. texts,
  52. model=os.getenv("EMBEDDING_MODEL"),
  53. api_key=os.getenv("EMBEDDING_BINDING_API_KEY"),
  54. base_url=os.getenv("EMBEDDING_BINDING_HOST"),
  55. )
  56. rerank_model_func = partial(
  57. cohere_rerank,
  58. model=os.getenv("RERANK_MODEL", "rerank-v3.5"),
  59. api_key=os.getenv("RERANK_BINDING_API_KEY"),
  60. base_url=os.getenv("RERANK_BINDING_HOST", "https://api.cohere.com/v2/rerank"),
  61. enable_chunking=os.getenv("RERANK_ENABLE_CHUNKING", "false").lower() == "true",
  62. max_tokens_per_doc=int(os.getenv("RERANK_MAX_TOKENS_PER_DOC", "4096")),
  63. )
  64. async def create_rag_with_rerank():
  65. """Create LightRAG instance with rerank configuration"""
  66. # Get embedding dimension
  67. test_embedding = await embedding_func(["test"])
  68. embedding_dim = test_embedding.shape[1]
  69. print(f"Detected embedding dimension: {embedding_dim}")
  70. # Method 1: Using custom rerank function
  71. rag = LightRAG(
  72. working_dir=WORKING_DIR,
  73. llm_model_func=llm_model_func,
  74. embedding_func=EmbeddingFunc(
  75. embedding_dim=embedding_dim,
  76. max_token_size=8192,
  77. func=embedding_func,
  78. ),
  79. # Rerank Configuration - provide the rerank function
  80. rerank_model_func=rerank_model_func,
  81. )
  82. await rag.initialize_storages() # Auto-initializes pipeline_status
  83. return rag
  84. async def test_rerank_with_different_settings():
  85. """
  86. Test rerank functionality with different enable_rerank settings
  87. """
  88. print("\n\n🚀 Setting up LightRAG with Rerank functionality...")
  89. rag = await create_rag_with_rerank()
  90. # Insert sample documents
  91. sample_docs = [
  92. "Reranking improves retrieval quality by re-ordering documents based on relevance.",
  93. "LightRAG is a powerful retrieval-augmented generation system with multiple query modes.",
  94. "Vector databases enable efficient similarity search in high-dimensional embedding spaces.",
  95. "Natural language processing has evolved with large language models and transformers.",
  96. "Machine learning algorithms can learn patterns from data without explicit programming.",
  97. ]
  98. print("📄 Inserting sample documents...")
  99. await rag.ainsert(sample_docs)
  100. query = "How does reranking improve retrieval quality?"
  101. print(f"\n🔍 Testing query: '{query}'")
  102. print("=" * 80)
  103. # Test with rerank enabled (default)
  104. print("\n📊 Testing with enable_rerank=True (default):")
  105. result_with_rerank = await rag.aquery(
  106. query,
  107. param=QueryParam(
  108. mode="naive",
  109. top_k=10,
  110. chunk_top_k=5,
  111. enable_rerank=True, # Explicitly enable rerank
  112. ),
  113. )
  114. print(f" Result length: {len(result_with_rerank)} characters")
  115. print(f" Preview: {result_with_rerank[:100]}...")
  116. # Test with rerank disabled
  117. print("\n📊 Testing with enable_rerank=False:")
  118. result_without_rerank = await rag.aquery(
  119. query,
  120. param=QueryParam(
  121. mode="naive",
  122. top_k=10,
  123. chunk_top_k=5,
  124. enable_rerank=False, # Disable rerank
  125. ),
  126. )
  127. print(f" Result length: {len(result_without_rerank)} characters")
  128. print(f" Preview: {result_without_rerank[:100]}...")
  129. # Test with default settings (enable_rerank defaults to True)
  130. print("\n📊 Testing with default settings (enable_rerank defaults to True):")
  131. result_default = await rag.aquery(
  132. query, param=QueryParam(mode="naive", top_k=10, chunk_top_k=5)
  133. )
  134. print(f" Result length: {len(result_default)} characters")
  135. print(f" Preview: {result_default[:100]}...")
  136. async def test_direct_rerank():
  137. """Test rerank function directly"""
  138. print("\n🔧 Direct Rerank API Test")
  139. print("=" * 40)
  140. documents = [
  141. "Vector search finds semantically similar documents",
  142. "LightRAG supports advanced reranking capabilities",
  143. "Reranking significantly improves retrieval quality",
  144. "Natural language processing with modern transformers",
  145. "The quick brown fox jumps over the lazy dog",
  146. ]
  147. query = "rerank improve quality"
  148. print(f"Query: '{query}'")
  149. print(f"Documents: {len(documents)}")
  150. try:
  151. reranked_results = await rerank_model_func(
  152. query=query,
  153. documents=documents,
  154. top_n=4,
  155. )
  156. print("\n✅ Rerank Results:")
  157. i = 0
  158. for result in reranked_results:
  159. index = result["index"]
  160. score = result["relevance_score"]
  161. content = documents[index]
  162. print(f" {index}. Score: {score:.4f} | {content}...")
  163. i += 1
  164. except Exception as e:
  165. print(f"❌ Rerank failed: {e}")
  166. async def main():
  167. """Main example function"""
  168. print("🎯 LightRAG Rerank Integration Example")
  169. print("=" * 60)
  170. try:
  171. # Test direct rerank
  172. await test_direct_rerank()
  173. # Test rerank with different enable_rerank settings
  174. await test_rerank_with_different_settings()
  175. print("\n✅ Example completed successfully!")
  176. print("\n💡 Key Points:")
  177. print(" ✓ Rerank is now controlled per query via 'enable_rerank' parameter")
  178. print(" ✓ Default value for enable_rerank is True")
  179. print(" ✓ Rerank function is configured at LightRAG initialization")
  180. print(" ✓ Per-query enable_rerank setting overrides default behavior")
  181. print(
  182. " ✓ If enable_rerank=True but no rerank model is configured, a warning is issued"
  183. )
  184. print(" ✓ Monitor API usage and costs when using rerank services")
  185. except Exception as e:
  186. print(f"\n❌ Example failed: {e}")
  187. import traceback
  188. traceback.print_exc()
  189. if __name__ == "__main__":
  190. asyncio.run(main())