Мне нужно сделать промежуточное ПО обратного прокси-сервера в ASP.NET Core. У меня есть что-то вроде этого. Но это не работает с SignalR.
Обратный прокси имеет адрес 192.168.187.72:5000; СигналR 192.168.187.62:5000. Клиент подключается к прокси, прокси изменяет URL-адрес на URL-адрес концентратора SignalR, но в клиенте я всегда получаю эту ошибку:
Сервер отключился до того, как можно было начать рукопожатие
Код:
public ReverseProxyMiddleware(RequestDelegate nextMiddleware,ILogger<ReverseProxyMiddleware> logger)
{
_nextMiddleware = nextMiddleware;
_logger = logger;
}
public async Task Invoke(HttpContext context)
{
var targetUri = BuildTargetUri(context.Request);
if (targetUri != null)
{
var targetRequestMessage = CreateTargetMessage(context, targetUri);
using (var responseMessage = await _httpClient.SendAsync(targetRequestMessage, HttpCompletionOption.ResponseHeadersRead, context.RequestAborted))
{
context.Response.StatusCode = (int)responseMessage.StatusCode;
CopyFromTargetResponseHeaders(context, responseMessage);
await responseMessage.Content.CopyToAsync(context.Response.Body);
}
return;
}
await _nextMiddleware(context);
}
private HttpRequestMessage CreateTargetMessage(HttpContext context, Uri targetUri)
{
var requestMessage = new HttpRequestMessage();
CopyFromOriginalRequestContentAndHeaders(context, requestMessage);
requestMessage.RequestUri = targetUri;
requestMessage.Headers.Host = targetUri.Host;
requestMessage.Method = GetMethod(context.Request.Method);
return requestMessage;
}
private void CopyFromOriginalRequestContentAndHeaders(HttpContext context, HttpRequestMessage requestMessage)
{
var requestMethod = context.Request.Method;
var streamContent = new StreamContent(context.Request.Body);
requestMessage.Content = streamContent;
foreach (var header in context.Request.Headers)
{
requestMessage.Content?.Headers.TryAddWithoutValidation(header.Key, header.Value.ToArray());
}
}
private void CopyFromTargetResponseHeaders(HttpContext context, HttpResponseMessage responseMessage)
{
foreach (var header in responseMessage.Headers)
{
context.Response.Headers[header.Key] = header.Value.ToArray();
}
foreach (var header in responseMessage.Content.Headers)
{
context.Response.Headers[header.Key] = header.Value.ToArray();
}
context.Response.Headers.Remove("transfer-encoding");
}
private static HttpMethod GetMethod(string method)
{
if (HttpMethods.IsDelete(method)) return HttpMethod.Delete;
if (HttpMethods.IsGet(method)) return HttpMethod.Get;
if (HttpMethods.IsHead(method)) return HttpMethod.Head;
if (HttpMethods.IsOptions(method)) return HttpMethod.Options;
if (HttpMethods.IsPost(method)) return HttpMethod.Post;
if (HttpMethods.IsPut(method)) return HttpMethod.Put;
if (HttpMethods.IsTrace(method)) return HttpMethod.Trace;
return new HttpMethod(method);
}
private Uri BuildTargetUri(HttpRequest request)
{
Uri targetUri = null;
StringValues clientId = string.Empty;
targetUri = new Uri($"http://192.168.187.62:5000{request.Path}{request.QueryString}");
_logger.LogInformation($"Request URL {request.Host + request.Path + request.QueryString} Target URL {targetUri.ToString()}");
return targetUri;
}
Вы уверены, что хотите создать такую функциональность вручную?
Есть много отличных библиотек для обратного проксирования, у Microsoft есть новый проект под названием YARP (https://microsoft.github.io/reverse-proxy/), который представляет собой обратный прокси-сервер с открытым исходным кодом, который, как говорят, обладает высокой производительностью и широкими возможностями настройки. .
Если хотите, я могу показать вам пример использования yarp в приложении ASP.NET Core.
У меня есть приложения, использующие YARP и работающие с signalR, они его поддерживают. Не могли бы вы показать свою конфигурацию маршрутизации и конвейер промежуточного программного обеспечения?
Адреса назначения кластера должны заканчиваться символом «/». Также в этой конфигурации, если ваш клиент подключается к 192.168.187.72:5000/signalrClincHub, прокси-сервер перенаправляется на 192.168.187.62:5000/signalrClincHub/signalrClincHub (дублирующая часть маршрута), это преднамеренно?
Фильтр заголовков выглядит нормально, у вас есть другие маршруты? Возможно, другие запросы передаются, потому что у вас есть другие маршруты, определенные после этого.
Будет здорово, я попробовал это с signalR, но что-то пошло не так.