2023-05-26 21:21:16 +00:00
|
|
|
/*
|
|
|
|
* This file is part of Stack Wallet.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2023 Cypher Stack
|
|
|
|
* All Rights Reserved.
|
|
|
|
* The code is distributed under GPLv3 license, see LICENSE file for details.
|
|
|
|
* Generated by Cypher Stack on 2023-05-26
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2022-08-26 08:11:35 +00:00
|
|
|
import 'dart:convert';
|
2022-11-08 16:18:48 +00:00
|
|
|
import 'dart:io';
|
2022-08-26 08:11:35 +00:00
|
|
|
|
2022-11-08 16:18:48 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2024-06-12 21:12:53 +00:00
|
|
|
import 'package:socks5_proxy/socks.dart';
|
2024-06-23 00:30:24 +00:00
|
|
|
import 'package:tor_ffi_plugin/socks_socket.dart';
|
2024-06-04 15:16:57 +00:00
|
|
|
|
2024-05-23 00:37:06 +00:00
|
|
|
import '../widgets/desktop/primary_button.dart';
|
|
|
|
import '../widgets/desktop/secondary_button.dart';
|
|
|
|
import '../widgets/stack_dialog.dart';
|
2024-06-04 15:16:57 +00:00
|
|
|
import 'format.dart';
|
|
|
|
import 'logger.dart';
|
2022-08-26 08:11:35 +00:00
|
|
|
|
2022-11-08 16:18:48 +00:00
|
|
|
class MoneroNodeConnectionResponse {
|
|
|
|
final X509Certificate? cert;
|
|
|
|
final String? url;
|
|
|
|
final int? port;
|
|
|
|
final bool success;
|
|
|
|
|
|
|
|
MoneroNodeConnectionResponse(this.cert, this.url, this.port, this.success);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<MoneroNodeConnectionResponse> testMoneroNodeConnection(
|
|
|
|
Uri uri,
|
2024-06-12 21:12:53 +00:00
|
|
|
bool allowBadX509Certificate, {
|
|
|
|
required ({
|
|
|
|
InternetAddress host,
|
|
|
|
int port,
|
|
|
|
})? proxyInfo,
|
|
|
|
}) async {
|
|
|
|
final httpClient = HttpClient();
|
2022-11-08 16:18:48 +00:00
|
|
|
MoneroNodeConnectionResponse? badCertResponse;
|
2024-06-12 21:12:53 +00:00
|
|
|
|
2022-08-26 08:11:35 +00:00
|
|
|
try {
|
2024-06-12 21:12:53 +00:00
|
|
|
if (proxyInfo != null) {
|
|
|
|
SocksTCPClient.assignToHttpClient(httpClient, [
|
|
|
|
ProxySettings(
|
|
|
|
proxyInfo.host,
|
|
|
|
proxyInfo.port,
|
|
|
|
),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
httpClient.badCertificateCallback = (cert, url, port) {
|
2022-11-08 16:18:48 +00:00
|
|
|
if (allowBadX509Certificate) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (badCertResponse == null) {
|
|
|
|
badCertResponse = MoneroNodeConnectionResponse(cert, url, port, false);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2024-07-30 22:32:05 +00:00
|
|
|
if (proxyInfo == null) {
|
|
|
|
// If the host ends in .onion, we can't access it without Tor.
|
|
|
|
if (uri.host.endsWith('.onion')) {
|
|
|
|
return MoneroNodeConnectionResponse(null, null, null, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Proxyless connections can use an HttpClient; proxied ones cannot.
|
2024-06-23 00:30:24 +00:00
|
|
|
final request = await httpClient.postUrl(uri);
|
|
|
|
|
|
|
|
final body = utf8.encode(
|
|
|
|
jsonEncode({
|
|
|
|
"jsonrpc": "2.0",
|
|
|
|
"id": "0",
|
|
|
|
"method": "get_info",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
request.headers.add(
|
|
|
|
'Content-Length',
|
|
|
|
body.length.toString(),
|
|
|
|
preserveHeaderCase: true,
|
|
|
|
);
|
|
|
|
request.headers.set(
|
|
|
|
'Content-Type',
|
|
|
|
'application/json',
|
|
|
|
preserveHeaderCase: true,
|
|
|
|
);
|
|
|
|
|
|
|
|
request.add(body);
|
|
|
|
|
|
|
|
final response = await request.close();
|
|
|
|
final result = await response.transform(utf8.decoder).join();
|
|
|
|
// TODO: json decoded without error so assume connection exists?
|
|
|
|
// or we can check for certain values in the response to decide
|
|
|
|
return MoneroNodeConnectionResponse(null, null, null, true);
|
|
|
|
} else {
|
2024-07-30 22:32:05 +00:00
|
|
|
// Proxied connections cannot use an HttpClient.
|
2024-06-23 00:30:24 +00:00
|
|
|
//
|
|
|
|
// The SOCKSSocket class from the tor_ffi_plugin package can be used to
|
|
|
|
// connect to .onion addresses. We'll do the same things as above but
|
|
|
|
// with SOCKSSocket instead of httpClient.
|
|
|
|
final socket = await SOCKSSocket.create(
|
2024-07-30 22:32:05 +00:00
|
|
|
proxyHost: proxyInfo.host.address,
|
2024-06-23 00:30:24 +00:00
|
|
|
proxyPort: proxyInfo.port,
|
|
|
|
sslEnabled: false,
|
|
|
|
);
|
|
|
|
await socket.connect();
|
|
|
|
await socket.connectTo(uri.host, uri.port);
|
|
|
|
|
|
|
|
final body = utf8.encode(
|
|
|
|
jsonEncode({
|
|
|
|
"jsonrpc": "2.0",
|
|
|
|
"id": "0",
|
|
|
|
"method": "get_info",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
// Write the request body to the socket.
|
|
|
|
socket.write(body);
|
|
|
|
|
2024-07-30 22:32:05 +00:00
|
|
|
// If this is an onion address and there are no errors yet, assume success.
|
|
|
|
if (uri.host.endsWith('.onion')) {
|
|
|
|
return MoneroNodeConnectionResponse(null, null, null, true);
|
|
|
|
}
|
|
|
|
|
2024-06-23 00:30:24 +00:00
|
|
|
// Read the response.
|
|
|
|
final response = await socket.inputStream.first;
|
|
|
|
final result = utf8.decode(response);
|
|
|
|
|
|
|
|
// Close the socket.
|
|
|
|
await socket.close();
|
|
|
|
return MoneroNodeConnectionResponse(null, null, null, true);
|
|
|
|
|
|
|
|
// Parse the response.
|
|
|
|
//
|
|
|
|
// This is commented because any issues should throw.
|
|
|
|
// final Map<String, dynamic> jsonResponse = jsonDecode(result);
|
|
|
|
// print(jsonResponse);
|
|
|
|
// if (jsonResponse.containsKey('result')) {
|
|
|
|
// return MoneroNodeConnectionResponse(null, null, null, true);
|
|
|
|
// } else {
|
|
|
|
// return MoneroNodeConnectionResponse(null, null, null, false);
|
|
|
|
// }
|
|
|
|
}
|
2022-08-26 08:11:35 +00:00
|
|
|
} catch (e, s) {
|
2022-11-08 16:18:48 +00:00
|
|
|
if (badCertResponse != null) {
|
|
|
|
return badCertResponse!;
|
|
|
|
} else {
|
|
|
|
Logging.instance.log("$e\n$s", level: LogLevel.Warning);
|
|
|
|
return MoneroNodeConnectionResponse(null, null, null, false);
|
|
|
|
}
|
|
|
|
} finally {
|
2024-06-12 21:12:53 +00:00
|
|
|
httpClient.close(force: true);
|
2022-08-26 08:11:35 +00:00
|
|
|
}
|
|
|
|
}
|
2022-11-08 16:18:48 +00:00
|
|
|
|
|
|
|
Future<bool> showBadX509CertificateDialog(
|
|
|
|
X509Certificate cert,
|
|
|
|
String url,
|
|
|
|
int port,
|
|
|
|
BuildContext context,
|
|
|
|
) async {
|
2022-11-08 17:41:12 +00:00
|
|
|
final chars = Format.uint8listToString(cert.sha1)
|
|
|
|
.toUpperCase()
|
|
|
|
.characters
|
|
|
|
.toList(growable: false);
|
|
|
|
|
|
|
|
String sha1 = chars.sublist(0, 2).join();
|
|
|
|
for (int i = 2; i < chars.length; i += 2) {
|
|
|
|
sha1 += ":${chars.sublist(i, i + 2).join()}";
|
|
|
|
}
|
|
|
|
|
2022-11-08 16:18:48 +00:00
|
|
|
final result = await showDialog<bool>(
|
|
|
|
context: context,
|
|
|
|
barrierDismissible: false,
|
|
|
|
builder: (context) {
|
|
|
|
return StackDialog(
|
|
|
|
title: "Untrusted X509Certificate",
|
2022-11-08 17:41:12 +00:00
|
|
|
message: "SHA1:\n$sha1",
|
2022-11-08 16:18:48 +00:00
|
|
|
leftButton: SecondaryButton(
|
|
|
|
label: "Cancel",
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop(false);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
rightButton: PrimaryButton(
|
|
|
|
label: "Trust",
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop(true);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
return result ?? false;
|
|
|
|
}
|