3.6 KiB
Mailbox Report Tutorial
This tutorial shows how to generate an email-channel evidence report from a return mailbox or from the bundled fixture mailbox.
1. Verify The Scanner
Run the tests and print the adapter descriptor:
PYTHONPATH=src python3 -m unittest discover -s tests
PYTHONPATH=src python3 -m email_connect.cli adapter-descriptor
2. Start With Fixtures
The example config uses tests/fixtures/mailbox:
PYTHONPATH=src python3 -m email_connect.cli scan-mailbox \
--config config/mailbox.example.yml \
--full-rescan \
--out reports/
The scanner writes a timestamped CSV file to reports/.
3. Configure A Live IMAP Mailbox
Copy config/mailbox.example.yml and set:
mailbox:
protocol: imap
host: imap.example.com
port: 993
tls: true
username_env: EMAIL_CONNECT_IMAP_USER
password_env: EMAIL_CONNECT_IMAP_PASSWORD
folder: INBOX
Then export credentials:
export EMAIL_CONNECT_IMAP_USER='mailbox@example.com'
export EMAIL_CONNECT_IMAP_PASSWORD='app-password'
IMAP scans select the folder read-only and fetch messages with BODY.PEEK[].
The scanner does not mark messages seen, move messages, or delete messages.
4. Add Expected Recipients
Expected recipients are optional. A newline-separated file can look like:
missing@example.com
recipient@example.com
Run:
PYTHONPATH=src python3 -m email_connect.cli scan-mailbox \
--config config/mailbox.example.yml \
--expected-recipients recipients.txt \
--out reports/
CSV input is also supported:
email,name
missing@example.com,Missing User
recipient@example.com,Known Recipient
Run:
PYTHONPATH=src python3 -m email_connect.cli scan-mailbox \
--config config/mailbox.example.yml \
--expected-recipients recipients.csv \
--expected-recipient-column email \
--out reports/
Invalid recipient rows are ignored and printed as warnings.
5. Limit The Time Range
Use an inclusive datetime range:
PYTHONPATH=src python3 -m email_connect.cli scan-mailbox \
--config config/mailbox.example.yml \
--from 2026-06-02T10:00:00Z \
--to 2026-06-02T11:00:00Z \
--out reports/
--since remains a compatibility alias for the lower bound. When a range is
active, messages with no parseable Date header are excluded because the
scanner cannot confirm that they originated inside the requested window.
6. Read The Report
Key columns:
known_recipient:truewhen the address was supplied in the expected list.normalized_event_type: the email evidence or diagnostic event.assessment_categoryandassessment_subclass: advisory interpretation.affected_email_address: the endpoint the row is about.
Known recipients appear first by default so spreadsheet filtering is easy.
Expected recipients with no mailbox evidence appear as:
normalized_event_type: diagnostic.expected_recipient.no_evidence
assessment_category: undef
assessment_subclass: undef.no_signal
evidence_strength: none
known_recipient: true
That row means only that no mailbox evidence was found for the supplied address inside the inspected range. It is not evidence of delivery success, delivery failure, recipient awareness, or legal acceptance.
7. Troubleshooting
- Empty report: check folder, time range, and whether incremental cursor state
already skipped older messages. Try
--full-rescan. - IMAP credential error: verify the environment variable names and values.
- Missing expected rows: check the recipient file path and CSV column name.
- Unexpected no-evidence rows: confirm that the relevant mailbox evidence is inside the configured datetime range.