Refactor ForexRateAPI integration to load API key from .env file and enhance error handling

Signed-off-by: Hans Kokx <hans.d.kokx@gmail.com>
This commit is contained in:
2026-03-19 23:48:26 +01:00
parent a775a20dd2
commit e6c16967f9
8 changed files with 244 additions and 81 deletions
+1
View File
@@ -0,0 +1 @@
EXCHANGE_RATE_API_KEY=REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY
+4
View File
@@ -43,3 +43,7 @@ app.*.map.json
/android/app/debug /android/app/debug
/android/app/profile /android/app/profile
/android/app/release /android/app/release
# Local runtime configuration
.env
!.env.example
+3 -3
View File
@@ -23,10 +23,10 @@ Simple Flutter app to track US rental income and convert to SEK for Swedish tax
## Setup ## Setup
1. Set your FX API app key in `lib/state/rent_controller.dart`: 1. Create `.env` in the project root with your FX API key:
```dart ```env
const String forexRateApiKey = 'REPLACE_WITH_YOUR_FOREXRATE_API_KEY'; EXCHANGE_RATE_API_KEY=REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY
``` ```
1. Install dependencies: 1. Install dependencies:
+11
View File
@@ -21,6 +21,7 @@ class HomeScreen extends StatelessWidget {
symbol: 'SEK ', symbol: 'SEK ',
decimalDigits: 2, decimalDigits: 2,
); );
final rateFormat = NumberFormat('0.000000');
final totalUsd = rows.isEmpty ? 0.0 : rows.last.runningUsd; final totalUsd = rows.isEmpty ? 0.0 : rows.last.runningUsd;
final totalSek = rows.isEmpty ? 0.0 : rows.last.runningSek; final totalSek = rows.isEmpty ? 0.0 : rows.last.runningSek;
@@ -80,6 +81,7 @@ class HomeScreen extends StatelessWidget {
DataColumn(label: Text('Status')), DataColumn(label: Text('Status')),
DataColumn(label: Text('USD')), DataColumn(label: Text('USD')),
DataColumn(label: Text('SEK')), DataColumn(label: Text('SEK')),
DataColumn(label: Text('Rate USD->SEK')),
DataColumn(label: Text('Running USD')), DataColumn(label: Text('Running USD')),
DataColumn(label: Text('Running SEK')), DataColumn(label: Text('Running SEK')),
], ],
@@ -110,6 +112,15 @@ class HomeScreen extends StatelessWidget {
: sekCurrency.format(row.sekAmount), : sekCurrency.format(row.sekAmount),
), ),
), ),
DataCell(
Text(
row.effectiveUsdToSekRate == null
? '-'
: rateFormat.format(
row.effectiveUsdToSekRate,
),
),
),
DataCell( DataCell(
Text(usdCurrency.format(row.runningUsd)), Text(usdCurrency.format(row.runningUsd)),
), ),
+5 -1
View File
@@ -91,6 +91,10 @@ class _SettingsScreenState extends State<SettingsScreen> {
content: Text('Backfill finished for last year.'), content: Text('Backfill finished for last year.'),
), ),
); );
} else {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(controller.errorMessage!)),
);
} }
}, },
icon: const Icon(Icons.history), icon: const Icon(Icons.history),
@@ -112,7 +116,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
const Text( const Text(
'ForexRateAPI key is configured in code constant forexRateApiKey.', 'ExchangeratesAPI key is loaded from .env (EXCHANGE_RATE_API_KEY).',
), ),
], ],
), ),
+100 -48
View File
@@ -1,5 +1,7 @@
import 'dart:convert'; import 'dart:convert';
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:http/http.dart' as http; import 'package:http/http.dart' as http;
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
@@ -16,79 +18,129 @@ class ForexRateApiService {
final http.Client httpClient; final http.Client httpClient;
final String apiKey; final String apiKey;
void _log(String message) {
if (kDebugMode) {
debugPrint('[ForexRateApiService] $message');
}
}
String _redactedUri(Uri uri) {
final redactedQuery = <String, String>{...uri.queryParameters};
if (redactedQuery.containsKey('access_key')) {
redactedQuery['access_key'] = '***REDACTED***';
}
return uri.replace(queryParameters: redactedQuery).toString();
}
bool _isRateLimitedResponse(http.Response response) {
if (response.statusCode == 429) {
return true;
}
try {
final parsed = jsonDecode(response.body) as Map<String, dynamic>;
final error = parsed['error'] as Map<String, dynamic>?;
final code = error?['code']?.toString().toLowerCase();
return code == 'rate_limit_reached';
} catch (_) {
return false;
}
}
Future<http.Response> _getWithRateLimitRetry(
Uri uri, {
required String operation,
}) async {
const maxAttempts = 3;
for (var attempt = 1; attempt <= maxAttempts; attempt++) {
final response = await httpClient.get(uri);
_log('$operation HTTP status: ${response.statusCode}');
_log('$operation raw response: ${response.body}');
final shouldRetry = _isRateLimitedResponse(response);
if (!shouldRetry || attempt == maxAttempts) {
return response;
}
final delayMs = 1200 * attempt;
_log(
'$operation rate limited; retrying in ${delayMs}ms (attempt ${attempt + 1}/$maxAttempts)',
);
await Future.delayed(Duration(milliseconds: delayMs));
}
throw StateError('Unexpected rate-limit retry flow for $operation.');
}
Future<double> fetchUsdToSekRate({required DateTime estDate}) async { Future<double> fetchUsdToSekRate({required DateTime estDate}) async {
if (apiKey == 'REPLACE_WITH_YOUR_FOREXRATE_API_KEY') { if (apiKey == 'REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY') {
throw StateError('ForexRateAPI key is not configured.'); throw StateError('ExchangeratesAPI key is not configured.');
} }
final date = DateFormat('yyyy-MM-dd').format(estDate); final date = DateFormat('yyyy-MM-dd').format(estDate);
final uri = Uri.https('api.forexrateapi.com', '/v1/$date', <String, String>{ final uri = Uri.https(
'api_key': apiKey, 'api.exchangeratesapi.io',
'base': 'USD', '/v1/$date',
'currencies': 'SEK', // Free-tier plans can restrict custom base currencies.
}); // We request EUR-based USD and SEK rates and derive USD->SEK via cross-rate.
<String, String>{'access_key': apiKey, 'symbols': 'USD,SEK'},
);
final response = await httpClient.get(uri); _log('Daily rate request: ${_redactedUri(uri)}');
final response = await _getWithRateLimitRetry(uri, operation: 'Daily rate');
if (response.statusCode != 200) { if (response.statusCode != 200) {
throw StateError('ForexRateAPI request failed (${response.statusCode}).'); throw StateError(
'ExchangeratesAPI request failed (${response.statusCode}).',
);
} }
final json = jsonDecode(response.body) as Map<String, dynamic>; final json = jsonDecode(response.body) as Map<String, dynamic>;
final success = json['success'] as bool? ?? false; final success = json['success'] as bool? ?? false;
if (!success) { if (!success) {
throw StateError('ForexRateAPI returned success=false.'); final error = json['error'];
_log('Daily rate success=false. error=$error');
throw StateError('ExchangeratesAPI returned success=false. error=$error');
} }
final rates = json['rates'] as Map<String, dynamic>?; final rates = json['rates'] as Map<String, dynamic>?;
final sek = rates?['SEK']; final usdPerEur = rates?['USD'];
if (sek == null) { final sekPerEur = rates?['SEK'];
throw StateError('SEK rate was missing in API response.'); if (usdPerEur == null || sekPerEur == null) {
throw StateError('USD/SEK rates were missing in API response.');
} }
return (sek as num).toDouble(); final usdPerEurValue = (usdPerEur as num).toDouble();
final sekPerEurValue = (sekPerEur as num).toDouble();
if (usdPerEurValue == 0) {
throw StateError('USD rate was zero; cannot derive USD->SEK.');
}
final usdToSek = sekPerEurValue / usdPerEurValue;
_log(
'Daily rate parsed usdPerEur=$usdPerEurValue sekPerEur=$sekPerEurValue derivedUsdToSek=$usdToSek',
);
return usdToSek;
} }
Future<ForexConversionResult> convertUsdToSek({ Future<ForexConversionResult> convertUsdToSek({
required DateTime estDate, required DateTime estDate,
required double amount, required double amount,
}) async { }) async {
if (apiKey == 'REPLACE_WITH_YOUR_FOREXRATE_API_KEY') { if (apiKey == 'REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY') {
throw StateError('ForexRateAPI key is not configured.'); throw StateError('ExchangeratesAPI key is not configured.');
} }
final date = DateFormat('yyyy-MM-dd').format(estDate); final quote = await fetchUsdToSekRate(estDate: estDate);
final uri = final result = quote * amount;
Uri.https('api.forexrateapi.com', '/v1/convert', <String, String>{
'api_key': apiKey,
'from': 'USD',
'to': 'SEK',
'amount': amount.toStringAsFixed(2),
'date': date,
});
final response = await httpClient.get(uri); _log(
if (response.statusCode != 200) { 'Computed conversion from daily rate quote=$quote result=$result amount=$amount date=${DateFormat('yyyy-MM-dd').format(estDate)}',
throw StateError('ForexRateAPI convert failed (${response.statusCode}).');
}
final json = jsonDecode(response.body) as Map<String, dynamic>;
final success = json['success'] as bool? ?? false;
if (!success) {
throw StateError('ForexRateAPI convert returned success=false.');
}
final info = json['info'] as Map<String, dynamic>?;
final quote = info?['quote'];
final result = json['result'];
if (quote == null || result == null) {
throw StateError(
'ForexRateAPI convert response was missing quote/result.',
);
}
return ForexConversionResult(
quote: (quote as num).toDouble(),
result: (result as num).toDouble(),
); );
return ForexConversionResult(quote: quote, result: result);
} }
} }
+117 -29
View File
@@ -2,6 +2,7 @@ import 'dart:convert';
import 'dart:io'; import 'dart:io';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter_dotenv/flutter_dotenv.dart';
import 'package:http/http.dart' as http; import 'package:http/http.dart' as http;
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:path_provider/path_provider.dart'; import 'package:path_provider/path_provider.dart';
@@ -13,8 +14,6 @@ import 'package:rental_income_tracker/services/storage_service.dart';
import 'package:rental_income_tracker/services/time_service.dart'; import 'package:rental_income_tracker/services/time_service.dart';
import 'package:timezone/timezone.dart' as tz; import 'package:timezone/timezone.dart' as tz;
const String forexRateApiKey = 'REPLACE_WITH_YOUR_FOREXRATE_API_KEY';
class RentTableRow { class RentTableRow {
RentTableRow({ RentTableRow({
required this.year, required this.year,
@@ -23,6 +22,7 @@ class RentTableRow {
required this.status, required this.status,
required this.usdAmount, required this.usdAmount,
required this.sekAmount, required this.sekAmount,
required this.effectiveUsdToSekRate,
required this.runningUsd, required this.runningUsd,
required this.runningSek, required this.runningSek,
}); });
@@ -33,6 +33,7 @@ class RentTableRow {
final PaymentStatus status; final PaymentStatus status;
final double? usdAmount; final double? usdAmount;
final double? sekAmount; final double? sekAmount;
final double? effectiveUsdToSekRate;
final double runningUsd; final double runningUsd;
final double runningSek; final double runningSek;
} }
@@ -46,11 +47,24 @@ class RentController extends ChangeNotifier {
exchangeService ?? exchangeService ??
ForexRateApiService( ForexRateApiService(
httpClient: http.Client(), httpClient: http.Client(),
apiKey: forexRateApiKey, apiKey: _exchangeRateApiKeyFromEnv(),
) { ) {
_notificationService = NotificationService(_onNotificationAction); _notificationService = NotificationService(_onNotificationAction);
} }
static String _exchangeRateApiKeyFromEnv() {
final key = dotenv.env['EXCHANGE_RATE_API_KEY']?.trim() ?? '';
if (key.isEmpty) {
if (kDebugMode) {
debugPrint(
'[RentController] EXCHANGE_RATE_API_KEY missing in .env, API calls will fail until set.',
);
}
return 'REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY';
}
return key;
}
final StorageService _storageService; final StorageService _storageService;
final ForexRateApiService _exchangeService; final ForexRateApiService _exchangeService;
late NotificationService _notificationService; late NotificationService _notificationService;
@@ -138,6 +152,12 @@ class RentController extends ChangeNotifier {
_errorMessage = null; _errorMessage = null;
_setLoading(true); _setLoading(true);
if (kDebugMode) {
debugPrint(
'[RentController] markMonthPaid start year=$year month=$month source=$source assumeOnTime=$assumeOnTime',
);
}
try { try {
if (_isNotOccupiedMonth(year, month)) { if (_isNotOccupiedMonth(year, month)) {
throw StateError('Cannot mark a non-occupied month as paid.'); throw StateError('Cannot mark a non-occupied month as paid.');
@@ -148,6 +168,11 @@ class RentController extends ChangeNotifier {
final key = TimeService.monthKey(year, month); final key = TimeService.monthKey(year, month);
if (_records.containsKey(key)) { if (_records.containsKey(key)) {
if (kDebugMode) {
debugPrint(
'[RentController] markMonthPaid skipped API call; cached record exists for key=$key',
);
}
await _notificationService.cancelForMonth(year, month); await _notificationService.cancelForMonth(year, month);
await _syncNotificationScheduleForCurrentMonth(); await _syncNotificationScheduleForCurrentMonth();
return; return;
@@ -161,6 +186,11 @@ class RentController extends ChangeNotifier {
estDate: estDate, estDate: estDate,
amount: usd, amount: usd,
); );
if (kDebugMode) {
debugPrint(
'[RentController] markMonthPaid conversion success quote=${conversion.quote} result=${conversion.result} usd=$usd estDate=$estDate',
);
}
final onTime = final onTime =
assumeOnTime || TimeService.isOnTimeByDeadline(paidAt, year, month); assumeOnTime || TimeService.isOnTimeByDeadline(paidAt, year, month);
@@ -178,9 +208,19 @@ class RentController extends ChangeNotifier {
_records[record.key] = record; _records[record.key] = record;
await _storageService.saveRecords(_records); await _storageService.saveRecords(_records);
if (kDebugMode) {
debugPrint(
'[RentController] markMonthPaid stored record key=${record.key}',
);
}
await _notificationService.cancelForMonth(year, month); await _notificationService.cancelForMonth(year, month);
await _syncNotificationScheduleForCurrentMonth(); await _syncNotificationScheduleForCurrentMonth();
} catch (err) { } catch (err, stackTrace) {
if (kDebugMode) {
debugPrint('[RentController] markMonthPaid failed: $err');
debugPrint(stackTrace.toString());
}
_errorMessage = err.toString(); _errorMessage = err.toString();
} finally { } finally {
_setLoading(false); _setLoading(false);
@@ -191,48 +231,93 @@ class RentController extends ChangeNotifier {
_errorMessage = null; _errorMessage = null;
_setLoading(true); _setLoading(true);
if (kDebugMode) {
debugPrint('[RentController] backfillLastYear start');
}
try { try {
if (_settings.rentUsd <= 0) { if (_settings.rentUsd <= 0) {
throw StateError('Set rent amount in Settings before backfill.'); throw StateError('Set rent amount in Settings before backfill.');
} }
final year = DateTime.now().year - 1; final year = DateTime.now().year - 1;
var savedMonths = 0;
var pausedByRateLimit = false;
for (var month = 1; month <= 12; month++) { for (var month = 1; month <= 12; month++) {
final key = TimeService.monthKey(year, month); final key = TimeService.monthKey(year, month);
if (_records.containsKey(key)) { if (_records.containsKey(key)) {
if (kDebugMode) {
debugPrint('[RentController] backfill skip cached key=$key');
}
continue; continue;
} }
final usd = _settings.rentUsd; try {
final estDate = DateTime(year, month, 1); final usd = _settings.rentUsd;
final conversion = await _exchangeService.convertUsdToSek( final estDate = DateTime(year, month, 1);
estDate: estDate, final conversion = await _exchangeService.convertUsdToSek(
amount: usd, estDate: estDate,
); amount: usd,
);
final paidAtUtc = tz.TZDateTime( if (kDebugMode) {
TimeService.est, debugPrint(
year, '[RentController] backfill conversion year=$year month=$month quote=${conversion.quote} result=${conversion.result}',
month, );
1, }
12,
).toUtc();
_records[key] = MonthlyRentRecord( final paidAtUtc = tz.TZDateTime(
year: year, TimeService.est,
month: month, year,
usdAmount: usd, month,
sekAmount: conversion.result, 1,
usdToSekRate: conversion.quote, 12,
paidAtUtc: paidAtUtc, ).toUtc();
onTime: true,
source: 'backfill', _records[key] = MonthlyRentRecord(
); year: year,
month: month,
usdAmount: usd,
sekAmount: conversion.result,
usdToSekRate: conversion.quote,
paidAtUtc: paidAtUtc,
onTime: true,
source: 'backfill',
);
await _storageService.saveRecords(_records);
savedMonths++;
} catch (err) {
final errorText = err.toString();
if (errorText.contains('429') ||
errorText.toLowerCase().contains('rate_limit_reached')) {
pausedByRateLimit = true;
_errorMessage =
'Backfill paused by API rate limits after $savedMonths new month(s). Run backfill again in a minute to continue.';
if (kDebugMode) {
debugPrint(
'[RentController] backfill paused by rate limit after $savedMonths month(s)',
);
}
break;
}
rethrow;
}
} }
await _storageService.saveRecords(_records); if (kDebugMode) {
debugPrint(
pausedByRateLimit
? '[RentController] backfillLastYear paused by rate limit'
: '[RentController] backfillLastYear completed',
);
}
notifyListeners(); notifyListeners();
} catch (err) { } catch (err, stackTrace) {
if (kDebugMode) {
debugPrint('[RentController] backfillLastYear failed: $err');
debugPrint(stackTrace.toString());
}
_errorMessage = err.toString(); _errorMessage = err.toString();
} finally { } finally {
_setLoading(false); _setLoading(false);
@@ -280,10 +365,12 @@ class RentController extends ChangeNotifier {
double? usd; double? usd;
double? sek; double? sek;
double? rate;
if (record != null) { if (record != null) {
usd = record.usdAmount; usd = record.usdAmount;
sek = record.sekAmount; sek = record.sekAmount;
rate = record.usdToSekRate;
runningUsd += usd; runningUsd += usd;
runningSek += sek; runningSek += sek;
} }
@@ -296,6 +383,7 @@ class RentController extends ChangeNotifier {
status: status, status: status,
usdAmount: usd, usdAmount: usd,
sekAmount: sek, sekAmount: sek,
effectiveUsdToSekRate: rate,
runningUsd: runningUsd, runningUsd: runningUsd,
runningSek: runningSek, runningSek: runningSek,
), ),
+3
View File
@@ -9,6 +9,7 @@ environment:
dependencies: dependencies:
flutter: flutter:
sdk: flutter sdk: flutter
flutter_dotenv: ^5.2.1
flutter_local_notifications: ^19.5.0 flutter_local_notifications: ^19.5.0
flutter_timezone: ^4.1.1 flutter_timezone: ^4.1.1
http: ^1.5.0 http: ^1.5.0
@@ -25,3 +26,5 @@ dev_dependencies:
flutter: flutter:
uses-material-design: true uses-material-design: true
assets:
- ./.env