2023-02-01 15:37:18 +00:00
|
|
|
import 'dart:convert';
|
|
|
|
import 'dart:io';
|
|
|
|
|
2023-02-07 14:53:57 +00:00
|
|
|
import 'package:cake_wallet/entities/preferences_key.dart';
|
2023-02-01 15:37:18 +00:00
|
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
|
|
import 'package:cake_wallet/main.dart';
|
|
|
|
import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart';
|
|
|
|
import 'package:cake_wallet/utils/show_pop_up.dart';
|
2023-02-06 16:33:12 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
2023-02-01 15:37:18 +00:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_mailer/flutter_mailer.dart';
|
|
|
|
import 'package:path_provider/path_provider.dart';
|
2023-02-07 14:53:57 +00:00
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
2023-02-01 15:37:18 +00:00
|
|
|
|
|
|
|
class ExceptionHandler {
|
|
|
|
static bool _hasError = false;
|
|
|
|
|
|
|
|
static void _saveException(String? error, StackTrace? stackTrace) async {
|
|
|
|
final appDocDir = await getApplicationDocumentsDirectory();
|
|
|
|
|
|
|
|
final file = File('${appDocDir.path}/error.txt');
|
|
|
|
final exception = {
|
|
|
|
"${DateTime.now()}": {
|
|
|
|
"Error": error,
|
|
|
|
"StackTrace": stackTrace.toString(),
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const String separator = '''\n\n==========================================================
|
|
|
|
==========================================================\n\n''';
|
|
|
|
|
|
|
|
await file.writeAsString(
|
|
|
|
jsonEncode(exception) + separator,
|
|
|
|
mode: FileMode.append,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _sendExceptionFile() async {
|
|
|
|
try {
|
|
|
|
final appDocDir = await getApplicationDocumentsDirectory();
|
|
|
|
|
|
|
|
final file = File('${appDocDir.path}/error.txt');
|
|
|
|
|
|
|
|
final MailOptions mailOptions = MailOptions(
|
|
|
|
subject: 'Mobile App Issue',
|
|
|
|
recipients: ['support@cakewallet.com'],
|
|
|
|
attachments: [file.path],
|
|
|
|
);
|
|
|
|
|
|
|
|
final result = await FlutterMailer.send(mailOptions);
|
|
|
|
|
|
|
|
// Clear file content if the error was sent or saved.
|
|
|
|
// On android we can't know if it was sent or saved
|
|
|
|
if (result.name == MailerResponse.sent.name ||
|
|
|
|
result.name == MailerResponse.saved.name ||
|
|
|
|
result.name == MailerResponse.android.name) {
|
|
|
|
file.writeAsString("", mode: FileMode.write);
|
|
|
|
}
|
|
|
|
} catch (e, s) {
|
|
|
|
_saveException(e.toString(), s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-07 14:53:57 +00:00
|
|
|
static void onError(FlutterErrorDetails errorDetails) async {
|
2023-02-06 16:33:12 +00:00
|
|
|
if (kDebugMode) {
|
|
|
|
FlutterError.presentError(errorDetails);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-03 12:44:13 +00:00
|
|
|
if (_ignoreError(errorDetails.exception.toString())) {
|
2023-02-01 15:37:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_saveException(errorDetails.exception.toString(), errorDetails.stack);
|
|
|
|
|
2023-02-07 14:53:57 +00:00
|
|
|
final sharedPrefs = await SharedPreferences.getInstance();
|
|
|
|
|
|
|
|
final lastPopupDate =
|
|
|
|
DateTime.tryParse(sharedPrefs.getString(PreferencesKey.lastPopupDate) ?? '') ??
|
|
|
|
DateTime.parse("2001-01-01");
|
|
|
|
|
|
|
|
final durationSinceLastReport = DateTime.now().difference(lastPopupDate).inDays;
|
|
|
|
|
|
|
|
// cool-down duration to be 7 days between reports
|
|
|
|
if (_hasError || durationSinceLastReport < 7) {
|
2023-02-01 15:37:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
_hasError = true;
|
|
|
|
|
2023-02-07 14:53:57 +00:00
|
|
|
sharedPrefs.setString(PreferencesKey.lastPopupDate, DateTime.now().toString());
|
|
|
|
|
2023-02-01 15:37:18 +00:00
|
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
|
|
(timeStamp) async {
|
|
|
|
await showPopUp<void>(
|
|
|
|
context: navigatorKey.currentContext!,
|
|
|
|
builder: (context) {
|
|
|
|
return AlertWithTwoActions(
|
|
|
|
isDividerExist: true,
|
|
|
|
alertTitle: S.of(context).error,
|
|
|
|
alertContent: S.of(context).error_dialog_content,
|
|
|
|
rightButtonText: S.of(context).send,
|
|
|
|
leftButtonText: S.of(context).do_not_send,
|
|
|
|
actionRightButton: () {
|
|
|
|
Navigator.of(context).pop();
|
|
|
|
_sendExceptionFile();
|
|
|
|
},
|
|
|
|
actionLeftButton: () {
|
|
|
|
Navigator.of(context).pop();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
_hasError = false;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-02-03 12:44:13 +00:00
|
|
|
/// Ignore User related errors or system errors
|
|
|
|
static bool _ignoreError(String error) {
|
|
|
|
return error.contains("errno = 103") || // SocketException: Software caused connection abort
|
2023-02-06 17:56:55 +00:00
|
|
|
error.contains("errno = 9") || // SocketException: Bad file descriptor
|
2023-02-07 16:03:31 +00:00
|
|
|
error.contains("errno = 32") || // SocketException: Write failed (OS Error: Broken pipe)
|
2023-02-07 20:11:25 +00:00
|
|
|
error.contains("errno = 60") || // SocketException: Operation timed out
|
|
|
|
error.contains("errno = 54") || // SocketException: Connection reset by peer
|
|
|
|
error.contains("errno = 49") || // SocketException: Can't assign requested address
|
|
|
|
error.contains("PERMISSION_NOT_GRANTED") ||
|
|
|
|
error.contains("errno = 28"); // OS Error: No space left on device
|
2023-02-01 15:37:18 +00:00
|
|
|
}
|
|
|
|
}
|