mirror of
https://github.com/Monadical-SAS/reflector.git
synced 2025-12-21 12:49:06 +00:00
- Add session fixture for async session management - Update all test files to use session parameter - Convert Core-style queries to ORM-style in tests - Fix controller calls to include session parameter - Remove obsolete get_database() references Test progress: 108/195 tests passing
194 lines
6.6 KiB
Python
194 lines
6.6 KiB
Python
import os
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
import pytest
|
|
|
|
from reflector.db.rooms import rooms_controller
|
|
from reflector.services.ics_sync import ICSSyncService
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_attendee_parsing_bug():
|
|
"""
|
|
Test that reproduces the attendee parsing bug where a string with comma-separated
|
|
emails gets parsed as individual characters instead of separate email addresses.
|
|
|
|
The bug manifests as getting 29 attendees with emails like "M", "A", "I", etc.
|
|
instead of properly parsed email addresses.
|
|
"""
|
|
# Create a test room
|
|
room = await rooms_controller.add(
|
|
session,
|
|
name="test-room",
|
|
user_id="test-user",
|
|
zulip_auto_post=False,
|
|
zulip_stream="",
|
|
zulip_topic="",
|
|
is_locked=False,
|
|
room_mode="normal",
|
|
recording_type="cloud",
|
|
recording_trigger="automatic-2nd-participant",
|
|
is_shared=False,
|
|
ics_url="http://test.com/test.ics",
|
|
ics_enabled=True,
|
|
)
|
|
|
|
# Read the test ICS file that reproduces the bug and update it with current time
|
|
from datetime import datetime, timedelta, timezone
|
|
|
|
test_ics_path = os.path.join(
|
|
os.path.dirname(__file__), "test_attendee_parsing_bug.ics"
|
|
)
|
|
with open(test_ics_path, "r") as f:
|
|
ics_content = f.read()
|
|
|
|
# Replace the dates with current time + 1 hour to ensure it's within the 24h window
|
|
now = datetime.now(timezone.utc)
|
|
future_time = now + timedelta(hours=1)
|
|
end_time = future_time + timedelta(hours=1)
|
|
|
|
# Format dates for ICS format
|
|
dtstart = future_time.strftime("%Y%m%dT%H%M%SZ")
|
|
dtend = end_time.strftime("%Y%m%dT%H%M%SZ")
|
|
dtstamp = now.strftime("%Y%m%dT%H%M%SZ")
|
|
|
|
# Update the ICS content with current dates
|
|
ics_content = ics_content.replace("20250910T180000Z", dtstart)
|
|
ics_content = ics_content.replace("20250910T190000Z", dtend)
|
|
ics_content = ics_content.replace("20250910T174000Z", dtstamp)
|
|
|
|
# Create sync service and mock the fetch
|
|
sync_service = ICSSyncService()
|
|
|
|
with patch.object(
|
|
sync_service.fetch_service, "fetch_ics", new_callable=AsyncMock
|
|
) as mock_fetch:
|
|
mock_fetch.return_value = ics_content
|
|
|
|
# Debug: Parse the ICS content directly to examine attendee parsing
|
|
calendar = sync_service.fetch_service.parse_ics(ics_content)
|
|
from reflector.settings import settings
|
|
|
|
room_url = f"{settings.UI_BASE_URL}/{room.name}"
|
|
|
|
print(f"Room URL being used for matching: {room_url}")
|
|
print(f"ICS content:\n{ics_content}")
|
|
|
|
events, total_events = sync_service.fetch_service.extract_room_events(
|
|
calendar, room.name, room_url
|
|
)
|
|
|
|
print(f"Total events in calendar: {total_events}")
|
|
print(f"Events matching room: {len(events)}")
|
|
|
|
# Perform the sync
|
|
result = await sync_service.sync_room_calendar(room)
|
|
|
|
# Check that the sync succeeded
|
|
assert result.get("status") == "success"
|
|
assert result.get("events_found", 0) >= 0 # Allow for debugging
|
|
|
|
# We already have the matching events from the debug code above
|
|
assert len(events) == 1
|
|
event = events[0]
|
|
|
|
# This is where the bug manifests - check the attendees
|
|
attendees = event["attendees"]
|
|
|
|
# Print attendee info for debugging
|
|
print(f"Number of attendees found: {len(attendees)}")
|
|
for i, attendee in enumerate(attendees):
|
|
print(
|
|
f"Attendee {i}: email='{attendee.get('email')}', name='{attendee.get('name')}'"
|
|
)
|
|
|
|
# With the fix, we should now get properly parsed email addresses
|
|
# Check that no single characters are parsed as emails
|
|
single_char_emails = [
|
|
att for att in attendees if att.get("email") and len(att["email"]) == 1
|
|
]
|
|
|
|
if single_char_emails:
|
|
print(
|
|
f"BUG DETECTED: Found {len(single_char_emails)} single-character emails:"
|
|
)
|
|
for att in single_char_emails:
|
|
print(f" - '{att['email']}'")
|
|
|
|
# Should have attendees but not single-character emails
|
|
assert len(attendees) > 0
|
|
assert (
|
|
len(single_char_emails) == 0
|
|
), f"Found {len(single_char_emails)} single-character emails, parsing is still buggy"
|
|
|
|
# Check that all emails are valid (contain @ symbol)
|
|
valid_emails = [
|
|
att for att in attendees if att.get("email") and "@" in att["email"]
|
|
]
|
|
assert len(valid_emails) == len(
|
|
attendees
|
|
), "Some attendees don't have valid email addresses"
|
|
|
|
# We expect around 29 attendees (28 from the comma-separated list + 1 organizer)
|
|
assert (
|
|
len(attendees) >= 25
|
|
), f"Expected around 29 attendees, got {len(attendees)}"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_correct_attendee_parsing():
|
|
"""
|
|
Test what correct attendee parsing should look like.
|
|
"""
|
|
from datetime import datetime, timezone
|
|
|
|
from icalendar import Event
|
|
|
|
from reflector.services.ics_sync import ICSFetchService
|
|
|
|
service = ICSFetchService()
|
|
|
|
# Create a properly formatted event with multiple attendees
|
|
event = Event()
|
|
event.add("uid", "test-correct-attendees")
|
|
event.add("summary", "Test Meeting")
|
|
event.add("location", "http://test.com/test")
|
|
event.add("dtstart", datetime.now(timezone.utc))
|
|
event.add("dtend", datetime.now(timezone.utc))
|
|
|
|
# Add attendees the correct way (separate ATTENDEE lines)
|
|
event.add("attendee", "mailto:alice@example.com", parameters={"CN": "Alice"})
|
|
event.add("attendee", "mailto:bob@example.com", parameters={"CN": "Bob"})
|
|
event.add("attendee", "mailto:charlie@example.com", parameters={"CN": "Charlie"})
|
|
event.add(
|
|
"organizer", "mailto:organizer@example.com", parameters={"CN": "Organizer"}
|
|
)
|
|
|
|
# Parse the event
|
|
result = service._parse_event(event)
|
|
|
|
assert result is not None
|
|
attendees = result["attendees"]
|
|
|
|
# Should have 4 attendees (3 attendees + 1 organizer)
|
|
assert len(attendees) == 4
|
|
|
|
# Check that all emails are valid email addresses
|
|
emails = [att["email"] for att in attendees if att.get("email")]
|
|
expected_emails = [
|
|
"alice@example.com",
|
|
"bob@example.com",
|
|
"charlie@example.com",
|
|
"organizer@example.com",
|
|
]
|
|
|
|
for email in emails:
|
|
assert "@" in email, f"Invalid email format: {email}"
|
|
assert len(email) > 5, f"Email too short: {email}"
|
|
|
|
# Check that we have the expected emails
|
|
assert "alice@example.com" in emails
|
|
assert "bob@example.com" in emails
|
|
assert "charlie@example.com" in emails
|
|
assert "organizer@example.com" in emails
|