using FastGithub.Configuration; using FastGithub.HttpServer.Certs; using FastGithub.HttpServer.TcpMiddlewares; using FastGithub.HttpServer.TlsMiddlewares; using Microsoft.AspNetCore.Connections; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Server.Kestrel.Core; using Microsoft.AspNetCore.Server.Kestrel.Https; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Options; using System; using System.Net.Security; using System.Security.Cryptography.X509Certificates; using System.Threading.Tasks; namespace FastGithub { /// /// Kestrel扩展 /// public static class KestrelServerExtensions { /// /// 无限制 /// /// public static void NoLimit(this KestrelServerOptions kestrel) { kestrel.Limits.MaxRequestBodySize = null; kestrel.Limits.MinResponseDataRate = null; kestrel.Limits.MinRequestBodyDataRate = null; } /// /// 监听http代理 /// /// public static void ListenHttpProxy(this KestrelServerOptions kestrel) { var options = kestrel.ApplicationServices.GetRequiredService>().Value; var httpProxyPort = options.HttpProxyPort; if (GlobalListener.CanListenTcp(httpProxyPort) == false) { throw new FastGithubException($"tcp端口{httpProxyPort}已经被其它进程占用,请在配置文件更换{nameof(FastGithubOptions.HttpProxyPort)}为其它端口"); } kestrel.ListenLocalhost(httpProxyPort, listen => { var proxyMiddleware = kestrel.ApplicationServices.GetRequiredService(); var tunnelMiddleware = kestrel.ApplicationServices.GetRequiredService(); listen.Use(next => context => proxyMiddleware.InvokeAsync(next, context)); listen.UseTls(); listen.Use(next => context => tunnelMiddleware.InvokeAsync(next, context)); }); kestrel.GetLogger().LogInformation($"已监听http://localhost:{httpProxyPort},http代理服务启动完成"); } /// /// 监听ssh协议代理 /// /// public static void ListenSshReverseProxy(this KestrelServerOptions kestrel) { var sshPort = GlobalListener.SshPort; kestrel.ListenLocalhost(sshPort, listen => { listen.UseFlowAnalyze(); listen.UseConnectionHandler(); }); kestrel.GetLogger().LogInformation($"已监听ssh://localhost:{sshPort},github的ssh反向代理服务启动完成"); } /// /// 监听git协议代理代理 /// /// public static void ListenGitReverseProxy(this KestrelServerOptions kestrel) { var gitPort = GlobalListener.GitPort; kestrel.ListenLocalhost(gitPort, listen => { listen.UseFlowAnalyze(); listen.UseConnectionHandler(); }); kestrel.GetLogger().LogInformation($"已监听git://localhost:{gitPort},github的git反向代理服务启动完成"); } /// /// 监听http反向代理 /// /// public static void ListenHttpReverseProxy(this KestrelServerOptions kestrel) { var httpPort = GlobalListener.HttpPort; kestrel.ListenLocalhost(httpPort); if (OperatingSystem.IsWindows()) { kestrel.GetLogger().LogInformation($"已监听http://localhost:{httpPort},http反向代理服务启动完成"); } } /// /// 监听https反向代理 /// /// /// public static void ListenHttpsReverseProxy(this KestrelServerOptions kestrel) { var httpsPort = GlobalListener.HttpsPort; kestrel.ListenLocalhost(httpsPort, listen => { if (OperatingSystem.IsWindows()) { listen.UseFlowAnalyze(); } listen.UseTls(); }); if (OperatingSystem.IsWindows()) { var logger = kestrel.GetLogger(); logger.LogInformation($"已监听https://localhost:{httpsPort},https反向代理服务启动完成"); } } /// /// 获取日志 /// /// /// private static ILogger GetLogger(this KestrelServerOptions kestrel) { var loggerFactory = kestrel.ApplicationServices.GetRequiredService(); return loggerFactory.CreateLogger($"{nameof(FastGithub)}.{nameof(HttpServer)}"); } /// /// 使用Tls中间件 /// /// /// https配置 /// public static ListenOptions UseTls(this ListenOptions listen) { var certService = listen.ApplicationServices.GetRequiredService(); certService.CreateCaCertIfNotExists(); certService.InstallAndTrustCaCert(); return listen.UseTls(domain => certService.GetOrCreateServerCert(domain)); } /// /// 使用Tls中间件 /// /// /// https配置 /// private static ListenOptions UseTls(this ListenOptions listen, Func certFactory) { var invadeMiddleware = listen.ApplicationServices.GetRequiredService(); var restoreMiddleware = listen.ApplicationServices.GetRequiredService(); listen.Use(next => context => invadeMiddleware.InvokeAsync(next, context)); listen.UseHttps(new TlsHandshakeCallbackOptions { OnConnection = context => { var options = new SslServerAuthenticationOptions { ServerCertificate = certFactory(context.ClientHelloInfo.ServerName) }; return ValueTask.FromResult(options); }, }); listen.Use(next => context => restoreMiddleware.InvokeAsync(next, context)); return listen; } } }