mirror of
https://github.com/wanghongenpin/proxypin.git
synced 2026-03-29 06:59:46 +08:00
131 lines
5.2 KiB
Dart
131 lines
5.2 KiB
Dart
/*
|
|
* Copyright 2023 Hongen Wang All rights reserved.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* https://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:proxypin/network/bin/listener.dart';
|
|
import 'package:proxypin/network/channel/channel.dart';
|
|
import 'package:proxypin/network/channel/channel_context.dart';
|
|
import 'package:proxypin/network/components/manager/request_rewrite_manager.dart';
|
|
import 'package:proxypin/network/components/manager/script_manager.dart';
|
|
import 'package:proxypin/network/channel/host_port.dart';
|
|
import 'package:proxypin/network/http/codec.dart';
|
|
import 'package:proxypin/network/http/http.dart';
|
|
import 'package:proxypin/network/http/http_headers.dart';
|
|
import 'package:proxypin/network/util/crts.dart';
|
|
import 'package:proxypin/network/util/localizations.dart';
|
|
|
|
import '../components/host_filter.dart';
|
|
|
|
class ProxyHelper {
|
|
//请求本服务
|
|
static localRequest(ChannelContext channelContext, HttpRequest msg, Channel channel) async {
|
|
//获取配置
|
|
if (msg.path == '/config') {
|
|
final requestRewrites = await RequestRewriteManager.instance;
|
|
var response = HttpResponse(HttpStatus.ok, protocolVersion: msg.protocolVersion);
|
|
var body = {
|
|
"requestRewrites": await requestRewrites.toFullJson(),
|
|
'whitelist': HostFilter.whitelist.toJson(),
|
|
'blacklist': HostFilter.blacklist.toJson(),
|
|
'scripts': await ScriptManager.instance.then((script) {
|
|
var list = script.list.map((e) async {
|
|
return {'name': e.name, 'enabled': e.enabled, 'url': e.url, 'script': await script.getScript(e)};
|
|
});
|
|
return Future.wait(list);
|
|
}),
|
|
};
|
|
response.body = utf8.encode(json.encode(body));
|
|
channel.writeAndClose(channelContext, response);
|
|
return;
|
|
}
|
|
|
|
var response = HttpResponse(HttpStatus.ok, protocolVersion: msg.protocolVersion);
|
|
response.body = utf8.encode('pong');
|
|
response.headers.set("os", Platform.operatingSystem);
|
|
response.headers.set("hostname", Platform.isAndroid ? Platform.operatingSystem : Platform.localHostname);
|
|
channel.writeAndClose(channelContext, response);
|
|
}
|
|
|
|
/// 下载证书
|
|
static void crtDownload(ChannelContext channelContext, Channel channel, HttpRequest request) async {
|
|
const String fileMimeType = 'application/x-x509-ca-cert';
|
|
var response = HttpResponse(HttpStatus.ok);
|
|
response.headers.set(HttpHeaders.CONTENT_TYPE, fileMimeType);
|
|
response.headers.set("Content-Disposition", 'inline;filename=ProxyPinCA.crt');
|
|
response.headers.set("Connection", 'close');
|
|
|
|
var caFile = await CertificateManager.certificateFile();
|
|
var caBytes = await caFile.readAsBytes();
|
|
response.headers.set("Content-Length", caBytes.lengthInBytes.toString());
|
|
|
|
if (request.method == HttpMethod.head) {
|
|
channel.writeAndClose(channelContext, response);
|
|
return;
|
|
}
|
|
response.body = caBytes;
|
|
channel.writeAndClose(channelContext, response);
|
|
}
|
|
|
|
///异常处理
|
|
static exceptionHandler(
|
|
ChannelContext channelContext, Channel channel, EventListener? listener, HttpRequest? request, error) async {
|
|
HostAndPort? hostAndPort = channelContext.host;
|
|
hostAndPort ??= HostAndPort.host(
|
|
scheme: HostAndPort.httpScheme, channel.remoteSocketAddress.host, channel.remoteSocketAddress.port);
|
|
String message = error.toString();
|
|
HttpStatus status = HttpStatus(-1, message);
|
|
if (error is HandshakeException) {
|
|
status = HttpStatus(
|
|
-2,
|
|
Localizations.isZH
|
|
? 'SSL handshake failed, 请检查证书安装是否正确'
|
|
: 'SSL handshake failed, please check the certificate');
|
|
} else if (error is ParserException) {
|
|
status = HttpStatus(-3, error.message);
|
|
} else if (error is SocketException) {
|
|
status = HttpStatus(-4, error.message);
|
|
} else if (error is SignalException) {
|
|
status.reason(Localizations.isZH ? '执行脚本异常' : 'Execute script exception');
|
|
}
|
|
|
|
request ??= HttpRequest(HttpMethod.connect, hostAndPort.domain)
|
|
..body = message.codeUnits
|
|
..headers.contentLength = message.codeUnits.length
|
|
..hostAndPort = hostAndPort;
|
|
request.processInfo ??= channelContext.processInfo;
|
|
|
|
if (request.method == HttpMethod.connect && !request.uri.startsWith("http")) {
|
|
request.uri = hostAndPort.domain;
|
|
}
|
|
|
|
if (request.response == null || request.method == HttpMethod.connect) {
|
|
request.response = HttpResponse(status)
|
|
..headers.contentType = 'text/plain'
|
|
..headers.contentLength = message.codeUnits.length
|
|
..body = message.codeUnits;
|
|
}
|
|
|
|
request.response?.request = request;
|
|
|
|
channelContext.host = hostAndPort;
|
|
|
|
listener?.onRequest(channel, request);
|
|
listener?.onResponse(channelContext, request.response!);
|
|
}
|
|
}
|