mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2024-12-28 22:49:49 +00:00
c5477e4f9e
* Add build scripts for macOS. Add macos for cw_monero plugin. Add macos proj to the application. * - Update Flutter secure storage to work with macos - Enable uni links only on Mobile - Update devcelocale to work with macos * Add network access to mac * Change Dashboard view on desktop size screens * Add on Tap to desktop_action_button.dart Remove unused functions * Fix arch match for monero lib for darwin x86_64 -> x86-64 * Add Bundle ID in entitlements files through app config script * Update deployment target to 10.13 * Revert back to Cake fork for secure storage * Revert back to Cake fork for secure storage * Revert mac os version * Revert mac os version * Add platform channel specific code for mac os * Add desktop sidebar * [skip ci] Add desktop sidebar * [skip ci] Add desktop sidebar * - Remove legacy migration from macos - Remove wake lock native code and just use the ready made package * Remove wake lock native code and just use the ready made package * Remove unstoppable domain from macos since it's not supported * Temporarily fetch unstoppable domains only on mobile * refactor desktop settings sidebar * Ignore increasing brightness for non-mobile platforms * Add Wallet selection dropdown to dashboard desktop view * Generate MacOS icons * localize settings * fix dashboard sidebar and responsive utils * Change Mac os app name and bundle id * Fix exchange page as fullScreenDialog * Remove constants * - Refactor onRamper to have a single point of modification - Enlarge initial app size - update Flutter and Packages * Add pubspec.lock and Podfile.lock to gitignore * Remove Podfile.lock from cache * Fix bug on sidebar reset * Fix issues from code review * [skip ci] reformat desktop dashboard * [skip ci] reformat desktop dashboard * Revert removing .lock files * Revert changes in .gitignore * [skip ci] remove .project changes * [skip ci] remove .project changes * Separate Dashboard desktop view from mobile view * constraint images and pincoded box * Remove drawer from mac os * - Listen to keyboard events in PIN screen - Fix PIN buttons style * Fix desktop nav bar UI * Add Marketplace to dashboard view * Update trailing icon to open transaction page * Update widget contraints * Add empty trailing to center page title on desktop * Refresh desktop dashboard actions on wallet change * Change ionia welcome page animation * Fix Constrained width screens UI * Refactor sidebar state management * remove empty line * Add max width constrain to Welcome page * Change Exchange page UI depending on platform * - Change design/paddings for Send page on desktop view - Make AddTemplateButton instead of having it duplicated in send/exchange * Fix Desktop dashboard actions background color * Constrain primary Buttons width * Make side menu items toggle back to dashboard * Add padding to support page * Add width constraints to desktop dashboard * Fix UI issues, paddings and alignments * Rename misleading variable Change initial mac window size * Fix wallet create in settings * remove unnecessary code * remove unnecessary code * Remove duplicated constrains * - Use close icon on main screens - Minor UI fixes * fix pageview controller reset index * Add create and restore wallet options to dropdown menu * Fix desktop background color and address book view issues * Fix input field * Add onFieldSubmitted to allow "enter" button interaction * Fix issue from code review * Fix Popup width constraint and add focus orders * Fix variable name * Fix issues from code review * refactor dropdown items * Fix alignment in create and restore wallet screens * Fix dropdown change state bug Hide scanner for desktop * remove space * override navbar with desktopnavbar * Remove autofocus * remove unused code * Fix ionia input field alignment * Replace removed code * Add app lock feature on mac * Add assertion to avoid null * Add Nano currency image * Enable adding contact from send screen * Fix UI issues Add missing translation * pop only PIN screen after successful auth * Add back wallet settings page to desktop settings actions * Fix Navigation animation for settings screens * Fixate MobX version to fix restore issue * CW-324 Refresh current settings page if wallet changed (#811) * Fix refresh current settings page if wallet changed * Fix refresh current settings page if wallet changed * Refresh Wallet Seeds/Keys List upon wallet change --------- Co-authored-by: OmarHatem <omarh.ismail1@gmail.com> * Remove navigation workaround for duplicate key, and fix the issue by handling creation/disposing of global key (#840) * Cw 323 add wallet list to settings on mac (#843) * Remove navigation workaround for duplicate key, and fix the issue by handling creation/disposing of global key * - Register Wallet List as singleton in Desktop to be modify the same instance from settings and dropdown - General Fixes and Enhancements * Fix Changing/Restoring wallet from settings * Fix Create wallet not showing seeds screens if launched from settings * Add max width constraint for Alerts * - Add Desktop API keys - Fix Change back up password issue - Fix Popup width * Sync Mac with latest main updates * Swap Transactions icon with lock icon * Save backup file locally on desktop * Sync with latest main updates * Fix Navigation issues with anonpay * Update macos build version * Remove deprecated custom wake lock code for Android * Remove Legacy CryptoSwift package from MacOS * - Refactor Payfura page code - Add OnRamper new configs to onramper_buy_provider.dart - Fix Conflicts with main * Updated device locale package * Update android tools * Revert changes and update only gradle version * Downgrade android tools version * Update gradle version * Update package/gradle/plugin version * - Fixate device locale version - Downgrade gradle version * Update kotlin version * Update gradle version * Trial for a custom fork from devicelocale * Fixate shared preferences package version * Revert gradle version * Revert kotlin version * Downgrade gradle version * Downgrade gradle version * Repair cache and clean before build * Fixate flutter version * update google services version * revert google services version * Force shared pref android version * Override shared prefs android package version * Override shared prefs android package [skip ci] --------- Co-authored-by: M <m@cakewallet.com> Co-authored-by: Godwin Asuquo <godilite@gmail.com> Co-authored-by: Godwin Asuquo <41484542+godilite@users.noreply.github.com>
230 lines
7.7 KiB
Dart
230 lines
7.7 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:cake_wallet/entities/preferences_key.dart';
|
|
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';
|
|
import 'package:device_info_plus/device_info_plus.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_mailer/flutter_mailer.dart';
|
|
import 'package:package_info/package_info.dart';
|
|
import 'package:path_provider/path_provider.dart';
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
class ExceptionHandler {
|
|
static bool _hasError = false;
|
|
static const _coolDownDurationInDays = 7;
|
|
|
|
static void _saveException(String? error, StackTrace? stackTrace, {String? library}) async {
|
|
final appDocDir = await getApplicationDocumentsDirectory();
|
|
|
|
final file = File('${appDocDir.path}/error.txt');
|
|
final exception = {
|
|
"${DateTime.now()}": {
|
|
"Error": "$error\n\n",
|
|
"Library": "$library\n\n",
|
|
"StackTrace": stackTrace.toString(),
|
|
}
|
|
};
|
|
|
|
const String separator = '''\n\n==========================================================
|
|
==========================================================\n\n''';
|
|
|
|
file.writeAsStringSync(
|
|
"$exception $separator",
|
|
mode: FileMode.append,
|
|
);
|
|
}
|
|
|
|
static void _sendExceptionFile() async {
|
|
try {
|
|
final appDocDir = await getApplicationDocumentsDirectory();
|
|
|
|
final file = File('${appDocDir.path}/error.txt');
|
|
|
|
await _addDeviceInfo(file);
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
static void onError(FlutterErrorDetails errorDetails) async {
|
|
if (kDebugMode) {
|
|
FlutterError.presentError(errorDetails);
|
|
debugPrint(errorDetails.toString());
|
|
return;
|
|
}
|
|
|
|
if (_ignoreError(errorDetails.exception.toString())) {
|
|
return;
|
|
}
|
|
|
|
_saveException(
|
|
errorDetails.exceptionAsString(),
|
|
errorDetails.stack,
|
|
library: errorDetails.library,
|
|
);
|
|
|
|
final sharedPrefs = await SharedPreferences.getInstance();
|
|
|
|
final lastPopupDate =
|
|
DateTime.tryParse(sharedPrefs.getString(PreferencesKey.lastPopupDate) ?? '') ??
|
|
DateTime.now().subtract(Duration(days: _coolDownDurationInDays + 1));
|
|
|
|
final durationSinceLastReport = DateTime.now().difference(lastPopupDate).inDays;
|
|
|
|
if (_hasError || durationSinceLastReport < _coolDownDurationInDays) {
|
|
return;
|
|
}
|
|
_hasError = true;
|
|
|
|
sharedPrefs.setString(PreferencesKey.lastPopupDate, DateTime.now().toString());
|
|
|
|
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;
|
|
},
|
|
);
|
|
}
|
|
|
|
/// Ignore User related errors or system errors
|
|
static bool _ignoreError(String error) =>
|
|
_ignoredErrors.any((element) => error.contains(element));
|
|
|
|
static const List<String> _ignoredErrors = const [
|
|
"errno = 9", // SocketException: Bad file descriptor
|
|
"errno = 28", // OS Error: No space left on device
|
|
"errno = 32", // SocketException: Write failed (OS Error: Broken pipe)
|
|
"errno = 49", // SocketException: Can't assign requested address
|
|
"errno = 54", // SocketException: Connection reset by peer
|
|
"errno = 57", // SocketException: Read failed (OS Error: Socket is not connected)
|
|
"errno = 60", // SocketException: Operation timed out
|
|
"errno = 65", // SocketException: No route to host
|
|
"errno = 103", // SocketException: Software caused connection abort
|
|
"errno = 104", // SocketException: Connection reset by peer
|
|
"errno = 110", // SocketException: Connection timed out
|
|
"HttpException: Connection reset by peer",
|
|
"HttpException: Connection closed before full header was received",
|
|
"HandshakeException: Connection terminated during handshake",
|
|
"PERMISSION_NOT_GRANTED",
|
|
];
|
|
|
|
static Future<void> _addDeviceInfo(File file) async {
|
|
final packageInfo = await PackageInfo.fromPlatform();
|
|
final currentVersion = packageInfo.version;
|
|
|
|
final deviceInfoPlugin = DeviceInfoPlugin();
|
|
Map<String, dynamic> deviceInfo = {};
|
|
|
|
if (Platform.isAndroid) {
|
|
deviceInfo = _readAndroidBuildData(await deviceInfoPlugin.androidInfo);
|
|
deviceInfo["Platform"] = "Android";
|
|
} else if (Platform.isIOS) {
|
|
deviceInfo = _readIosDeviceInfo(await deviceInfoPlugin.iosInfo);
|
|
deviceInfo["Platform"] = "iOS";
|
|
} else if (Platform.isLinux) {
|
|
deviceInfo = _readLinuxDeviceInfo(await deviceInfoPlugin.linuxInfo);
|
|
deviceInfo["Platform"] = "Linux";
|
|
} else if (Platform.isMacOS) {
|
|
deviceInfo = _readMacOsDeviceInfo(await deviceInfoPlugin.macOsInfo);
|
|
deviceInfo["Platform"] = "MacOS";
|
|
} else if (Platform.isWindows) {
|
|
deviceInfo = _readWindowsDeviceInfo(await deviceInfoPlugin.windowsInfo);
|
|
deviceInfo["Platform"] = "Windows";
|
|
}
|
|
|
|
await file.writeAsString(
|
|
"App Version: $currentVersion\n\nDevice Info $deviceInfo",
|
|
mode: FileMode.append,
|
|
);
|
|
}
|
|
|
|
static Map<String, dynamic> _readAndroidBuildData(AndroidDeviceInfo build) {
|
|
return <String, dynamic>{
|
|
'brand': build.brand,
|
|
'device': build.device,
|
|
'manufacturer': build.manufacturer,
|
|
'model': build.model,
|
|
'product': build.product,
|
|
};
|
|
}
|
|
|
|
static Map<String, dynamic> _readIosDeviceInfo(IosDeviceInfo data) {
|
|
return <String, dynamic>{
|
|
'systemName': data.systemName,
|
|
'systemVersion': data.systemVersion,
|
|
'model': data.model,
|
|
'localizedModel': data.localizedModel,
|
|
};
|
|
}
|
|
|
|
static Map<String, dynamic> _readLinuxDeviceInfo(LinuxDeviceInfo data) {
|
|
return <String, dynamic>{
|
|
'name': data.name,
|
|
'version': data.version,
|
|
'versionCodename': data.versionCodename,
|
|
'versionId': data.versionId,
|
|
'prettyName': data.prettyName,
|
|
'buildId': data.buildId,
|
|
'variant': data.variant,
|
|
'variantId': data.variantId,
|
|
};
|
|
}
|
|
|
|
static Map<String, dynamic> _readMacOsDeviceInfo(MacOsDeviceInfo data) {
|
|
return <String, dynamic>{
|
|
'arch': data.arch,
|
|
'model': data.model,
|
|
'kernelVersion': data.kernelVersion,
|
|
'osRelease': data.osRelease,
|
|
};
|
|
}
|
|
|
|
static Map<String, dynamic> _readWindowsDeviceInfo(WindowsDeviceInfo data) {
|
|
return <String, dynamic>{
|
|
'majorVersion': data.majorVersion,
|
|
'minorVersion': data.minorVersion,
|
|
'buildNumber': data.buildNumber,
|
|
'productType': data.productType,
|
|
'productName': data.productName,
|
|
};
|
|
}
|
|
}
|