2023-11-14 14:52:32 +00:00
|
|
|
import 'dart:typed_data';
|
|
|
|
|
|
|
|
import 'package:bech32/bech32.dart';
|
|
|
|
import 'package:bitbox/bitbox.dart' as bitbox;
|
|
|
|
import 'package:bs58check/bs58check.dart' as bs58check;
|
|
|
|
import 'package:coinlib_flutter/coinlib_flutter.dart' as coinlib;
|
2024-05-27 23:56:22 +00:00
|
|
|
|
2024-05-23 00:37:06 +00:00
|
|
|
import '../../../models/isar/models/blockchain_data/address.dart';
|
|
|
|
import '../../../models/node_model.dart';
|
|
|
|
import '../../../utilities/amount/amount.dart';
|
|
|
|
import '../../../utilities/default_nodes.dart';
|
|
|
|
import '../../../utilities/enums/derive_path_type_enum.dart';
|
|
|
|
import '../crypto_currency.dart';
|
|
|
|
import '../interfaces/electrumx_currency_interface.dart';
|
|
|
|
import '../intermediate/bip39_hd_currency.dart';
|
2023-11-14 14:52:32 +00:00
|
|
|
|
2024-05-15 21:20:45 +00:00
|
|
|
class Ecash extends Bip39HDCurrency with ElectrumXCurrencyInterface {
|
2023-11-14 14:52:32 +00:00
|
|
|
Ecash(super.network) {
|
2024-05-15 21:20:45 +00:00
|
|
|
_idMain = "eCash";
|
2023-11-14 14:52:32 +00:00
|
|
|
switch (network) {
|
|
|
|
case CryptoCurrencyNetwork.main:
|
2024-05-15 21:20:45 +00:00
|
|
|
_id = _idMain;
|
|
|
|
_name = "eCash";
|
|
|
|
_ticker = "XEC";
|
|
|
|
_uriScheme = "ecash";
|
2023-11-14 14:52:32 +00:00
|
|
|
default:
|
|
|
|
throw Exception("Unsupported network: $network");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-15 21:20:45 +00:00
|
|
|
late final String _id;
|
|
|
|
@override
|
|
|
|
String get identifier => _id;
|
|
|
|
|
|
|
|
late final String _idMain;
|
|
|
|
@override
|
|
|
|
String get mainNetId => _idMain;
|
|
|
|
|
|
|
|
late final String _name;
|
|
|
|
@override
|
|
|
|
String get prettyName => _name;
|
|
|
|
|
|
|
|
late final String _uriScheme;
|
|
|
|
@override
|
|
|
|
String get uriScheme => _uriScheme;
|
|
|
|
|
|
|
|
late final String _ticker;
|
|
|
|
@override
|
|
|
|
String get ticker => _ticker;
|
|
|
|
|
2023-11-14 14:52:32 +00:00
|
|
|
@override
|
|
|
|
int get maxUnusedAddressGap => 50;
|
|
|
|
|
|
|
|
@override
|
|
|
|
// change this to change the number of confirms a tx needs in order to show as confirmed
|
|
|
|
int get minConfirms => 0; // bch zeroconf
|
|
|
|
|
2024-04-16 22:42:51 +00:00
|
|
|
@override
|
|
|
|
bool get torSupport => true;
|
|
|
|
|
2023-11-14 14:52:32 +00:00
|
|
|
@override
|
|
|
|
List<DerivePathType> get supportedDerivationPathTypes => [
|
2023-11-14 19:04:10 +00:00
|
|
|
DerivePathType.eCash44,
|
2023-11-14 14:52:32 +00:00
|
|
|
DerivePathType.bip44,
|
|
|
|
];
|
|
|
|
|
|
|
|
@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
|
2024-03-22 21:19:48 +00:00
|
|
|
coinlib.Network get networkParams {
|
2023-11-14 14:52:32 +00:00
|
|
|
switch (network) {
|
|
|
|
case CryptoCurrencyNetwork.main:
|
2024-03-22 21:19:48 +00:00
|
|
|
return coinlib.Network(
|
2023-11-14 14:52:32 +00:00
|
|
|
wifPrefix: 0x80,
|
|
|
|
p2pkhPrefix: 0x00,
|
|
|
|
p2shPrefix: 0x05,
|
|
|
|
privHDPrefix: 0x0488ade4,
|
|
|
|
pubHDPrefix: 0x0488b21e,
|
|
|
|
bech32Hrp: "bc",
|
|
|
|
messagePrefix: '\x18Bitcoin Signed Message:\n',
|
2024-06-07 14:45:56 +00:00
|
|
|
minFee: BigInt.from(1), // Not used in stack wallet currently
|
|
|
|
minOutput: dustLimit.raw, // Not used in stack wallet currently
|
|
|
|
feePerKb: BigInt.from(1), // Not used in stack wallet currently
|
2023-11-14 14:52:32 +00:00
|
|
|
);
|
|
|
|
case CryptoCurrencyNetwork.test:
|
2024-03-22 21:19:48 +00:00
|
|
|
return coinlib.Network(
|
2023-11-14 14:52:32 +00:00
|
|
|
wifPrefix: 0xef,
|
|
|
|
p2pkhPrefix: 0x6f,
|
|
|
|
p2shPrefix: 0xc4,
|
|
|
|
privHDPrefix: 0x04358394,
|
|
|
|
pubHDPrefix: 0x043587cf,
|
|
|
|
bech32Hrp: "tb",
|
|
|
|
messagePrefix: "\x18Bitcoin Signed Message:\n",
|
2024-06-07 14:45:56 +00:00
|
|
|
minFee: BigInt.from(1), // Not used in stack wallet currently
|
|
|
|
minOutput: dustLimit.raw, // Not used in stack wallet currently
|
|
|
|
feePerKb: BigInt.from(1), // Not used in stack wallet currently
|
2023-11-14 14:52:32 +00:00
|
|
|
);
|
|
|
|
default:
|
|
|
|
throw Exception("Unsupported network: $network");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-14 15:56:07 +00:00
|
|
|
@override
|
|
|
|
String addressToScriptHash({required String address}) {
|
|
|
|
try {
|
|
|
|
if (bitbox.Address.detectFormat(address) ==
|
|
|
|
bitbox.Address.formatCashAddr &&
|
|
|
|
_validateCashAddr(address)) {
|
|
|
|
address = bitbox.Address.toLegacyAddress(address);
|
|
|
|
}
|
|
|
|
|
|
|
|
final addr = coinlib.Address.fromString(address, networkParams);
|
|
|
|
return Bip39HDCurrency.convertBytesToScriptHash(
|
2024-05-27 23:56:22 +00:00
|
|
|
addr.program.script.compiled,
|
|
|
|
);
|
2023-11-14 15:56:07 +00:00
|
|
|
} catch (e) {
|
|
|
|
rethrow;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-14 14:52:32 +00:00
|
|
|
@override
|
|
|
|
String constructDerivePath({
|
|
|
|
required DerivePathType derivePathType,
|
|
|
|
int account = 0,
|
|
|
|
required int chain,
|
|
|
|
required int index,
|
|
|
|
}) {
|
|
|
|
String coinType;
|
|
|
|
switch (networkParams.wifPrefix) {
|
|
|
|
case 0x80: // mainnet wif
|
|
|
|
switch (derivePathType) {
|
|
|
|
case DerivePathType.bip44:
|
|
|
|
coinType = "145";
|
|
|
|
break;
|
|
|
|
case DerivePathType.eCash44:
|
|
|
|
coinType = "899";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw Exception(
|
2024-05-27 23:56:22 +00:00
|
|
|
"DerivePathType $derivePathType not supported for coinType",
|
|
|
|
);
|
2023-11-14 14:52:32 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0xef: // testnet wif
|
|
|
|
throw Exception(
|
2024-05-27 23:56:22 +00:00
|
|
|
"DerivePathType $derivePathType not supported for coinType",
|
|
|
|
);
|
2023-11-14 14:52:32 +00:00
|
|
|
default:
|
|
|
|
throw Exception("Invalid ECash network wif used!");
|
|
|
|
}
|
|
|
|
|
|
|
|
int purpose;
|
|
|
|
switch (derivePathType) {
|
|
|
|
case DerivePathType.bip44:
|
|
|
|
case DerivePathType.eCash44:
|
|
|
|
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.eCash44:
|
|
|
|
final addr = coinlib.P2PKHAddress.fromPublicKey(
|
|
|
|
publicKey,
|
|
|
|
version: networkParams.p2pkhPrefix,
|
|
|
|
);
|
|
|
|
|
|
|
|
return (address: addr, addressType: AddressType.p2pkh);
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw Exception("DerivePathType $derivePathType not supported");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool validateAddress(String address) {
|
|
|
|
try {
|
|
|
|
// 0 for bitcoincash: address scheme, 1 for legacy address
|
|
|
|
final format = bitbox.Address.detectFormat(address);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2024-02-26 16:23:34 +00:00
|
|
|
return addr.startsWith("q") /*|| addr.startsWith("p")*/;
|
|
|
|
// Do not validate "p" (P2SH) addresses.
|
2023-11-14 14:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
DerivePathType addressType({required String address}) {
|
|
|
|
Uint8List? decodeBase58;
|
|
|
|
Segwit? decodeBech32;
|
|
|
|
try {
|
|
|
|
if (bitbox.Address.detectFormat(address) ==
|
|
|
|
bitbox.Address.formatCashAddr) {
|
|
|
|
if (_validateCashAddr(address)) {
|
|
|
|
address = bitbox.Address.toLegacyAddress(address);
|
|
|
|
} else {
|
|
|
|
throw ArgumentError('$address is not currently supported');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (_) {
|
|
|
|
// invalid cash addr format
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
decodeBase58 = bs58check.decode(address);
|
|
|
|
} catch (err) {
|
|
|
|
// Base58check decode fail
|
|
|
|
}
|
|
|
|
if (decodeBase58 != null) {
|
|
|
|
if (decodeBase58[0] == networkParams.p2pkhPrefix) {
|
|
|
|
// P2PKH
|
|
|
|
return DerivePathType.bip44;
|
|
|
|
}
|
|
|
|
if (decodeBase58[0] == networkParams.p2shPrefix) {
|
|
|
|
// P2SH
|
|
|
|
return DerivePathType.bip49;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw ArgumentError('Invalid version or Network mismatch');
|
|
|
|
} else {
|
|
|
|
try {
|
|
|
|
decodeBech32 = segwit.decode(address);
|
|
|
|
} catch (err) {
|
|
|
|
// Bech32 decode fail
|
|
|
|
}
|
|
|
|
|
|
|
|
if (decodeBech32 != null) {
|
|
|
|
if (networkParams.bech32Hrp != decodeBech32.hrp) {
|
|
|
|
throw ArgumentError('Invalid prefix or Network mismatch');
|
|
|
|
}
|
|
|
|
if (decodeBech32.version != 0) {
|
|
|
|
throw ArgumentError('Invalid address version');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw ArgumentError('$address has no matching Script');
|
|
|
|
}
|
2023-11-17 01:44:17 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
NodeModel get defaultNode {
|
|
|
|
switch (network) {
|
|
|
|
case CryptoCurrencyNetwork.main:
|
|
|
|
return NodeModel(
|
2024-06-03 20:50:16 +00:00
|
|
|
host: "ecash.stackwallet.com",
|
|
|
|
port: 59002,
|
2023-11-17 01:44:17 +00:00
|
|
|
name: DefaultNodes.defaultName,
|
2024-05-15 21:20:45 +00:00
|
|
|
id: DefaultNodes.buildId(this),
|
2023-11-17 01:44:17 +00:00
|
|
|
useSSL: true,
|
|
|
|
enabled: true,
|
2024-05-15 21:20:45 +00:00
|
|
|
coinName: identifier,
|
2023-11-17 01:44:17 +00:00
|
|
|
isFailover: true,
|
|
|
|
isDown: false,
|
2024-11-25 19:33:58 +00:00
|
|
|
torEnabled: true,
|
2024-11-26 15:18:35 +00:00
|
|
|
clearnetEnabled: true,
|
2023-11-17 01:44:17 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw UnimplementedError();
|
|
|
|
}
|
|
|
|
}
|
2024-04-17 18:04:10 +00:00
|
|
|
|
|
|
|
@override
|
2024-05-15 21:20:45 +00:00
|
|
|
int get defaultSeedPhraseLength => 12;
|
|
|
|
|
|
|
|
@override
|
|
|
|
int get fractionDigits => 2;
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool get hasBuySupport => false;
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool get hasMnemonicPassphraseSupport => true;
|
2024-04-17 18:04:10 +00:00
|
|
|
|
|
|
|
@override
|
2024-05-15 21:20:45 +00:00
|
|
|
List<int> get possibleMnemonicLengths => [defaultSeedPhraseLength, 24];
|
|
|
|
|
|
|
|
@override
|
2024-06-20 16:16:12 +00:00
|
|
|
AddressType get defaultAddressType => defaultDerivePathType.getAddressType();
|
2024-05-15 21:20:45 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
BigInt get satsPerCoin => BigInt.from(100);
|
|
|
|
|
|
|
|
@override
|
|
|
|
int get targetBlockTimeSeconds => 600;
|
|
|
|
|
|
|
|
@override
|
2024-06-20 16:16:12 +00:00
|
|
|
DerivePathType get defaultDerivePathType => DerivePathType.eCash44;
|
2024-05-15 21:20:45 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
Uri defaultBlockExplorer(String txid) {
|
|
|
|
switch (network) {
|
|
|
|
case CryptoCurrencyNetwork.main:
|
|
|
|
return Uri.parse("https://explorer.e.cash/tx/$txid");
|
|
|
|
default:
|
|
|
|
throw Exception(
|
|
|
|
"Unsupported network for defaultBlockExplorer(): $network",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2024-06-05 22:19:33 +00:00
|
|
|
|
|
|
|
@override
|
2024-06-05 23:36:32 +00:00
|
|
|
int get transactionVersion => 2;
|
2024-06-07 20:56:57 +00:00
|
|
|
|
|
|
|
@override
|
2024-06-07 22:04:42 +00:00
|
|
|
BigInt get defaultFeeRate => BigInt.from(200);
|
2023-11-14 14:52:32 +00:00
|
|
|
}
|