Files
proxypin/lib/network/http_client.dart
2024-08-18 02:48:02 +08:00

167 lines
6.1 KiB
Dart

/*
* Copyright 2023 the original author or authors.
*
* 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:async';
import 'dart:io';
import 'package:network_proxy/network/host_port.dart';
import 'package:network_proxy/network/http/http.dart';
import 'package:network_proxy/network/network.dart';
import 'package:network_proxy/network/util/system_proxy.dart';
import 'package:proxy_manager/proxy_manager.dart';
import 'channel.dart';
import 'http/codec.dart';
class HttpClients {
/// 建立连接
static Future<Channel> startConnect(
HostAndPort hostAndPort, ChannelHandler handler, ChannelContext channelContext) async {
var client = Client()
..initChannel((channel) => channel.pipeline.handle(HttpResponseCodec(), HttpRequestCodec(), handler));
return client.connect(hostAndPort, channelContext);
}
///代理建立连接
static Future<Channel> proxyConnect(HostAndPort hostAndPort, ChannelHandler handler, ChannelContext channelContext,
{ProxyInfo? proxyInfo}) async {
var client = Client()
..initChannel((channel) => channel.pipeline.handle(HttpResponseCodec(), HttpRequestCodec(), handler));
if (proxyInfo == null) {
var proxyTypes = hostAndPort.isSsl() ? ProxyTypes.https : ProxyTypes.http;
proxyInfo = await SystemProxy.getSystemProxy(proxyTypes);
}
HostAndPort connectHost = proxyInfo == null ? hostAndPort : HostAndPort.host(proxyInfo.host, proxyInfo.port!);
var channel = await client.connect(connectHost, channelContext);
if (proxyInfo != null) {
await connectRequest(hostAndPort, channel);
}
if (hostAndPort.isSsl()) {
await channel.secureSocket(channelContext, host: hostAndPort.host);
}
return channel;
}
///发起代理连接请求
static Future<Channel> connectRequest(HostAndPort hostAndPort, Channel channel) async {
ChannelHandler handler = channel.pipeline.handler;
//代理 发送connect请求
var httpResponseHandler = HttpResponseHandler();
channel.pipeline.handler = httpResponseHandler;
HttpRequest proxyRequest = HttpRequest(HttpMethod.connect, '${hostAndPort.host}:${hostAndPort.port}');
proxyRequest.headers.set(HttpHeaders.hostHeader, '${hostAndPort.host}:${hostAndPort.port}');
await channel.write(proxyRequest);
var response = await httpResponseHandler.getResponse(const Duration(seconds: 5));
channel.pipeline.handler = handler;
if (!response.status.isSuccessful()) {
throw Exception("$hostAndPort Proxy failed to establish tunnel "
"(${response.status.code} ${response..status.reasonPhrase})");
}
return channel;
}
/// 建立连接
static Future<Channel> connect(Uri uri, ChannelHandler handler, ChannelContext channelContext) async {
Client client = Client()
..initChannel((channel) => channel.pipeline.handle(HttpResponseCodec(), HttpRequestCodec(), handler));
if (uri.scheme == "https" || uri.scheme == "wss") {
return client.secureConnect(HostAndPort.of(uri.toString()), channelContext);
}
return client.connect(HostAndPort.of(uri.toString()), channelContext);
}
/// 发送get请求
static Future<HttpResponse> get(String url, {Duration duration = const Duration(seconds: 3)}) async {
HttpRequest msg = HttpRequest(HttpMethod.get, url);
return request(HostAndPort.of(url), msg, duration: duration);
}
/// 发送请求
static Future<HttpResponse> request(HostAndPort hostAndPort, HttpRequest request,
{Duration duration = const Duration(seconds: 3)}) async {
var httpResponseHandler = HttpResponseHandler();
var client = Client()
..initChannel((channel) => channel.pipeline.handle(HttpResponseCodec(), HttpRequestCodec(), httpResponseHandler));
ChannelContext channelContext = ChannelContext();
Channel channel = await client.connect(hostAndPort, channelContext);
await channel.write(request);
return httpResponseHandler.getResponse(duration).whenComplete(() => channel.close());
}
/// 发送代理请求
static Future<HttpResponse> proxyRequest(HttpRequest request,
{ProxyInfo? proxyInfo, Duration timeout = const Duration(seconds: 10)}) async {
if (request.headers.host == null || request.headers.host?.trim().isEmpty == true) {
try {
var uri = Uri.parse(request.requestUrl);
request.headers.host = '${uri.host}${uri.hasPort ? ':${uri.port}' : ''}';
} catch (_) {}
}
request.protocolVersion = 'HTTP/1.1';
ChannelContext channelContext = ChannelContext();
var httpResponseHandler = HttpResponseHandler();
request.hostAndPort ??= HostAndPort.of(request.requestUrl);
Channel channel = await proxyConnect(proxyInfo: proxyInfo, request.hostAndPort!, httpResponseHandler, channelContext);
if (channel.isSsl && !request.uri.startsWith("/")) {
Uri? uri = request.requestUri;
request = request.copy(uri: '${uri!.path}${uri.hasQuery ? '?${uri.query}' : ''}');
}
await channel.write(request);
return httpResponseHandler.getResponse(timeout).whenComplete(() => channel.close());
}
}
class HttpResponseHandler extends ChannelHandler<HttpResponse> {
Completer<HttpResponse> _completer = Completer<HttpResponse>();
@override
void channelRead(ChannelContext channelContext, Channel channel, HttpResponse msg) {
// log.i("[${channel.id}] Response $msg");
_completer.complete(msg);
}
Future<HttpResponse> getResponse(Duration duration) {
return _completer.future.timeout(duration);
}
void resetResponse() {
_completer = Completer<HttpResponse>();
}
@override
void channelInactive(ChannelContext channelContext, Channel channel) {
// log.i("[${channel.id}] channelInactive");
}
}