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:
@@ -0,0 +1 @@
|
|||||||
|
EXCHANGE_RATE_API_KEY=REPLACE_WITH_YOUR_EXCHANGE_RATE_API_KEY
|
||||||
@@ -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
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -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)),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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).',
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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
@@ -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,
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
Reference in New Issue
Block a user