2023-09-14 23:34:01 +00:00
|
|
|
import 'package:bip39/bip39.dart' as bip39;
|
|
|
|
import 'package:coinlib/coinlib.dart' as coinlib;
|
|
|
|
import 'package:isar/isar.dart';
|
|
|
|
import 'package:stackwallet/models/isar/models/isar_models.dart';
|
|
|
|
import 'package:stackwallet/utilities/enums/derive_path_type_enum.dart';
|
2023-09-18 21:28:31 +00:00
|
|
|
import 'package:stackwallet/wallets/crypto_currency/bip39_hd_currency.dart';
|
2023-09-14 23:34:01 +00:00
|
|
|
import 'package:stackwallet/wallets/models/tx_data.dart';
|
2023-09-18 21:28:31 +00:00
|
|
|
import 'package:stackwallet/wallets/wallet/bip39_wallet.dart';
|
2023-09-14 23:34:01 +00:00
|
|
|
|
2023-09-18 21:28:31 +00:00
|
|
|
abstract class Bip39HDWallet<T extends Bip39HDCurrency> extends Bip39Wallet<T> {
|
2023-09-14 23:34:01 +00:00
|
|
|
Bip39HDWallet(super.cryptoCurrency);
|
|
|
|
|
|
|
|
/// Generates a receiving address of [walletInfo.mainAddressType]. If none
|
|
|
|
/// are in the current wallet db it will generate at index 0, otherwise the
|
|
|
|
/// highest index found in the current wallet db.
|
|
|
|
Future<Address> generateNewReceivingAddress() async {
|
|
|
|
final current = await _currentReceivingAddress;
|
|
|
|
final index = current?.derivationIndex ?? 0;
|
|
|
|
const chain = 0; // receiving address
|
|
|
|
|
|
|
|
final DerivePathType derivePathType;
|
|
|
|
switch (walletInfo.mainAddressType) {
|
|
|
|
case AddressType.p2pkh:
|
|
|
|
derivePathType = DerivePathType.bip44;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AddressType.p2sh:
|
|
|
|
derivePathType = DerivePathType.bip44;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case AddressType.p2wpkh:
|
|
|
|
derivePathType = DerivePathType.bip44;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw Exception(
|
|
|
|
"Invalid AddressType accessed in $runtimeType generateNewReceivingAddress()",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
final address = await _generateAddress(
|
|
|
|
chain: chain,
|
|
|
|
index: index,
|
|
|
|
derivePathType: derivePathType,
|
|
|
|
);
|
|
|
|
|
|
|
|
await mainDB.putAddress(address);
|
|
|
|
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ========== Private ========================================================
|
|
|
|
|
|
|
|
Future<Address?> get _currentReceivingAddress async =>
|
|
|
|
await mainDB.isar.addresses
|
|
|
|
.where()
|
|
|
|
.walletIdEqualTo(walletId)
|
|
|
|
.filter()
|
|
|
|
.typeEqualTo(walletInfo.mainAddressType)
|
|
|
|
.subTypeEqualTo(AddressSubType.receiving)
|
|
|
|
.sortByDerivationIndexDesc()
|
|
|
|
.findFirst();
|
|
|
|
|
|
|
|
Future<coinlib.HDPrivateKey> _generateRootHDNode() async {
|
|
|
|
final seed = bip39.mnemonicToSeed(
|
|
|
|
await getMnemonic(),
|
|
|
|
passphrase: await getMnemonicPassphrase(),
|
|
|
|
);
|
|
|
|
return coinlib.HDPrivateKey.fromSeed(seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<Address> _generateAddress({
|
|
|
|
required int chain,
|
|
|
|
required int index,
|
|
|
|
required DerivePathType derivePathType,
|
|
|
|
}) async {
|
|
|
|
final root = await _generateRootHDNode();
|
|
|
|
|
|
|
|
final derivationPath = cryptoCurrency.constructDerivePath(
|
|
|
|
derivePathType: derivePathType,
|
|
|
|
chain: chain,
|
|
|
|
index: index,
|
|
|
|
);
|
|
|
|
|
|
|
|
final keys = root.derivePath(derivationPath);
|
|
|
|
|
|
|
|
final data = cryptoCurrency.getAddressForPublicKey(
|
|
|
|
publicKey: keys.publicKey,
|
|
|
|
derivePathType: derivePathType,
|
|
|
|
);
|
|
|
|
|
|
|
|
final AddressSubType subType;
|
|
|
|
|
|
|
|
if (chain == 0) {
|
|
|
|
subType = AddressSubType.receiving;
|
|
|
|
} else if (chain == 1) {
|
|
|
|
subType = AddressSubType.change;
|
|
|
|
} else {
|
|
|
|
// TODO others?
|
|
|
|
subType = AddressSubType.unknown;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Address(
|
|
|
|
walletId: walletId,
|
|
|
|
value: data.address.toString(),
|
|
|
|
publicKey: keys.publicKey.data,
|
|
|
|
derivationIndex: index,
|
|
|
|
derivationPath: DerivationPath()..value = derivationPath,
|
|
|
|
type: data.addressType,
|
|
|
|
subType: subType,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ========== Overrides ======================================================
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<TxData> confirmSend({required TxData txData}) {
|
|
|
|
// TODO: implement confirmSend
|
|
|
|
throw UnimplementedError();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<TxData> prepareSend({required TxData txData}) {
|
|
|
|
// TODO: implement prepareSend
|
|
|
|
throw UnimplementedError();
|
|
|
|
}
|
2023-09-18 21:28:31 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
Future<void> recover({required bool isRescan}) {
|
|
|
|
// TODO: implement recover
|
|
|
|
throw UnimplementedError();
|
|
|
|
}
|
2023-09-14 23:34:01 +00:00
|
|
|
}
|