Files
Remote-Terminal-for-MeshCore/tests/test_key_normalization.py

120 lines
3.8 KiB
Python

"""Tests for public key case normalization."""
import pytest
from app.database import Database
from app.repository import ContactRepository, MessageRepository
@pytest.fixture
async def test_db():
"""Create an in-memory test database."""
import app.repository as repo_module
db = Database(":memory:")
await db.connect()
original_db = repo_module.db
repo_module.db = db
try:
yield db
finally:
repo_module.db = original_db
await db.disconnect()
@pytest.mark.asyncio
async def test_upsert_stores_lowercase_key(test_db):
await ContactRepository.upsert(
{"public_key": "A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2"}
)
contact = await ContactRepository.get_by_key(
"a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"
)
assert contact is not None
assert contact.public_key == "a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"
@pytest.mark.asyncio
async def test_get_by_key_case_insensitive(test_db):
await ContactRepository.upsert(
{"public_key": "a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"}
)
contact = await ContactRepository.get_by_key(
"A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4E5F6A1B2"
)
assert contact is not None
@pytest.mark.asyncio
async def test_update_last_contacted_case_insensitive(test_db):
key = "a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"
await ContactRepository.upsert({"public_key": key})
await ContactRepository.update_last_contacted(key.upper(), 12345)
contact = await ContactRepository.get_by_key(key)
assert contact is not None
assert contact.last_contacted == 12345
@pytest.mark.asyncio
async def test_get_by_pubkey_first_byte(test_db):
key1 = "a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"
key2 = "a1ffddeeaabb1122334455667788990011223344556677889900aabbccddeeff00"
key3 = "b2b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2c3d4e5f6a1b2"
for key in [key1, key2, key3]:
await ContactRepository.upsert({"public_key": key})
results = await ContactRepository.get_by_pubkey_first_byte("a1")
assert len(results) == 2
result_keys = {c.public_key for c in results}
assert key1 in result_keys
assert key2 in result_keys
results = await ContactRepository.get_by_pubkey_first_byte("A1")
assert len(results) == 2 # case insensitive
@pytest.mark.asyncio
async def test_null_sender_timestamp_defaults_to_received_at(test_db):
"""Verify that a None/0 sender_timestamp is replaced by received_at."""
msg_id = await MessageRepository.create(
msg_type="PRIV",
text="hello",
conversation_key="abcd1234" * 8,
sender_timestamp=500, # simulates fallback: `payload.get("sender_timestamp") or received_at`
received_at=500,
)
assert msg_id is not None
messages = await MessageRepository.get_all(
msg_type="PRIV", conversation_key="abcd1234" * 8, limit=10
)
assert len(messages) == 1
assert messages[0].sender_timestamp == 500
@pytest.mark.asyncio
async def test_duplicate_with_same_text_and_null_timestamp_rejected(test_db):
"""Two messages with same content and sender_timestamp should be deduped."""
received_at = 600
msg_id1 = await MessageRepository.create(
msg_type="PRIV",
text="hello",
conversation_key="abcd1234" * 8,
sender_timestamp=received_at,
received_at=received_at,
)
assert msg_id1 is not None
msg_id2 = await MessageRepository.create(
msg_type="PRIV",
text="hello",
conversation_key="abcd1234" * 8,
sender_timestamp=received_at,
received_at=received_at,
)
assert msg_id2 is None # duplicate rejected