fix: task check — SSL error in integration test + coverage gate

account_sync_manager_test: inject _connectImapPlain so the test
connects to the plain-IMAP dev Stalwart without triggering the
production SSL guard in connectImap().

check_coverage: add sieve_script_edit_screen, sieve_scripts_screen,
thread_detail_screen, and sieve_repository to _excluded (screens and
JMAP client without unit tests, consistent with existing exclusions).

New tests restore the 80% coverage gate:
- sieve_script_test: trivial model construction
- mailbox_repository_impl_test: findMailboxByRole (found + not found),
  syncMailboxes with no jmapUrl, syncMailboxes with JMAP error response
- try_connection_button_test: okMessage and errorMessage rendering
- email_list_screen_test: selection mode, deselect via checkbox,
  search-clear button, search-result tap, preview snippet
- helpers: pass email.preview through to EmailThread in FakeEmailRepository

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Thomas Güttler
2026-04-25 06:38:21 +02:00
co-authored by Claude Sonnet 4.6
parent 681e0c0167
commit 92a8a79952
8 changed files with 368 additions and 0 deletions
+9
View File
@@ -6,6 +6,15 @@ Tasks get moved from next.md to done.md
## Tasks
## Fix task check: unencrypted IMAP error + coverage gate
- `account_sync_manager_test.dart`: inject `_connectImapPlain` (bypasses the production SSL check) so the test works against the plain-IMAP dev Stalwart.
- `scripts/check_coverage.dart`: add three new screens (`sieve_script_edit_screen`, `sieve_scripts_screen`, `thread_detail_screen`) and `sieve_repository` to `_excluded` (all are screens/JMAP clients without unit tests).
- New unit tests: `sieve_script_test.dart`, plus `findMailboxByRole`, JMAP no-URL error, and JMAP API error tests in `mailbox_repository_impl_test.dart`.
- New widget tests: `try_connection_button_test.dart` (okMessage/errorMessage rendering) plus selection-mode, deselect, search-clear, and search-result-tap tests in `email_list_screen_test.dart`.
- Fixed `FakeEmailRepository.observeThreads` in `helpers.dart` to propagate `preview` from email to thread.
- Coverage gate now passes at 80%+ (84% with integration coverage merged).
## Android integration test via Stalwart
Added `stalwart-dev/integration_android_test.sh` and `task integration-android`. Starts Stalwart on random ports, detects a connected emulator via `adb devices`, sets `STALWART_IMAP_HOST=10.0.2.2` (emulator host alias), and runs the existing `integration_test/app_e2e_test.dart` on the emulator.
+4
View File
@@ -47,12 +47,16 @@ const _excluded = {
'lib/ui/screens/email_detail_screen.dart',
'lib/ui/screens/mailbox_list_screen.dart',
'lib/ui/screens/search_screen.dart',
'lib/ui/screens/sieve_script_edit_screen.dart',
'lib/ui/screens/sieve_scripts_screen.dart',
'lib/ui/screens/sync_log_screen.dart',
'lib/ui/screens/thread_detail_screen.dart',
'lib/ui/widgets/folder_drawer.dart',
// Repositories and sync orchestration that are exercised primarily through
// integration tests against real servers.
'lib/core/sync/account_sync_manager.dart',
'lib/data/jmap/jmap_client.dart',
'lib/data/jmap/sieve_repository.dart',
'lib/data/repositories/account_repository_impl.dart',
'lib/data/repositories/email_repository_impl.dart',
'lib/data/repositories/sync_log_repository_impl.dart',
@@ -7,6 +7,7 @@
import 'dart:async';
import 'dart:io';
import 'package:enough_mail/enough_mail.dart' show ImapClient;
import 'package:flutter_test/flutter_test.dart';
import 'package:sharedinbox/core/models/account.dart';
@@ -128,6 +129,21 @@ class _FakeEmails implements EmailRepository {
Future<void> retryMutation(int id) async {}
}
// Plain (non-TLS) IMAP connect for the local dev Stalwart, which has no TLS.
// Production connectImap() rejects imapSsl:false, so tests inject this instead.
Future<ImapClient> _connectImapPlain(
Account account,
String username,
String password,
) async {
final client = ImapClient(
defaultResponseTimeout: const Duration(seconds: 20),
);
await client.connectToServer(account.imapHost, account.imapPort);
await client.login(username, password);
return client;
}
// ── Tests ─────────────────────────────────────────────────────────────────────
void main() {
@@ -165,6 +181,7 @@ void main() {
fakeAccounts,
_FakeMailboxes(),
_FakeEmails(),
imapConnect: _connectImapPlain,
);
mgr.start();
@@ -8,6 +8,7 @@ import 'package:http/testing.dart';
import 'package:sharedinbox/core/models/account.dart';
import 'package:sharedinbox/data/db/database.dart' hide Account;
import 'package:sharedinbox/data/jmap/jmap_client.dart' show JmapException;
import 'package:sharedinbox/data/repositories/account_repository_impl.dart';
import 'package:sharedinbox/data/repositories/mailbox_repository_impl.dart';
@@ -360,6 +361,71 @@ void main() {
final state = await r.db.select(r.db.syncStates).get();
expect(state.first.state, 'st1');
});
test('syncMailboxes throws when JMAP account has no jmapUrl', () async {
const noUrlAccount = Account(
id: 'jmap-no-url',
displayName: 'NoUrl',
email: 'nourl@example.com',
type: AccountType.jmap,
);
final r = _makeRepos();
await r.accounts.addAccount(noUrlAccount, 'pw');
await expectLater(
r.mailboxes.syncMailboxes('jmap-no-url'),
throwsA(isA<Exception>()),
);
});
test('syncMailboxes throws JmapException on API error response',
() async {
final r = _makeRepos(
httpClient: _mockJmap(
apiResponses: [
{
'sessionState': 'sess1',
'methodResponses': [
[
'error',
<String, dynamic>{'type': 'serverFail'},
'0',
],
],
},
],
),
);
await r.accounts.addAccount(_jmapAccount, 'pw');
await expectLater(
r.mailboxes.syncMailboxes('jmap-1'),
throwsA(isA<JmapException>()),
);
});
});
test('findMailboxByRole returns null when no matching mailbox', () async {
final r = _makeRepos();
await r.accounts.addAccount(_jmapAccount, 'pw');
final result = await r.mailboxes.findMailboxByRole('jmap-1', 'inbox');
expect(result, isNull);
});
test('findMailboxByRole returns matching mailbox', () async {
final r = _makeRepos();
await r.accounts.addAccount(_jmapAccount, 'pw');
await r.db.into(r.db.mailboxes).insert(
MailboxesCompanion.insert(
id: 'jmap-1:mbx-inbox',
accountId: 'jmap-1',
path: 'INBOX',
name: 'Inbox',
role: const Value('inbox'),
),
);
final result = await r.mailboxes.findMailboxByRole('jmap-1', 'inbox');
expect(result, isNotNull);
expect(result!.role, 'inbox');
});
});
}
+18
View File
@@ -0,0 +1,18 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:sharedinbox/core/models/sieve_script.dart';
void main() {
test('SieveScript holds fields', () {
const s = SieveScript(
id: 'id1',
name: 'My filter',
blobId: 'blob1',
isActive: true,
);
expect(s.id, 'id1');
expect(s.name, 'My filter');
expect(s.blobId, 'blob1');
expect(s.isActive, true);
});
}
+192
View File
@@ -1,10 +1,13 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:sharedinbox/core/models/email.dart';
import 'package:sharedinbox/di.dart';
import 'helpers.dart';
final _kDate = DateTime(2024, 6);
void main() {
group('EmailListScreen', () {
testWidgets('shows "No emails" when list is empty', (tester) async {
@@ -210,5 +213,194 @@ void main() {
expect(find.text('Search…'), findsNothing);
expect(find.text('INBOX'), findsOneWidget);
});
testWidgets('long-press enters selection mode with selection bar',
(tester) async {
final email = testEmail(subject: 'Select me');
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider
.overrideWithValue(FakeEmailRepository(emails: [email])),
],
),
);
await tester.pumpAndSettle();
await tester.longPress(find.text('Select me'));
await tester.pumpAndSettle();
expect(find.text('1 selected'), findsOneWidget);
expect(find.byType(BottomAppBar), findsOneWidget);
expect(find.byIcon(Icons.close), findsOneWidget);
});
testWidgets('selection bar close button exits selection mode',
(tester) async {
final email = testEmail(subject: 'Select me');
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider
.overrideWithValue(FakeEmailRepository(emails: [email])),
],
),
);
await tester.pumpAndSettle();
await tester.longPress(find.text('Select me'));
await tester.pumpAndSettle();
await tester.tap(find.byIcon(Icons.close));
await tester.pumpAndSettle();
expect(find.text('INBOX'), findsOneWidget);
expect(find.byType(BottomAppBar), findsNothing);
});
testWidgets('tapping clear icon in search bar clears results',
(tester) async {
final email = testEmail(subject: 'Found it');
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider.overrideWithValue(
FakeEmailRepository(emails: [email], searchResults: [email]),
),
],
),
);
await tester.pumpAndSettle();
await tester.tap(find.byIcon(Icons.search));
await tester.pumpAndSettle();
await tester.enterText(find.byType(TextField), 'hello');
await tester.testTextInput.receiveAction(TextInputAction.search);
await tester.pumpAndSettle();
expect(find.text('Found it'), findsOneWidget);
await tester.tap(find.byIcon(Icons.clear));
await tester.pumpAndSettle();
expect(find.text('Found it'), findsNothing);
});
testWidgets('tapping selected-email checkbox deselects it', (tester) async {
final email = testEmail(subject: 'Toggle me');
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider
.overrideWithValue(FakeEmailRepository(emails: [email])),
],
),
);
await tester.pumpAndSettle();
await tester.longPress(find.text('Toggle me'));
await tester.pumpAndSettle();
expect(find.text('1 selected'), findsOneWidget);
await tester.tap(find.byType(Checkbox));
await tester.pumpAndSettle();
// Deselecting the only email exits selection mode automatically.
expect(find.text('INBOX'), findsOneWidget);
});
testWidgets('tapping a search result navigates to email detail',
(tester) async {
final email = testEmail(subject: 'Result email');
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider.overrideWithValue(
FakeEmailRepository(
searchResults: [email],
emailDetail: email,
emailBody:
const EmailBody(emailId: 'acc-1:42', attachments: []),
),
),
],
),
);
await tester.pumpAndSettle();
await tester.tap(find.byIcon(Icons.search));
await tester.pumpAndSettle();
await tester.enterText(find.byType(TextField), 'Result');
await tester.testTextInput.receiveAction(TextInputAction.search);
await tester.pumpAndSettle();
await tester.tap(find.text('Result email'));
await tester.pumpAndSettle();
// Navigated to email detail (subject appears in the detail body)
expect(find.text('Result email'), findsWidgets);
});
testWidgets('shows preview snippet when email has preview', (tester) async {
final email = Email(
id: 'acc-1:99',
accountId: 'acc-1',
mailboxPath: 'INBOX',
uid: 99,
subject: 'Hello',
receivedAt: _kDate,
sentAt: _kDate,
from: const [EmailAddress(name: 'Bob', email: 'bob@example.com')],
to: const [EmailAddress(email: 'alice@example.com')],
cc: [],
preview: 'This is the preview text',
isSeen: false,
isFlagged: false,
hasAttachment: false,
);
await tester.pumpWidget(
buildApp(
initialLocation: '/accounts/acc-1/mailboxes/INBOX/emails',
overrides: [
accountRepositoryProvider
.overrideWithValue(FakeAccountRepository([kTestAccount])),
mailboxRepositoryProvider
.overrideWithValue(FakeMailboxRepository()),
emailRepositoryProvider
.overrideWithValue(FakeEmailRepository(emails: [email])),
],
),
);
await tester.pumpAndSettle();
expect(find.text('This is the preview text'), findsOneWidget);
});
});
}
+1
View File
@@ -164,6 +164,7 @@ class FakeEmailRepository implements EmailRepository {
return EmailThread(
threadId: e.threadId ?? e.id,
subject: e.subject,
preview: e.preview,
participants: e.from,
latestDate: e.sentAt ?? e.receivedAt,
messageCount: 1,
@@ -0,0 +1,61 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:sharedinbox/ui/widgets/try_connection_button.dart';
Widget _wrap(Widget child) => MaterialApp(
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.indigo),
useMaterial3: true,
),
home: Scaffold(body: child),
);
void main() {
group('TryConnectionButton', () {
testWidgets('shows "Try connection" button when idle', (tester) async {
await tester.pumpWidget(
_wrap(
const TryConnectionButton(testing: false, onPressed: null),
),
);
expect(find.text('Try connection'), findsOneWidget);
});
testWidgets('shows spinner when testing', (tester) async {
await tester.pumpWidget(
_wrap(
const TryConnectionButton(testing: true, onPressed: null),
),
);
expect(find.byType(CircularProgressIndicator), findsOneWidget);
expect(find.text('Try connection'), findsNothing);
});
testWidgets('shows okMessage when provided', (tester) async {
await tester.pumpWidget(
_wrap(
const TryConnectionButton(
testing: false,
onPressed: null,
okMessage: 'Connected!',
),
),
);
expect(find.text('Connected!'), findsOneWidget);
});
testWidgets('shows errorMessage when provided', (tester) async {
await tester.pumpWidget(
_wrap(
const TryConnectionButton(
testing: false,
onPressed: null,
errorMessage: 'Connection failed',
),
),
);
expect(find.text('Connection failed'), findsOneWidget);
});
});
}