feat: adds integration test for usage logger
This commit is contained in:
parent
da35f028df
commit
707269e8b8
1 changed files with 348 additions and 0 deletions
348
cognee/tests/integration/shared/test_usage_logger_integration.py
Normal file
348
cognee/tests/integration/shared/test_usage_logger_integration.py
Normal file
|
|
@ -0,0 +1,348 @@
|
||||||
|
import os
|
||||||
|
import pytest
|
||||||
|
import asyncio
|
||||||
|
from datetime import datetime, timezone
|
||||||
|
from types import SimpleNamespace
|
||||||
|
from uuid import UUID
|
||||||
|
from cognee.shared.usage_logger import log_usage
|
||||||
|
from cognee.infrastructure.databases.cache.config import get_cache_config
|
||||||
|
from cognee.infrastructure.databases.cache.get_cache_engine import get_cache_engine, create_cache_engine
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def usage_logging_config():
|
||||||
|
"""Fixture to enable usage logging via environment variables."""
|
||||||
|
original_env = os.environ.copy()
|
||||||
|
os.environ["USAGE_LOGGING"] = "true"
|
||||||
|
os.environ["CACHE_BACKEND"] = "redis"
|
||||||
|
os.environ["CACHE_HOST"] = "localhost"
|
||||||
|
os.environ["CACHE_PORT"] = "6379"
|
||||||
|
get_cache_config.cache_clear()
|
||||||
|
create_cache_engine.cache_clear()
|
||||||
|
yield
|
||||||
|
os.environ.clear()
|
||||||
|
os.environ.update(original_env)
|
||||||
|
get_cache_config.cache_clear()
|
||||||
|
create_cache_engine.cache_clear()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def usage_logging_disabled():
|
||||||
|
"""Fixture to disable usage logging via environment variables."""
|
||||||
|
original_env = os.environ.copy()
|
||||||
|
os.environ["USAGE_LOGGING"] = "false"
|
||||||
|
os.environ["CACHE_BACKEND"] = "redis"
|
||||||
|
get_cache_config.cache_clear()
|
||||||
|
create_cache_engine.cache_clear()
|
||||||
|
yield
|
||||||
|
os.environ.clear()
|
||||||
|
os.environ.update(original_env)
|
||||||
|
get_cache_config.cache_clear()
|
||||||
|
create_cache_engine.cache_clear()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def redis_adapter():
|
||||||
|
"""Real RedisAdapter instance for testing."""
|
||||||
|
from cognee.infrastructure.databases.cache.redis.RedisAdapter import RedisAdapter
|
||||||
|
|
||||||
|
try:
|
||||||
|
adapter = RedisAdapter(host="localhost", port=6379, log_key="test_usage_logs")
|
||||||
|
yield adapter
|
||||||
|
except Exception as e:
|
||||||
|
pytest.skip(f"Redis not available: {e}")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def test_user():
|
||||||
|
"""Test user object."""
|
||||||
|
return SimpleNamespace(id="test-user-123")
|
||||||
|
|
||||||
|
|
||||||
|
class TestDecoratorBehavior:
|
||||||
|
"""Test decorator behavior with real components."""
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_decorator_skips_when_disabled(self, usage_logging_disabled):
|
||||||
|
"""Test decorator skips logging when usage_logging=False."""
|
||||||
|
call_count = 0
|
||||||
|
|
||||||
|
@log_usage(function_name="test_func", log_type="test")
|
||||||
|
async def test_func():
|
||||||
|
nonlocal call_count
|
||||||
|
call_count += 1
|
||||||
|
return "result"
|
||||||
|
|
||||||
|
assert await test_func() == "result"
|
||||||
|
assert call_count == 1
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_decorator_basic_logging(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test decorator logs to Redis and handles various scenarios."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="test_func", log_type="test")
|
||||||
|
async def test_func(param1: str, param2: int = 42, user=None):
|
||||||
|
await asyncio.sleep(0.01)
|
||||||
|
return {"result": f"{param1}_{param2}"}
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
# Test basic logging
|
||||||
|
result = await test_func("value1", user=test_user)
|
||||||
|
assert result == {"result": "value1_42"}
|
||||||
|
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert len(logs) > 0
|
||||||
|
log = logs[0]
|
||||||
|
assert log["function_name"] == "test_func"
|
||||||
|
assert log["type"] == "test"
|
||||||
|
assert log["user_id"] == "test-user-123"
|
||||||
|
assert log["parameters"]["param1"] == "value1"
|
||||||
|
assert log["parameters"]["param2"] == 42
|
||||||
|
assert log["success"] is True
|
||||||
|
|
||||||
|
# Test log entry structure
|
||||||
|
required_fields = [
|
||||||
|
"timestamp", "type", "function_name", "user_id", "parameters",
|
||||||
|
"result", "success", "error", "duration_ms", "start_time", "end_time", "metadata"
|
||||||
|
]
|
||||||
|
for field in required_fields:
|
||||||
|
assert field in log
|
||||||
|
assert "cognee_version" in log["metadata"]
|
||||||
|
assert "environment" in log["metadata"]
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_decorator_handles_cache_engine_none(self, usage_logging_config):
|
||||||
|
"""Test decorator handles gracefully when cache engine is None."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="test_func", log_type="test")
|
||||||
|
async def test_func():
|
||||||
|
return "result"
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = None
|
||||||
|
assert await test_func() == "result"
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_success_and_failure_logging(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test successful and failed execution logging."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="success_test", log_type="test")
|
||||||
|
async def success_func(data: str, user=None):
|
||||||
|
await asyncio.sleep(0.01)
|
||||||
|
return {"status": "success", "data": data}
|
||||||
|
|
||||||
|
@log_usage(function_name="fail_test", log_type="test")
|
||||||
|
async def fail_func(user=None):
|
||||||
|
raise ValueError("Test error")
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
# Test success
|
||||||
|
result = await success_func("test_data", user=test_user)
|
||||||
|
assert result == {"status": "success", "data": "test_data"}
|
||||||
|
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=2)
|
||||||
|
success_log = logs[0]
|
||||||
|
assert success_log["success"] is True
|
||||||
|
assert success_log["error"] is None
|
||||||
|
assert success_log["result"]["status"] == "success"
|
||||||
|
assert success_log["duration_ms"] > 0
|
||||||
|
|
||||||
|
# Test failure
|
||||||
|
with pytest.raises(ValueError, match="Test error"):
|
||||||
|
await fail_func(user=test_user)
|
||||||
|
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=2)
|
||||||
|
fail_log = logs[0]
|
||||||
|
assert fail_log["success"] is False
|
||||||
|
assert fail_log["error"] == "Test error"
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_timing_and_multiple_calls(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test timing accuracy and multiple consecutive calls."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="timing_test", log_type="test")
|
||||||
|
async def timing_func(user=None):
|
||||||
|
await asyncio.sleep(0.1)
|
||||||
|
return "done"
|
||||||
|
|
||||||
|
@log_usage(function_name="multi_test", log_type="test")
|
||||||
|
async def multi_func(call_num: int, user=None):
|
||||||
|
return {"call": call_num}
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
# Test timing
|
||||||
|
await timing_func(user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=1)
|
||||||
|
assert 50 <= logs[0]["duration_ms"] <= 200
|
||||||
|
|
||||||
|
# Test multiple calls
|
||||||
|
for i in range(3):
|
||||||
|
await multi_func(i, user=test_user)
|
||||||
|
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert len(logs) >= 3
|
||||||
|
call_nums = [log["parameters"]["call_num"] for log in logs[:3]]
|
||||||
|
assert set(call_nums) == {0, 1, 2}
|
||||||
|
|
||||||
|
|
||||||
|
class TestRealRedisIntegration:
|
||||||
|
"""Test real Redis integration."""
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_redis_storage_and_retrieval(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test logs are stored in Redis and can be retrieved with correct order and limits."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="redis_test", log_type="test")
|
||||||
|
async def redis_func(data: str, user=None):
|
||||||
|
return {"processed": data}
|
||||||
|
|
||||||
|
@log_usage(function_name="order_test", log_type="test")
|
||||||
|
async def order_func(num: int, user=None):
|
||||||
|
return {"num": num}
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
# Test storage
|
||||||
|
await redis_func("test_data", user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert len(logs) > 0
|
||||||
|
assert logs[0]["function_name"] == "redis_test"
|
||||||
|
assert logs[0]["parameters"]["data"] == "test_data"
|
||||||
|
|
||||||
|
# Test order (most recent first)
|
||||||
|
for i in range(3):
|
||||||
|
await order_func(i, user=test_user)
|
||||||
|
await asyncio.sleep(0.01)
|
||||||
|
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert logs[0]["parameters"]["num"] == 2
|
||||||
|
assert logs[1]["parameters"]["num"] == 1
|
||||||
|
assert logs[2]["parameters"]["num"] == 0
|
||||||
|
|
||||||
|
# Test limit parameter
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=2)
|
||||||
|
assert len(logs) == 2
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_ttl_set_correctly(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test that TTL is set correctly on Redis keys."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="ttl_test", log_type="test")
|
||||||
|
async def ttl_func(user=None):
|
||||||
|
return "result"
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
await ttl_func(user=test_user)
|
||||||
|
|
||||||
|
key = f"test_usage_logs:test-user-123"
|
||||||
|
ttl = await redis_adapter.async_redis.ttl(key)
|
||||||
|
assert ttl > 0
|
||||||
|
assert ttl <= 604800
|
||||||
|
|
||||||
|
|
||||||
|
class TestEdgeCases:
|
||||||
|
"""Test edge cases in integration tests."""
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_edge_cases(
|
||||||
|
self, usage_logging_config, redis_adapter, test_user
|
||||||
|
):
|
||||||
|
"""Test various edge cases: no params, defaults, complex structures, exceptions, None, circular refs."""
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
@log_usage(function_name="no_params", log_type="test")
|
||||||
|
async def no_params_func(user=None):
|
||||||
|
return "result"
|
||||||
|
|
||||||
|
@log_usage(function_name="defaults_only", log_type="test")
|
||||||
|
async def defaults_only_func(param1: str = "default1", param2: int = 42, user=None):
|
||||||
|
return {"param1": param1, "param2": param2}
|
||||||
|
|
||||||
|
@log_usage(function_name="complex_test", log_type="test")
|
||||||
|
async def complex_func(user=None):
|
||||||
|
return {
|
||||||
|
"nested": {
|
||||||
|
"list": [1, 2, 3],
|
||||||
|
"uuid": UUID("123e4567-e89b-12d3-a456-426614174000"),
|
||||||
|
"datetime": datetime(2024, 1, 15, tzinfo=timezone.utc),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@log_usage(function_name="exception_test", log_type="test")
|
||||||
|
async def exception_func(user=None):
|
||||||
|
raise RuntimeError("Test exception")
|
||||||
|
|
||||||
|
@log_usage(function_name="none_test", log_type="test")
|
||||||
|
async def none_func(user=None):
|
||||||
|
return None
|
||||||
|
|
||||||
|
@log_usage(function_name="circular_test", log_type="test")
|
||||||
|
async def circular_func(user=None):
|
||||||
|
a = []
|
||||||
|
a.append(a)
|
||||||
|
return a
|
||||||
|
|
||||||
|
with patch("cognee.shared.usage_logger.get_cache_engine") as mock_get:
|
||||||
|
mock_get.return_value = redis_adapter
|
||||||
|
|
||||||
|
# No parameters
|
||||||
|
await no_params_func(user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert logs[0]["parameters"] == {}
|
||||||
|
|
||||||
|
# Default parameters
|
||||||
|
await defaults_only_func(user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert logs[0]["parameters"]["param1"] == "default1"
|
||||||
|
assert logs[0]["parameters"]["param2"] == 42
|
||||||
|
|
||||||
|
# Complex nested structures
|
||||||
|
result = await complex_func(user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert "nested" in logs[0]["result"]
|
||||||
|
assert isinstance(logs[0]["result"]["nested"]["uuid"], str)
|
||||||
|
assert isinstance(logs[0]["result"]["nested"]["datetime"], str)
|
||||||
|
|
||||||
|
# Exception handling
|
||||||
|
with pytest.raises(RuntimeError):
|
||||||
|
await exception_func(user=test_user)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert logs[0]["success"] is False
|
||||||
|
assert "Test exception" in logs[0]["error"]
|
||||||
|
|
||||||
|
# None return value
|
||||||
|
result = await none_func(user=test_user)
|
||||||
|
assert result is None
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert logs[0]["result"] is None
|
||||||
|
|
||||||
|
# Circular reference
|
||||||
|
result = await circular_func(user=test_user)
|
||||||
|
assert isinstance(result, list)
|
||||||
|
logs = await redis_adapter.get_usage_logs("test-user-123", limit=10)
|
||||||
|
assert "result" in logs[0]
|
||||||
Loading…
Add table
Reference in a new issue