import 'package:bip39/bip39.dart' as bip39; import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:cw_ethereum/ethereum_client.dart'; import 'package:cw_ethereum/ethereum_mnemonics_exception.dart'; import 'package:cw_ethereum/ethereum_wallet.dart'; import 'package:cw_evm/evm_chain_wallet_creation_credentials.dart'; import 'package:cw_evm/evm_chain_wallet_service.dart'; class EthereumWalletService extends EVMChainWalletService { EthereumWalletService(super.walletInfoSource, {required this.client}); late EthereumClient client; @override WalletType getType() => WalletType.ethereum; @override Future create(EVMChainNewWalletCredentials credentials, {bool? isTestnet}) async { final strength = credentials.seedPhraseLength == 24 ? 256 : 128; final mnemonic = bip39.generateMnemonic(strength: strength); final wallet = EthereumWallet( walletInfo: credentials.walletInfo!, mnemonic: mnemonic, password: credentials.password!, client: client, ); await wallet.init(); wallet.addInitialTokens(); await wallet.save(); return wallet; } @override Future openWallet(String name, String password) async { final walletInfo = walletInfoSource.values.firstWhere((info) => info.id == WalletBase.idFor(name, getType())); try { final wallet = await EthereumWallet.open( name: name, password: password, walletInfo: walletInfo, ); await wallet.init(); await wallet.save(); saveBackup(name); return wallet; } catch (_) { await restoreWalletFilesFromBackup(name); final wallet = await EthereumWallet.open( name: name, password: password, walletInfo: walletInfo, ); await wallet.init(); await wallet.save(); return wallet; } } @override Future rename(String currentName, String password, String newName) async { final currentWalletInfo = walletInfoSource.values .firstWhere((info) => info.id == WalletBase.idFor(currentName, getType())); final currentWallet = await EthereumWallet.open( password: password, name: currentName, walletInfo: currentWalletInfo); await currentWallet.renameWalletFiles(newName); await saveBackup(newName); final newWalletInfo = currentWalletInfo; newWalletInfo.id = WalletBase.idFor(newName, getType()); newWalletInfo.name = newName; await walletInfoSource.put(currentWalletInfo.key, newWalletInfo); } @override Future restoreFromHardwareWallet( EVMChainRestoreWalletFromHardware credentials) async { credentials.walletInfo!.derivationInfo = DerivationInfo( derivationType: DerivationType.bip39, derivationPath: "m/44'/60'/${credentials.hwAccountData.accountIndex}'/0/0" ); credentials.walletInfo!.hardwareWalletType = credentials.hardwareWalletType; credentials.walletInfo!.address = credentials.hwAccountData.address; final wallet = EthereumWallet( walletInfo: credentials.walletInfo!, password: credentials.password!, client: client, ); await wallet.init(); wallet.addInitialTokens(); await wallet.save(); return wallet; } @override Future restoreFromKeys(EVMChainRestoreWalletFromPrivateKey credentials, {bool? isTestnet}) async { final wallet = EthereumWallet( password: credentials.password!, privateKey: credentials.privateKey, walletInfo: credentials.walletInfo!, client: client, ); await wallet.init(); wallet.addInitialTokens(); await wallet.save(); return wallet; } @override Future restoreFromSeed(EVMChainRestoreWalletFromSeedCredentials credentials, {bool? isTestnet}) async { if (!bip39.validateMnemonic(credentials.mnemonic)) { throw EthereumMnemonicIsIncorrectException(); } final wallet = EthereumWallet( password: credentials.password!, mnemonic: credentials.mnemonic, walletInfo: credentials.walletInfo!, client: client, ); await wallet.init(); wallet.addInitialTokens(); await wallet.save(); return wallet; } }