✨ A well-done gateway
This commit is contained in:
@ -1,10 +1,12 @@
|
|||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using Yarp.ReverseProxy.Configuration;
|
||||||
|
|
||||||
namespace DysonNetwork.Gateway.Controllers;
|
namespace DysonNetwork.Gateway.Controllers;
|
||||||
|
|
||||||
[ApiController]
|
[ApiController]
|
||||||
[Route("/.well-known")]
|
[Route("/.well-known")]
|
||||||
public class WellKnownController(IConfiguration configuration) : ControllerBase
|
public class WellKnownController(IConfiguration configuration, IProxyConfigProvider proxyConfigProvider)
|
||||||
|
: ControllerBase
|
||||||
{
|
{
|
||||||
[HttpGet("domains")]
|
[HttpGet("domains")]
|
||||||
public IActionResult GetDomainMappings()
|
public IActionResult GetDomainMappings()
|
||||||
@ -13,4 +15,33 @@ public class WellKnownController(IConfiguration configuration) : ControllerBase
|
|||||||
.ToDictionary(x => x.Key, x => x.Value);
|
.ToDictionary(x => x.Key, x => x.Value);
|
||||||
return Ok(domainMappings);
|
return Ok(domainMappings);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[HttpGet("routes")]
|
||||||
|
public IActionResult GetProxyRules()
|
||||||
|
{
|
||||||
|
var config = proxyConfigProvider.GetConfig();
|
||||||
|
var rules = config.Routes.Select(r => new
|
||||||
|
{
|
||||||
|
r.RouteId,
|
||||||
|
r.ClusterId,
|
||||||
|
Match = new
|
||||||
|
{
|
||||||
|
r.Match.Path,
|
||||||
|
Hosts = r.Match.Hosts != null ? string.Join(", ", r.Match.Hosts) : null
|
||||||
|
},
|
||||||
|
Transforms = r.Transforms?.Select(t => t.Select(kv => $"{kv.Key}: {kv.Value}").ToList())
|
||||||
|
}).ToList();
|
||||||
|
|
||||||
|
var clusters = config.Clusters.Select(c => new
|
||||||
|
{
|
||||||
|
c.ClusterId,
|
||||||
|
Destinations = c.Destinations?.Select(d => new
|
||||||
|
{
|
||||||
|
d.Key,
|
||||||
|
d.Value.Address
|
||||||
|
}).ToList()
|
||||||
|
}).ToList();
|
||||||
|
|
||||||
|
return Ok(new { Rules = rules, Clusters = clusters });
|
||||||
|
}
|
||||||
}
|
}
|
@ -8,8 +8,7 @@ builder.Services.AddControllers();
|
|||||||
|
|
||||||
var app = builder.Build();
|
var app = builder.Build();
|
||||||
|
|
||||||
// app.UseHttpsRedirection();
|
app.MapControllers();
|
||||||
|
|
||||||
app.MapReverseProxy();
|
app.MapReverseProxy();
|
||||||
|
|
||||||
app.Run();
|
app.Run();
|
||||||
|
@ -4,15 +4,15 @@ using Yarp.ReverseProxy.Configuration;
|
|||||||
|
|
||||||
namespace DysonNetwork.Gateway;
|
namespace DysonNetwork.Gateway;
|
||||||
|
|
||||||
public class EtcdProxyConfigProvider : IProxyConfigProvider, IDisposable
|
public class RegistryProxyConfigProvider : IProxyConfigProvider, IDisposable
|
||||||
{
|
{
|
||||||
private readonly IEtcdClient _etcdClient;
|
private readonly IEtcdClient _etcdClient;
|
||||||
private readonly IConfiguration _configuration;
|
private readonly IConfiguration _configuration;
|
||||||
private readonly ILogger<EtcdProxyConfigProvider> _logger;
|
private readonly ILogger<RegistryProxyConfigProvider> _logger;
|
||||||
private readonly CancellationTokenSource _watchCts = new();
|
private readonly CancellationTokenSource _watchCts = new();
|
||||||
private CancellationTokenSource _cts = new();
|
private CancellationTokenSource _cts = new();
|
||||||
|
|
||||||
public EtcdProxyConfigProvider(IEtcdClient etcdClient, IConfiguration configuration, ILogger<EtcdProxyConfigProvider> logger)
|
public RegistryProxyConfigProvider(IEtcdClient etcdClient, IConfiguration configuration, ILogger<RegistryProxyConfigProvider> logger)
|
||||||
{
|
{
|
||||||
_etcdClient = etcdClient;
|
_etcdClient = etcdClient;
|
||||||
_configuration = configuration;
|
_configuration = configuration;
|
||||||
@ -34,20 +34,78 @@ public class EtcdProxyConfigProvider : IProxyConfigProvider, IDisposable
|
|||||||
var response = _etcdClient.GetRange("/services/");
|
var response = _etcdClient.GetRange("/services/");
|
||||||
var kvs = response.Kvs;
|
var kvs = response.Kvs;
|
||||||
|
|
||||||
|
var serviceMap = kvs.ToDictionary(
|
||||||
|
kv => Encoding.UTF8.GetString(kv.Key.ToByteArray()).Replace("/services/", ""),
|
||||||
|
kv => Encoding.UTF8.GetString(kv.Value.ToByteArray())
|
||||||
|
);
|
||||||
|
|
||||||
var clusters = new List<ClusterConfig>();
|
var clusters = new List<ClusterConfig>();
|
||||||
var routes = new List<RouteConfig>();
|
var routes = new List<RouteConfig>();
|
||||||
|
|
||||||
var domainMappings = _configuration.GetSection("DomainMappings").GetChildren()
|
var domainMappings = _configuration.GetSection("DomainMappings").GetChildren()
|
||||||
.ToDictionary(x => x.Key, x => x.Value);
|
.ToDictionary(x => x.Key, x => x.Value);
|
||||||
|
|
||||||
|
var pathAliases = _configuration.GetSection("PathAliases").GetChildren()
|
||||||
|
.ToDictionary(x => x.Key, x => x.Value);
|
||||||
|
|
||||||
|
var directRoutes = _configuration.GetSection("DirectRoutes").Get<List<DirectRouteConfig>>() ?? new List<DirectRouteConfig>();
|
||||||
|
|
||||||
_logger.LogInformation("Indexing {ServiceCount} services from Etcd.", kvs.Count);
|
_logger.LogInformation("Indexing {ServiceCount} services from Etcd.", kvs.Count);
|
||||||
|
|
||||||
foreach (var kv in kvs)
|
var gatewayServiceName = _configuration["Service:Name"];
|
||||||
{
|
|
||||||
var serviceName = Encoding.UTF8.GetString(kv.Key.ToByteArray()).Replace("/services/", "");
|
|
||||||
var serviceUrl = Encoding.UTF8.GetString(kv.Value.ToByteArray());
|
|
||||||
|
|
||||||
_logger.LogInformation(" Service: {ServiceName}, URL: {ServiceUrl}", serviceName, serviceUrl);
|
// Add direct routes
|
||||||
|
foreach (var directRoute in directRoutes)
|
||||||
|
{
|
||||||
|
if (serviceMap.TryGetValue(directRoute.Service, out var serviceUrl))
|
||||||
|
{
|
||||||
|
var cluster = new ClusterConfig
|
||||||
|
{
|
||||||
|
ClusterId = directRoute.Service,
|
||||||
|
Destinations = new Dictionary<string, DestinationConfig>
|
||||||
|
{
|
||||||
|
{ "destination1", new DestinationConfig { Address = serviceUrl } }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
clusters.Add(cluster);
|
||||||
|
|
||||||
|
var route = new RouteConfig
|
||||||
|
{
|
||||||
|
RouteId = $"direct-{directRoute.Service}-{directRoute.Path.Replace("/", "-")}",
|
||||||
|
ClusterId = directRoute.Service,
|
||||||
|
Match = new RouteMatch { Path = directRoute.Path }
|
||||||
|
};
|
||||||
|
routes.Add(route);
|
||||||
|
_logger.LogInformation(" Added Direct Route: {Path} -> {Service}", directRoute.Path, directRoute.Service);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning(" Direct route service {Service} not found in Etcd.", directRoute.Service);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var serviceName in serviceMap.Keys)
|
||||||
|
{
|
||||||
|
if (serviceName == gatewayServiceName)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Skipping gateway service: {ServiceName}", serviceName);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var serviceUrl = serviceMap[serviceName];
|
||||||
|
|
||||||
|
// Determine the path alias
|
||||||
|
string pathAlias;
|
||||||
|
if (pathAliases.TryGetValue(serviceName, out var alias))
|
||||||
|
{
|
||||||
|
pathAlias = alias;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pathAlias = serviceName.Split('.').Last().ToLowerInvariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation(" Service: {ServiceName}, URL: {ServiceUrl}, Path Alias: {PathAlias}", serviceName, serviceUrl, pathAlias);
|
||||||
|
|
||||||
var cluster = new ClusterConfig
|
var cluster = new ClusterConfig
|
||||||
{
|
{
|
||||||
@ -81,7 +139,11 @@ public class EtcdProxyConfigProvider : IProxyConfigProvider, IDisposable
|
|||||||
{
|
{
|
||||||
RouteId = $"{serviceName}-path",
|
RouteId = $"{serviceName}-path",
|
||||||
ClusterId = serviceName,
|
ClusterId = serviceName,
|
||||||
Match = new RouteMatch { Path = $"/{serviceName}/{{**catch-all}}" }
|
Match = new RouteMatch { Path = $"/{pathAlias}/{{**catch-all}}" },
|
||||||
|
Transforms = new List<Dictionary<string, string>>
|
||||||
|
{
|
||||||
|
new Dictionary<string, string> { { "PathRemovePrefix", $"/{pathAlias}" } }
|
||||||
|
}
|
||||||
};
|
};
|
||||||
routes.Add(pathRoute);
|
routes.Add(pathRoute);
|
||||||
_logger.LogInformation(" Added Path-based Route: {Path}", pathRoute.Match.Path);
|
_logger.LogInformation(" Added Path-based Route: {Path}", pathRoute.Match.Path);
|
||||||
@ -98,6 +160,12 @@ public class EtcdProxyConfigProvider : IProxyConfigProvider, IDisposable
|
|||||||
public Microsoft.Extensions.Primitives.IChangeToken ChangeToken { get; } = new Microsoft.Extensions.Primitives.CancellationChangeToken(CancellationToken.None);
|
public Microsoft.Extensions.Primitives.IChangeToken ChangeToken { get; } = new Microsoft.Extensions.Primitives.CancellationChangeToken(CancellationToken.None);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private class DirectRouteConfig
|
||||||
|
{
|
||||||
|
public string Path { get; set; }
|
||||||
|
public string Service { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
public void Dispose()
|
public void Dispose()
|
||||||
{
|
{
|
||||||
_cts.Cancel();
|
_cts.Cancel();
|
@ -9,7 +9,7 @@ public static class ServiceCollectionExtensions
|
|||||||
{
|
{
|
||||||
services.AddReverseProxy();
|
services.AddReverseProxy();
|
||||||
services.AddRegistryService(configuration);
|
services.AddRegistryService(configuration);
|
||||||
services.AddSingleton<IProxyConfigProvider, EtcdProxyConfigProvider>();
|
services.AddSingleton<IProxyConfigProvider, RegistryProxyConfigProvider>();
|
||||||
|
|
||||||
return services;
|
return services;
|
||||||
}
|
}
|
||||||
|
@ -21,5 +21,23 @@
|
|||||||
"DysonNetwork.Drive": "drive.solsynth.dev",
|
"DysonNetwork.Drive": "drive.solsynth.dev",
|
||||||
"DysonNetwork.Pusher": "push.solsynth.dev",
|
"DysonNetwork.Pusher": "push.solsynth.dev",
|
||||||
"DysonNetwork.Sphere": "sphere.solsynth.dev"
|
"DysonNetwork.Sphere": "sphere.solsynth.dev"
|
||||||
}
|
},
|
||||||
|
"PathAliases": {
|
||||||
|
"DysonNetwork.Pass": "id",
|
||||||
|
"DysonNetwork.Drive": "drive"
|
||||||
|
},
|
||||||
|
"DirectRoutes": [
|
||||||
|
{
|
||||||
|
"Path": "/ws",
|
||||||
|
"Service": "DysonNetwork.Pusher"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Path": "/.well-known/openid-configuration",
|
||||||
|
"Service": "DysonNetwork.Pass"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Path": "/.well-known/jwks",
|
||||||
|
"Service": "DysonNetwork.Pass"
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user