mirror of
https://github.com/getzep/graphiti.git
synced 2025-11-12 16:27:42 +00:00
* Prepare code * Fix tests * As -> AS, remove trailing spaces * Enable more tests for FalkorDB * Fix more cypher queries * Return all created nodes and edges * Add Neo4j service to unit tests workflow - Introduced Neo4j as a service in the GitHub Actions workflow for unit tests. - Configured Neo4j with appropriate ports, authentication, and health checks. - Updated test steps to include waiting for Neo4j and running integration tests against it. - Set environment variables for Neo4j connection in both non-integration and integration test steps. * Update Neo4j authentication in unit tests workflow - Changed Neo4j authentication password from 'test' to 'testpass' in the GitHub Actions workflow. - Updated health check command to reflect the new password. - Ensured consistency across all test steps that utilize Neo4j credentials. * fix health check * Fix Neo4j integration tests in CI workflow Remove reference to non-existent test_neo4j_driver.py file from test command. Integration tests now run via parametrized tests using the drivers list. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> * Add OPENAI_API_KEY to Neo4j integration tests Neo4j integration tests require OpenAI API access for LLM functionality. Add the secret environment variable to enable these tests to run properly. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> * Fix Neo4j Cypher syntax error in BFS search queries Replace parameter substitution in relationship pattern ranges (*1..$depth) with direct string interpolation (*1..{bfs_max_depth}). Neo4j doesn't allow parameter maps in MATCH pattern ranges - they must be literal values. Fixed in both node_bfs_search and edge_bfs_search functions. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> * Fix variable name mismatch in edge_bfs_search query Change relationship variable from 'r' to 'e' to match ENTITY_EDGE_RETURN constant expectations. The ENTITY_EDGE_RETURN constant references variable 'e' for relationships, but the query was using 'r', causing "Variable e not defined" errors. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> * Isolate database tests in CI workflow - FalkorDB tests: Add DISABLE_NEO4J=1 and remove Neo4j env vars - Neo4j tests: Keep current setup without DISABLE_NEO4J flag This ensures proper test isolation where each test suite only runs against its intended database backend. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> --------- Co-authored-by: Siddhartha Sahu <sid@kuzudb.com> Co-authored-by: Claude <noreply@anthropic.com>
244 lines
7.6 KiB
Python
244 lines
7.6 KiB
Python
"""
|
|
Copyright 2024, Zep Software, Inc.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
"""
|
|
|
|
from datetime import datetime
|
|
from uuid import uuid4
|
|
|
|
import numpy as np
|
|
import pytest
|
|
|
|
from graphiti_core.driver.driver import GraphDriver
|
|
from graphiti_core.nodes import (
|
|
CommunityNode,
|
|
EntityNode,
|
|
EpisodeType,
|
|
EpisodicNode,
|
|
)
|
|
from tests.helpers_test import drivers, get_driver
|
|
|
|
group_id = f'test_group_{str(uuid4())}'
|
|
|
|
|
|
@pytest.fixture
|
|
def sample_entity_node():
|
|
return EntityNode(
|
|
uuid=str(uuid4()),
|
|
name='Test Entity',
|
|
group_id=group_id,
|
|
labels=[],
|
|
name_embedding=[0.5] * 1024,
|
|
summary='Entity Summary',
|
|
)
|
|
|
|
|
|
@pytest.fixture
|
|
def sample_episodic_node():
|
|
return EpisodicNode(
|
|
uuid=str(uuid4()),
|
|
name='Episode 1',
|
|
group_id=group_id,
|
|
source=EpisodeType.text,
|
|
source_description='Test source',
|
|
content='Some content here',
|
|
valid_at=datetime.now(),
|
|
)
|
|
|
|
|
|
@pytest.fixture
|
|
def sample_community_node():
|
|
return CommunityNode(
|
|
uuid=str(uuid4()),
|
|
name='Community A',
|
|
name_embedding=[0.5] * 1024,
|
|
group_id=group_id,
|
|
summary='Community summary',
|
|
)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
@pytest.mark.parametrize(
|
|
'driver',
|
|
drivers,
|
|
ids=drivers,
|
|
)
|
|
async def test_entity_node(sample_entity_node, driver):
|
|
driver = get_driver(driver)
|
|
uuid = sample_entity_node.uuid
|
|
|
|
# Create node
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
await sample_entity_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
|
|
retrieved = await EntityNode.get_by_uuid(driver, sample_entity_node.uuid)
|
|
assert retrieved.uuid == sample_entity_node.uuid
|
|
assert retrieved.name == 'Test Entity'
|
|
assert retrieved.group_id == group_id
|
|
|
|
retrieved = await EntityNode.get_by_uuids(driver, [sample_entity_node.uuid])
|
|
assert retrieved[0].uuid == sample_entity_node.uuid
|
|
assert retrieved[0].name == 'Test Entity'
|
|
assert retrieved[0].group_id == group_id
|
|
|
|
retrieved = await EntityNode.get_by_group_ids(driver, [group_id], limit=2)
|
|
assert len(retrieved) == 1
|
|
assert retrieved[0].uuid == sample_entity_node.uuid
|
|
assert retrieved[0].name == 'Test Entity'
|
|
assert retrieved[0].group_id == group_id
|
|
|
|
await sample_entity_node.load_name_embedding(driver)
|
|
assert np.allclose(sample_entity_node.name_embedding, [0.5] * 1024)
|
|
|
|
# Delete node by uuid
|
|
await sample_entity_node.delete(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
# Delete node by group id
|
|
await sample_entity_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
await sample_entity_node.delete_by_group_id(driver, group_id)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
await driver.close()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
@pytest.mark.parametrize(
|
|
'driver',
|
|
drivers,
|
|
ids=drivers,
|
|
)
|
|
async def test_community_node(sample_community_node, driver):
|
|
driver = get_driver(driver)
|
|
uuid = sample_community_node.uuid
|
|
|
|
# Create node
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
await sample_community_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
|
|
retrieved = await CommunityNode.get_by_uuid(driver, sample_community_node.uuid)
|
|
assert retrieved.uuid == sample_community_node.uuid
|
|
assert retrieved.name == 'Community A'
|
|
assert retrieved.group_id == group_id
|
|
assert retrieved.summary == 'Community summary'
|
|
|
|
retrieved = await CommunityNode.get_by_uuids(driver, [sample_community_node.uuid])
|
|
assert retrieved[0].uuid == sample_community_node.uuid
|
|
assert retrieved[0].name == 'Community A'
|
|
assert retrieved[0].group_id == group_id
|
|
assert retrieved[0].summary == 'Community summary'
|
|
|
|
retrieved = await CommunityNode.get_by_group_ids(driver, [group_id], limit=2)
|
|
assert len(retrieved) == 1
|
|
assert retrieved[0].uuid == sample_community_node.uuid
|
|
assert retrieved[0].name == 'Community A'
|
|
assert retrieved[0].group_id == group_id
|
|
|
|
# Delete node by uuid
|
|
await sample_community_node.delete(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
# Delete node by group id
|
|
await sample_community_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
await sample_community_node.delete_by_group_id(driver, group_id)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
await driver.close()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
@pytest.mark.parametrize(
|
|
'driver',
|
|
drivers,
|
|
ids=drivers,
|
|
)
|
|
async def test_episodic_node(sample_episodic_node, driver):
|
|
driver = get_driver(driver)
|
|
uuid = sample_episodic_node.uuid
|
|
|
|
# Create node
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
await sample_episodic_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
|
|
retrieved = await EpisodicNode.get_by_uuid(driver, sample_episodic_node.uuid)
|
|
assert retrieved.uuid == sample_episodic_node.uuid
|
|
assert retrieved.name == 'Episode 1'
|
|
assert retrieved.group_id == group_id
|
|
assert retrieved.source == EpisodeType.text
|
|
assert retrieved.source_description == 'Test source'
|
|
assert retrieved.content == 'Some content here'
|
|
assert retrieved.valid_at == sample_episodic_node.valid_at
|
|
|
|
retrieved = await EpisodicNode.get_by_uuids(driver, [sample_episodic_node.uuid])
|
|
assert retrieved[0].uuid == sample_episodic_node.uuid
|
|
assert retrieved[0].name == 'Episode 1'
|
|
assert retrieved[0].group_id == group_id
|
|
assert retrieved[0].source == EpisodeType.text
|
|
assert retrieved[0].source_description == 'Test source'
|
|
assert retrieved[0].content == 'Some content here'
|
|
assert retrieved[0].valid_at == sample_episodic_node.valid_at
|
|
|
|
retrieved = await EpisodicNode.get_by_group_ids(driver, [group_id], limit=2)
|
|
assert len(retrieved) == 1
|
|
assert retrieved[0].uuid == sample_episodic_node.uuid
|
|
assert retrieved[0].name == 'Episode 1'
|
|
assert retrieved[0].group_id == group_id
|
|
assert retrieved[0].source == EpisodeType.text
|
|
assert retrieved[0].source_description == 'Test source'
|
|
assert retrieved[0].content == 'Some content here'
|
|
assert retrieved[0].valid_at == sample_episodic_node.valid_at
|
|
|
|
# Delete node by uuid
|
|
await sample_episodic_node.delete(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
# Delete node by group id
|
|
await sample_episodic_node.save(driver)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 1
|
|
await sample_episodic_node.delete_by_group_id(driver, group_id)
|
|
node_count = await get_node_count(driver, uuid)
|
|
assert node_count == 0
|
|
|
|
await driver.close()
|
|
|
|
|
|
async def get_node_count(driver: GraphDriver, uuid: str):
|
|
result, _, _ = await driver.execute_query(
|
|
"""
|
|
MATCH (n {uuid: $uuid})
|
|
RETURN COUNT(n) as count
|
|
""",
|
|
uuid=uuid,
|
|
)
|
|
return int(result[0]['count'])
|