mirror of
https://github.com/meta-llama/llama-stack.git
synced 2025-10-24 16:57:21 +00:00
# What does this PR do? Refactor setting default vector store provider and embedding model to use an optional `vector_stores` config in the `StackRunConfig` and clean up code to do so (had to add back in some pieces of VectorDB). Also added remote Qdrant and Weaviate to starter distro (based on other PR where inference providers were added for UX). New config is simply (default for Starter distro): ```yaml vector_stores: default_provider_id: faiss default_embedding_model: provider_id: sentence-transformers model_id: nomic-ai/nomic-embed-text-v1.5 ``` ## Test Plan CI and Unit tests. --------- Signed-off-by: Francisco Javier Arceo <farceo@redhat.com> Co-authored-by: Ashwin Bharambe <ashwin.bharambe@gmail.com>
154 lines
5 KiB
Python
154 lines
5 KiB
Python
# Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
# All rights reserved.
|
|
#
|
|
# This source code is licensed under the terms described in the LICENSE file in
|
|
# the root directory of this source tree.
|
|
|
|
import asyncio
|
|
from unittest.mock import MagicMock, patch
|
|
|
|
import numpy as np
|
|
import pytest
|
|
|
|
from llama_stack.apis.files import Files
|
|
from llama_stack.apis.vector_dbs import VectorDB
|
|
from llama_stack.apis.vector_io import Chunk, QueryChunksResponse
|
|
from llama_stack.providers.datatypes import HealthStatus
|
|
from llama_stack.providers.inline.vector_io.faiss.config import FaissVectorIOConfig
|
|
from llama_stack.providers.inline.vector_io.faiss.faiss import (
|
|
FaissIndex,
|
|
FaissVectorIOAdapter,
|
|
)
|
|
|
|
# This test is a unit test for the FaissVectorIOAdapter class. This should only contain
|
|
# tests which are specific to this class. More general (API-level) tests should be placed in
|
|
# tests/integration/vector_io/
|
|
#
|
|
# How to run this test:
|
|
#
|
|
# pytest tests/unit/providers/vector_io/test_faiss.py \
|
|
# -v -s --tb=short --disable-warnings --asyncio-mode=auto
|
|
|
|
FAISS_PROVIDER = "faiss"
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
def loop():
|
|
return asyncio.new_event_loop()
|
|
|
|
|
|
@pytest.fixture
|
|
def embedding_dimension():
|
|
return 768
|
|
|
|
|
|
@pytest.fixture
|
|
def vector_db_id():
|
|
return "test_vector_db"
|
|
|
|
|
|
@pytest.fixture
|
|
def sample_chunks():
|
|
return [
|
|
Chunk(content="MOCK text content 1", mime_type="text/plain", metadata={"document_id": "mock-doc-1"}),
|
|
Chunk(content="MOCK text content 1", mime_type="text/plain", metadata={"document_id": "mock-doc-2"}),
|
|
]
|
|
|
|
|
|
@pytest.fixture
|
|
def sample_embeddings(embedding_dimension):
|
|
return np.random.rand(2, embedding_dimension).astype(np.float32)
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_vector_db(vector_db_id, embedding_dimension) -> MagicMock:
|
|
mock_vector_db = MagicMock(spec=VectorDB)
|
|
mock_vector_db.embedding_model = "mock_embedding_model"
|
|
mock_vector_db.identifier = vector_db_id
|
|
mock_vector_db.embedding_dimension = embedding_dimension
|
|
return mock_vector_db
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_files_api():
|
|
mock_api = MagicMock(spec=Files)
|
|
return mock_api
|
|
|
|
|
|
@pytest.fixture
|
|
def faiss_config():
|
|
config = MagicMock(spec=FaissVectorIOConfig)
|
|
config.kvstore = None
|
|
return config
|
|
|
|
|
|
@pytest.fixture
|
|
async def faiss_index(embedding_dimension):
|
|
index = await FaissIndex.create(dimension=embedding_dimension)
|
|
yield index
|
|
|
|
|
|
async def test_faiss_query_vector_returns_infinity_when_query_and_embedding_are_identical(
|
|
faiss_index, sample_chunks, sample_embeddings, embedding_dimension
|
|
):
|
|
await faiss_index.add_chunks(sample_chunks, sample_embeddings)
|
|
query_embedding = np.random.rand(embedding_dimension).astype(np.float32)
|
|
|
|
with patch.object(faiss_index.index, "search") as mock_search:
|
|
mock_search.return_value = (np.array([[0.0, 0.1]]), np.array([[0, 1]]))
|
|
|
|
response = await faiss_index.query_vector(embedding=query_embedding, k=2, score_threshold=0.0)
|
|
|
|
assert isinstance(response, QueryChunksResponse)
|
|
assert len(response.chunks) == 2
|
|
assert len(response.scores) == 2
|
|
|
|
assert response.scores[0] == float("inf") # infinity (1.0 / 0.0)
|
|
assert response.scores[1] == 10.0 # (1.0 / 0.1 = 10.0)
|
|
|
|
assert response.chunks[0] == sample_chunks[0]
|
|
assert response.chunks[1] == sample_chunks[1]
|
|
|
|
|
|
async def test_health_success():
|
|
"""Test that the health check returns OK status when faiss is working correctly."""
|
|
# Create a fresh instance of FaissVectorIOAdapter for testing
|
|
config = MagicMock()
|
|
inference_api = MagicMock()
|
|
files_api = MagicMock()
|
|
|
|
with patch("llama_stack.providers.inline.vector_io.faiss.faiss.faiss.IndexFlatL2") as mock_index_flat:
|
|
mock_index_flat.return_value = MagicMock()
|
|
adapter = FaissVectorIOAdapter(config=config, inference_api=inference_api, files_api=files_api)
|
|
|
|
# Calling the health method directly
|
|
response = await adapter.health()
|
|
|
|
# Verifying the response
|
|
assert isinstance(response, dict)
|
|
assert response["status"] == HealthStatus.OK
|
|
assert "message" not in response
|
|
|
|
# Verifying that IndexFlatL2 was called with the correct dimension
|
|
mock_index_flat.assert_called_once_with(128) # VECTOR_DIMENSION is 128
|
|
|
|
|
|
async def test_health_failure():
|
|
"""Test that the health check returns ERROR status when faiss encounters an error."""
|
|
# Create a fresh instance of FaissVectorIOAdapter for testing
|
|
config = MagicMock()
|
|
inference_api = MagicMock()
|
|
files_api = MagicMock()
|
|
|
|
with patch("llama_stack.providers.inline.vector_io.faiss.faiss.faiss.IndexFlatL2") as mock_index_flat:
|
|
mock_index_flat.side_effect = Exception("Test error")
|
|
|
|
adapter = FaissVectorIOAdapter(config=config, inference_api=inference_api, files_api=files_api)
|
|
|
|
# Calling the health method directly
|
|
response = await adapter.health()
|
|
|
|
# Verifying the response
|
|
assert isinstance(response, dict)
|
|
assert response["status"] == HealthStatus.ERROR
|
|
assert response["message"] == "Health check failed: Test error"
|