mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-02-03 19:56:31 +00:00
add bch skeleton
This commit is contained in:
parent
f2c27a724c
commit
5e5f2607a2
2 changed files with 262 additions and 0 deletions
164
lib/wallets/crypto_currency/coins/bitcoincash.dart
Normal file
164
lib/wallets/crypto_currency/coins/bitcoincash.dart
Normal file
|
@ -0,0 +1,164 @@
|
||||||
|
import 'package:bitbox/bitbox.dart' as bitbox;
|
||||||
|
import 'package:coinlib_flutter/coinlib_flutter.dart' as coinlib;
|
||||||
|
import 'package:stackwallet/models/isar/models/blockchain_data/address.dart';
|
||||||
|
import 'package:stackwallet/utilities/amount/amount.dart';
|
||||||
|
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
||||||
|
import 'package:stackwallet/utilities/enums/derive_path_type_enum.dart';
|
||||||
|
import 'package:stackwallet/wallets/crypto_currency/bip39_hd_currency.dart';
|
||||||
|
import 'package:stackwallet/wallets/crypto_currency/crypto_currency.dart';
|
||||||
|
|
||||||
|
class Bitcoincash extends Bip39HDCurrency {
|
||||||
|
Bitcoincash(super.network) {
|
||||||
|
switch (network) {
|
||||||
|
case CryptoCurrencyNetwork.main:
|
||||||
|
coin = Coin.bitcoin;
|
||||||
|
case CryptoCurrencyNetwork.test:
|
||||||
|
coin = Coin.bitcoinTestNet;
|
||||||
|
default:
|
||||||
|
throw Exception("Unsupported network: $network");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get genesisHash {
|
||||||
|
switch (network) {
|
||||||
|
case CryptoCurrencyNetwork.main:
|
||||||
|
return "000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f";
|
||||||
|
case CryptoCurrencyNetwork.test:
|
||||||
|
return "000000000933ea01ad0ee984209779baaec3ced90fa3f408719526f8d77f4943";
|
||||||
|
default:
|
||||||
|
throw Exception("Unsupported network: $network");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Amount get dustLimit => Amount(
|
||||||
|
rawValue: BigInt.from(546),
|
||||||
|
fractionDigits: fractionDigits,
|
||||||
|
);
|
||||||
|
|
||||||
|
@override
|
||||||
|
coinlib.NetworkParams get networkParams {
|
||||||
|
switch (network) {
|
||||||
|
case CryptoCurrencyNetwork.main:
|
||||||
|
return const coinlib.NetworkParams(
|
||||||
|
wifPrefix: 0x80,
|
||||||
|
p2pkhPrefix: 0x00,
|
||||||
|
p2shPrefix: 0x05,
|
||||||
|
privHDPrefix: 0x0488ade4,
|
||||||
|
pubHDPrefix: 0x0488b21e,
|
||||||
|
bech32Hrp: "bc",
|
||||||
|
messagePrefix: '\x18Bitcoin Signed Message:\n',
|
||||||
|
);
|
||||||
|
case CryptoCurrencyNetwork.test:
|
||||||
|
return const coinlib.NetworkParams(
|
||||||
|
wifPrefix: 0xef,
|
||||||
|
p2pkhPrefix: 0x6f,
|
||||||
|
p2shPrefix: 0xc4,
|
||||||
|
privHDPrefix: 0x04358394,
|
||||||
|
pubHDPrefix: 0x043587cf,
|
||||||
|
bech32Hrp: "tb",
|
||||||
|
messagePrefix: "\x18Bitcoin Signed Message:\n",
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
throw Exception("Unsupported network: $network");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
String constructDerivePath({
|
||||||
|
required DerivePathType derivePathType,
|
||||||
|
int account = 0,
|
||||||
|
required int chain,
|
||||||
|
required int index,
|
||||||
|
}) {
|
||||||
|
String coinType;
|
||||||
|
|
||||||
|
switch (networkParams.wifPrefix) {
|
||||||
|
case 0x80:
|
||||||
|
switch (derivePathType) {
|
||||||
|
case DerivePathType.bip44:
|
||||||
|
coinType = "145"; // bch mainnet
|
||||||
|
break;
|
||||||
|
case DerivePathType.bch44: // bitcoin.com wallet specific
|
||||||
|
coinType = "0"; // bch mainnet
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw Exception(
|
||||||
|
"DerivePathType $derivePathType not supported for coinType");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 0xef:
|
||||||
|
coinType = "1"; // btc testnet
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw Exception("Invalid Bitcoin network wif used!");
|
||||||
|
}
|
||||||
|
|
||||||
|
final int purpose;
|
||||||
|
switch (derivePathType) {
|
||||||
|
case DerivePathType.bip44:
|
||||||
|
case DerivePathType.bch44:
|
||||||
|
purpose = 44;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw Exception("DerivePathType $derivePathType not supported");
|
||||||
|
}
|
||||||
|
|
||||||
|
return "m/$purpose'/$coinType'/$account'/$chain/$index";
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
({coinlib.Address address, AddressType addressType}) getAddressForPublicKey({
|
||||||
|
required coinlib.ECPublicKey publicKey,
|
||||||
|
required DerivePathType derivePathType,
|
||||||
|
}) {
|
||||||
|
switch (derivePathType) {
|
||||||
|
case DerivePathType.bip44:
|
||||||
|
case DerivePathType.bch44:
|
||||||
|
final addr = coinlib.P2PKHAddress.fromPublicKey(
|
||||||
|
publicKey,
|
||||||
|
version: networkParams.p2pkhPrefix,
|
||||||
|
);
|
||||||
|
|
||||||
|
return (address: addr, addressType: AddressType.p2sh);
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw Exception("DerivePathType $derivePathType not supported");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
// change this to change the number of confirms a tx needs in order to show as confirmed
|
||||||
|
int get minConfirms => 0; // bch zeroconf
|
||||||
|
|
||||||
|
// TODO: [prio=med] bch p2sh addresses (complaints regarding sending to)
|
||||||
|
@override
|
||||||
|
bool validateAddress(String address) {
|
||||||
|
try {
|
||||||
|
// 0 for bitcoincash: address scheme, 1 for legacy address
|
||||||
|
final format = bitbox.Address.detectFormat(address);
|
||||||
|
|
||||||
|
if (coin == Coin.bitcoincashTestnet) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (format == bitbox.Address.formatCashAddr) {
|
||||||
|
return _validateCashAddr(address);
|
||||||
|
} else {
|
||||||
|
return address.startsWith("1");
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _validateCashAddr(String cashAddr) {
|
||||||
|
String addr = cashAddr;
|
||||||
|
if (cashAddr.contains(":")) {
|
||||||
|
addr = cashAddr.split(":").last;
|
||||||
|
}
|
||||||
|
|
||||||
|
return addr.startsWith("q");
|
||||||
|
}
|
||||||
|
}
|
98
lib/wallets/wallet/impl/bitcoincash_wallet.dart
Normal file
98
lib/wallets/wallet/impl/bitcoincash_wallet.dart
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
import 'package:isar/isar.dart';
|
||||||
|
import 'package:stackwallet/models/isar/models/blockchain_data/address.dart';
|
||||||
|
import 'package:stackwallet/services/event_bus/events/global/updated_in_background_event.dart';
|
||||||
|
import 'package:stackwallet/services/event_bus/global_event_bus.dart';
|
||||||
|
import 'package:stackwallet/services/node_service.dart';
|
||||||
|
import 'package:stackwallet/utilities/prefs.dart';
|
||||||
|
import 'package:stackwallet/wallets/crypto_currency/coins/bitcoin.dart';
|
||||||
|
import 'package:stackwallet/wallets/wallet/bip39_hd_wallet.dart';
|
||||||
|
import 'package:stackwallet/wallets/wallet/mixins/electrumx_mixin.dart';
|
||||||
|
import 'package:tuple/tuple.dart';
|
||||||
|
|
||||||
|
class BitcoincashWallet extends Bip39HDWallet with ElectrumXMixin {
|
||||||
|
@override
|
||||||
|
int get isarTransactionVersion => 2;
|
||||||
|
|
||||||
|
BitcoincashWallet(
|
||||||
|
super.cryptoCurrency, {
|
||||||
|
required NodeService nodeService,
|
||||||
|
required Prefs prefs,
|
||||||
|
}) {
|
||||||
|
// TODO: [prio=low] ensure this hack isn't needed
|
||||||
|
assert(cryptoCurrency is Bitcoin);
|
||||||
|
|
||||||
|
this.prefs = prefs;
|
||||||
|
this.nodeService = nodeService;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ===========================================================================
|
||||||
|
|
||||||
|
Future<List<Address>> _fetchAllOwnAddresses() async {
|
||||||
|
final allAddresses = await mainDB
|
||||||
|
.getAddresses(walletId)
|
||||||
|
.filter()
|
||||||
|
.not()
|
||||||
|
.group(
|
||||||
|
(q) => q
|
||||||
|
.typeEqualTo(AddressType.nonWallet)
|
||||||
|
.or()
|
||||||
|
.subTypeEqualTo(AddressSubType.nonWallet),
|
||||||
|
)
|
||||||
|
.findAll();
|
||||||
|
return allAddresses;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ===========================================================================
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> refresh() {
|
||||||
|
// TODO: implement refresh
|
||||||
|
throw UnimplementedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> updateBalance() {
|
||||||
|
// TODO: implement updateBalance
|
||||||
|
throw UnimplementedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> updateTransactions() async {
|
||||||
|
final currentChainHeight = await fetchChainHeight();
|
||||||
|
|
||||||
|
final data = await fetchTransactions(
|
||||||
|
addresses: await _fetchAllOwnAddresses(),
|
||||||
|
currentChainHeight: currentChainHeight,
|
||||||
|
);
|
||||||
|
|
||||||
|
await mainDB.addNewTransactionData(
|
||||||
|
data
|
||||||
|
.map(
|
||||||
|
(e) => Tuple2(
|
||||||
|
e.transaction,
|
||||||
|
e.address,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(),
|
||||||
|
walletId,
|
||||||
|
);
|
||||||
|
|
||||||
|
// TODO: [prio=med] get rid of this and watch isar instead
|
||||||
|
// quick hack to notify manager to call notifyListeners if
|
||||||
|
// transactions changed
|
||||||
|
if (data.isNotEmpty) {
|
||||||
|
GlobalEventBus.instance.fire(
|
||||||
|
UpdatedInBackgroundEvent(
|
||||||
|
"Transactions updated/added for: $walletId ${walletInfo.name}",
|
||||||
|
walletId,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> updateUTXOs() {
|
||||||
|
// TODO: implement updateUTXOs
|
||||||
|
throw UnimplementedError();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue