cake_wallet/lib/bitcoin/electrum.dart

456 lines
12 KiB
Dart
Raw Normal View History

2020-05-12 12:04:54 +00:00
import 'dart:async';
import 'dart:convert';
import 'dart:io';
2020-08-31 14:39:12 +00:00
import 'dart:typed_data';
import 'package:bitcoin_flutter/bitcoin_flutter.dart';
2021-02-12 22:38:34 +00:00
import 'package:cake_wallet/bitcoin/bitcoin_amount_format.dart';
2020-08-25 16:32:40 +00:00
import 'package:cake_wallet/bitcoin/script_hash.dart';
2020-05-12 12:04:54 +00:00
import 'package:flutter/foundation.dart';
import 'package:rxdart/rxdart.dart';
2020-09-09 14:13:44 +00:00
class UriParseException implements Exception {
UriParseException(this.uri);
final String uri;
@override
String toString() =>
'Cannot parse host and port from uri. Invalid uri format. Uri: $uri';
}
2020-05-12 12:04:54 +00:00
String jsonrpcparams(List<Object> params) {
final _params = params?.map((val) => '"${val.toString()}"')?.join(',');
2020-08-25 16:32:40 +00:00
return '[$_params]';
2020-05-12 12:04:54 +00:00
}
String jsonrpc(
2021-02-12 22:38:34 +00:00
{String method, List<Object> params, int id, double version = 2.0}) =>
'{"jsonrpc": "$version", "method": "$method", "id": "$id", "params": ${json.encode(params)}}\n';
2020-05-12 12:04:54 +00:00
class SocketTask {
SocketTask({this.completer, this.isSubscription, this.subject});
final Completer completer;
final BehaviorSubject subject;
final bool isSubscription;
}
class ElectrumClient {
ElectrumClient()
: _id = 0,
_isConnected = false,
_tasks = {},
unterminatedString = '';
2020-05-12 12:04:54 +00:00
static const connectionTimeout = Duration(seconds: 5);
2020-09-21 11:50:26 +00:00
static const aliveTimerDuration = Duration(seconds: 2);
2020-05-12 12:04:54 +00:00
bool get isConnected => _isConnected;
Socket socket;
2020-08-29 10:19:27 +00:00
void Function(bool) onConnectionStatusChange;
2020-05-12 12:04:54 +00:00
int _id;
final Map<String, SocketTask> _tasks;
bool _isConnected;
2020-07-06 20:09:03 +00:00
Timer _aliveTimer;
String unterminatedString;
2020-05-12 12:04:54 +00:00
2020-08-27 16:54:34 +00:00
Future<void> connectToUri(String uri) async {
2020-09-09 14:13:44 +00:00
final splittedUri = uri.split(':');
if (splittedUri.length != 2) {
throw UriParseException(uri);
}
final host = splittedUri.first;
final port = int.parse(splittedUri.last);
2020-08-27 16:54:34 +00:00
await connect(host: host, port: port);
}
2020-05-12 12:04:54 +00:00
2020-08-27 16:54:34 +00:00
Future<void> connect({@required String host, @required int port}) async {
2020-08-29 10:19:27 +00:00
try {
await socket?.close();
} catch (_) {}
2020-05-12 12:04:54 +00:00
2020-09-09 14:13:44 +00:00
socket = await SecureSocket.connect(host, port,
timeout: connectionTimeout, onBadCertificate: (_) => true);
2020-08-29 10:19:27 +00:00
_setIsConnected(true);
2020-08-31 14:39:12 +00:00
socket.listen((Uint8List event) {
2020-05-12 12:04:54 +00:00
try {
final response =
2021-02-12 22:38:34 +00:00
json.decode(utf8.decode(event.toList())) as Map<String, Object>;
_handleResponse(response);
} on FormatException catch (e) {
final msg = e.message.toLowerCase();
if (e.source is String) {
unterminatedString += e.source as String;
2020-05-12 12:04:54 +00:00
}
if (msg.contains("not a subtype of type")) {
unterminatedString += e.source as String;
return;
}
if (isJSONStringCorrect(unterminatedString)) {
final response =
2021-02-12 22:38:34 +00:00
json.decode(unterminatedString) as Map<String, Object>;
_handleResponse(response);
unterminatedString = '';
}
} on TypeError catch (e) {
if (!e.toString().contains('Map<String, Object>')) {
return;
}
final source = utf8.decode(event.toList());
unterminatedString += source;
if (isJSONStringCorrect(unterminatedString)) {
final response =
2021-02-12 22:38:34 +00:00
json.decode(unterminatedString) as Map<String, Object>;
_handleResponse(response);
unterminatedString = null;
}
2020-05-12 12:04:54 +00:00
} catch (e) {
print(e.toString());
2020-05-12 12:04:54 +00:00
}
2020-08-31 14:39:12 +00:00
}, onError: (Object error) {
print(error.toString());
_setIsConnected(false);
}, onDone: () {
_setIsConnected(false);
});
2020-07-06 20:09:03 +00:00
keepAlive();
}
void keepAlive() {
_aliveTimer?.cancel();
2020-09-21 11:50:26 +00:00
_aliveTimer = Timer.periodic(aliveTimerDuration, (_) async => ping());
2020-05-12 12:04:54 +00:00
}
2020-08-29 10:19:27 +00:00
Future<void> ping() async {
try {
await callWithTimeout(method: 'server.ping');
2020-08-29 10:19:27 +00:00
_setIsConnected(true);
} on RequestFailedTimeoutException catch (_) {
_setIsConnected(false);
}
}
2020-05-12 12:04:54 +00:00
Future<List<String>> version() =>
call(method: 'server.version').then((dynamic result) {
if (result is List) {
return result.map((dynamic val) => val.toString()).toList();
}
return [];
});
2020-08-25 16:32:40 +00:00
Future<Map<String, Object>> getBalance(String scriptHash) =>
call(method: 'blockchain.scripthash.get_balance', params: [scriptHash])
2020-05-12 12:04:54 +00:00
.then((dynamic result) {
if (result is Map<String, Object>) {
return result;
}
2020-08-25 16:32:40 +00:00
return <String, Object>{};
2020-05-12 12:04:54 +00:00
});
2020-08-25 16:32:40 +00:00
Future<List<Map<String, dynamic>>> getHistory(String scriptHash) =>
call(method: 'blockchain.scripthash.get_history', params: [scriptHash])
2020-05-12 12:04:54 +00:00
.then((dynamic result) {
if (result is List) {
return result.map((dynamic val) {
if (val is Map<String, Object>) {
return val;
}
2020-08-25 16:32:40 +00:00
return <String, Object>{};
2020-05-12 12:04:54 +00:00
}).toList();
}
return [];
});
2020-08-25 16:32:40 +00:00
Future<List<Map<String, dynamic>>> getListUnspentWithAddress(
String address, NetworkType networkType) =>
2020-08-25 16:32:40 +00:00
call(
method: 'blockchain.scripthash.listunspent',
params: [scriptHash(address, networkType: networkType)])
.then((dynamic result) {
2020-08-25 16:32:40 +00:00
if (result is List) {
return result.map((dynamic val) {
if (val is Map<String, Object>) {
val['address'] = address;
return val;
}
return <String, Object>{};
}).toList();
}
return [];
});
Future<List<Map<String, dynamic>>> getListUnspent(String scriptHash) =>
call(method: 'blockchain.scripthash.listunspent', params: [scriptHash])
2020-05-12 12:04:54 +00:00
.then((dynamic result) {
2020-08-25 16:32:40 +00:00
if (result is List) {
return result.map((dynamic val) {
if (val is Map<String, Object>) {
return val;
}
return <String, Object>{};
}).toList();
}
return [];
});
Future<List<Map<String, dynamic>>> getMempool(String scriptHash) =>
call(method: 'blockchain.scripthash.get_mempool', params: [scriptHash])
.then((dynamic result) {
if (result is List) {
return result.map((dynamic val) {
if (val is Map<String, Object>) {
return val;
}
return <String, Object>{};
}).toList();
}
return [];
});
Future<Map<String, Object>> getTransactionRaw(
{@required String hash}) async =>
2020-08-25 16:32:40 +00:00
call(method: 'blockchain.transaction.get', params: [hash, true])
.then((dynamic result) {
if (result is Map<String, Object>) {
2020-05-12 12:04:54 +00:00
return result;
}
2020-08-25 16:32:40 +00:00
return <String, Object>{};
2020-05-12 12:04:54 +00:00
});
2020-08-25 16:32:40 +00:00
Future<Map<String, Object>> getTransactionExpanded(
{@required String hash}) async {
2020-11-30 17:17:44 +00:00
try {
final originalTx = await getTransactionRaw(hash: hash);
final vins = originalTx['vin'] as List<Object>;
2020-08-25 16:32:40 +00:00
2020-11-30 17:17:44 +00:00
for (dynamic vin in vins) {
if (vin is Map<String, Object>) {
vin['tx'] = await getTransactionRaw(hash: vin['txid'] as String);
}
2020-08-25 16:32:40 +00:00
}
2020-11-30 17:17:44 +00:00
return originalTx;
} catch (_) {
return {};
}
2020-08-25 16:32:40 +00:00
}
Future<String> broadcastTransaction(
2021-02-12 22:38:34 +00:00
{@required String transactionRaw}) async =>
2020-07-06 20:09:03 +00:00
call(method: 'blockchain.transaction.broadcast', params: [transactionRaw])
.then((dynamic result) {
if (result is String) {
return result;
}
2020-07-06 20:09:03 +00:00
return '';
});
2020-05-12 12:04:54 +00:00
Future<Map<String, dynamic>> getMerkle(
2021-02-12 22:38:34 +00:00
{@required String hash, @required int height}) async =>
2020-05-12 12:04:54 +00:00
await call(
method: 'blockchain.transaction.get_merkle',
params: [hash, height]) as Map<String, dynamic>;
Future<Map<String, dynamic>> getHeader({@required int height}) async =>
await call(method: 'blockchain.block.get_header', params: [height])
2021-02-12 22:38:34 +00:00
as Map<String, dynamic>;
2020-05-12 12:04:54 +00:00
Future<double> estimatefee({@required int p}) =>
call(method: 'blockchain.estimatefee', params: [p])
.then((dynamic result) {
if (result is double) {
return result;
}
if (result is String) {
return double.parse(result);
}
return 0;
});
2021-02-12 22:38:34 +00:00
Future<List<List<int>>> feeHistogram() =>
call(method: 'mempool.get_fee_histogram').then((dynamic result) {
if (result is List) {
return result.map((dynamic e) {
if (e is List) {
return e.map((dynamic ee) => ee is int ? ee : null).toList();
}
return null;
}).toList();
}
return [];
});
Future<List<int>> feeRates() async {
try {
final topDoubleString = await estimatefee(p: 1);
final middleDoubleString = await estimatefee(p: 20);
2021-05-07 07:39:08 +00:00
final bottomDoubleString = await estimatefee(p: 100);
final top =
(stringDoubleToBitcoinAmount(topDoubleString.toString()) / 1000)
.round();
final middle =
(stringDoubleToBitcoinAmount(middleDoubleString.toString()) / 1000)
.round();
final bottom =
(stringDoubleToBitcoinAmount(bottomDoubleString.toString()) / 1000)
.round();
2021-05-07 07:39:08 +00:00
return [bottom, middle, top];
} catch (_) {
return [];
}
2021-02-12 22:38:34 +00:00
}
2020-11-30 17:17:44 +00:00
BehaviorSubject<Object> scripthashUpdate(String scripthash) {
_id += 1;
return subscribe<Object>(
id: 'blockchain.scripthash.subscribe:$scripthash',
method: 'blockchain.scripthash.subscribe',
params: [scripthash]);
}
2020-05-12 12:04:54 +00:00
2021-02-12 22:38:34 +00:00
BehaviorSubject<T> subscribe<T>(
{@required String id,
@required String method,
List<Object> params = const []}) {
2020-05-12 12:04:54 +00:00
final subscription = BehaviorSubject<T>();
_regisrySubscription(id, subscription);
socket.write(jsonrpc(method: method, id: _id, params: params));
return subscription;
}
2020-11-30 17:17:44 +00:00
Future<dynamic> call({String method, List<Object> params = const []}) async {
2020-05-12 12:04:54 +00:00
final completer = Completer<dynamic>();
_id += 1;
final id = _id;
2021-01-27 13:51:51 +00:00
_registryTask(id, completer);
socket.write(jsonrpc(method: method, id: id, params: params));
2020-05-12 12:04:54 +00:00
return completer.future;
}
2021-02-12 22:38:34 +00:00
Future<dynamic> callWithTimeout(
{String method,
List<Object> params = const [],
int timeout = 2000}) async {
2020-08-29 10:19:27 +00:00
final completer = Completer<dynamic>();
_id += 1;
final id = _id;
2021-01-27 13:51:51 +00:00
_registryTask(id, completer);
socket.write(jsonrpc(method: method, id: id, params: params));
2020-08-29 10:19:27 +00:00
Timer(Duration(milliseconds: timeout), () {
if (!completer.isCompleted) {
2021-01-27 13:51:51 +00:00
completer.completeError(RequestFailedTimeoutException(method, id));
2020-08-29 10:19:27 +00:00
}
});
2021-01-27 13:51:51 +00:00
return completer.future;
2020-05-12 12:04:54 +00:00
}
Future<void> close() async {
_aliveTimer.cancel();
await socket.close();
onConnectionStatusChange = null;
}
2021-02-12 22:38:34 +00:00
void _registryTask(int id, Completer completer) => _tasks[id.toString()] =
SocketTask(completer: completer, isSubscription: false);
2020-05-12 12:04:54 +00:00
void _regisrySubscription(String id, BehaviorSubject subject) =>
_tasks[id] = SocketTask(subject: subject, isSubscription: true);
void _finish(String id, Object data) {
if (_tasks[id] == null) {
return;
}
2020-08-29 10:19:27 +00:00
if (!(_tasks[id]?.completer?.isCompleted ?? false)) {
_tasks[id]?.completer?.complete(data);
}
2020-05-12 12:04:54 +00:00
if (!(_tasks[id]?.isSubscription ?? false)) {
_tasks[id] = null;
} else {
_tasks[id].subject.add(data);
}
}
void _methodHandler(
{@required String method, @required Map<String, Object> request}) {
switch (method) {
2020-08-25 16:32:40 +00:00
case 'blockchain.scripthash.subscribe':
2020-05-12 12:04:54 +00:00
final params = request['params'] as List<dynamic>;
2020-08-25 16:32:40 +00:00
final scripthash = params.first as String;
final id = 'blockchain.scripthash.subscribe:$scripthash';
2020-05-12 12:04:54 +00:00
2020-08-25 16:32:40 +00:00
_tasks[id]?.subject?.add(params.last);
2020-05-12 12:04:54 +00:00
break;
default:
break;
}
}
2020-08-29 10:19:27 +00:00
void _setIsConnected(bool isConnected) {
if (_isConnected != isConnected) {
onConnectionStatusChange?.call(isConnected);
}
_isConnected = isConnected;
}
void _handleResponse(Map<String, Object> response) {
final method = response['method'];
final id = response['id'] as String;
final result = response['result'];
if (method is String) {
_methodHandler(method: method, request: response);
return;
}
_finish(id, result);
}
}
// FIXME: move me
bool isJSONStringCorrect(String source) {
try {
json.decode(source);
return true;
} catch (_) {
return false;
}
2020-08-29 10:19:27 +00:00
}
class RequestFailedTimeoutException implements Exception {
RequestFailedTimeoutException(this.method, this.id);
final String method;
final int id;
2020-05-12 12:04:54 +00:00
}