Files
sharedinbox/lib/core/services/connection_test_service.dart
T
Thomas GüttlerandClaude Sonnet 4.6 e3ba18285d refactor: enforce always_use_package_imports across all lib files
Added lint rule to analysis_options.yaml and ran dart fix --apply to convert
125 relative imports in 33 files to package:sharedinbox/... style.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-24 16:30:59 +02:00

113 lines
3.6 KiB
Dart

import 'dart:convert';
import 'package:enough_mail/enough_mail.dart' as imap;
import 'package:http/http.dart' as http;
import 'package:sharedinbox/core/models/account.dart';
import 'package:sharedinbox/data/imap/imap_client_factory.dart';
typedef ImapConnectForTestFn = Future<imap.ImapClient> Function(
Account,
String username,
String password,
);
abstract class ConnectionTestService {
/// Verifies credentials and returns the effective username used.
/// Throws a descriptive [Exception] on failure.
Future<String> testConnection(Account account, String password);
}
class ConnectionTestServiceImpl implements ConnectionTestService {
ConnectionTestServiceImpl(
this._httpClient, {
ImapConnectForTestFn imapConnect = connectImap,
}) : _imapConnect = imapConnect;
final http.Client _httpClient;
final ImapConnectForTestFn _imapConnect;
@override
Future<String> testConnection(Account account, String password) async {
switch (account.type) {
case AccountType.imap:
return _testImap(account, password);
case AccountType.jmap:
return _testJmap(account, password);
}
}
/// Returns the username candidates to try in order.
/// If [account.username] is non-empty it's the only candidate.
/// Otherwise: full email first, then the local part before '@'.
List<String> _usernamesFor(Account account) {
if (account.username.isNotEmpty) return [account.username];
final email = account.email;
final atIdx = email.indexOf('@');
if (atIdx <= 0) return [email];
final localPart = email.substring(0, atIdx);
return [email, localPart];
}
Future<String> _testImap(Account account, String password) async {
final candidates = _usernamesFor(account);
Object? lastError;
for (final username in candidates) {
try {
final client = await _imapConnect(account, username, password);
await client.logout();
return username;
} catch (e) {
lastError = e;
}
}
throw lastError!;
}
Future<String> _testJmap(Account account, String password) async {
final jmapUrl = account.jmapUrl;
if (jmapUrl == null || jmapUrl.isEmpty) {
throw Exception('No JMAP URL configured for this account');
}
final sessionUri = Uri.parse(jmapUrl);
final candidates = _usernamesFor(account);
Object? lastError;
for (final username in candidates) {
try {
final credentials = base64.encode(utf8.encode('$username:$password'));
final resp = await _httpClient.get(
sessionUri,
headers: {
'Authorization': 'Basic $credentials',
},
).timeout(const Duration(seconds: 10));
if (resp.statusCode == 401 || resp.statusCode == 403) {
lastError =
Exception('Authentication failed: wrong username or password');
continue;
}
if (resp.statusCode != 200) {
throw Exception('Connection failed (HTTP ${resp.statusCode})');
}
final Map<String, dynamic> session;
try {
session = jsonDecode(resp.body) as Map<String, dynamic>;
} on FormatException {
throw Exception(
'Not a JMAP server — unexpected response from $jmapUrl',
);
}
final caps = session['capabilities'];
if (caps is! Map || !caps.containsKey('urn:ietf:params:jmap:core')) {
throw Exception(
'Not a JMAP server — missing core capability at $jmapUrl',
);
}
return username;
} catch (e) {
lastError = e;
}
}
throw lastError!;
}
}