c# - 使用反向代理在服务中强制执行 SSL

标签 c# ssl asp.net-web-api reverse-proxy web-farm-framework

基于文章Working with SSL in Web API我实现了一个授权过滤器,要求 SSL 用于 Web API (2.1) Controller 的方法:

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true,
                AllowMultiple = false)]
public sealed class RequireHttpsAttribute : AuthorizationFilterAttribute
{
    public override void OnAuthorization(HttpActionContext actionContext)
    {
        if (actionContext.Request.RequestUri.Scheme != Uri.UriSchemeHttps)
        {
            actionContext.Response = new HttpResponseMessage(HttpStatusCode.Forbidden)
                                         {
                                             ReasonPhrase = "HTTPS Required"
                                         };
        }
        else
        {
            base.OnAuthorization(actionContext);
        }
    }
}

这在某些网络服务器上工作正常。如果 Web Farm Framework (WFF) 用作反向代理,它可能会失败(通过阻止有效的 HTTPS 请求)。

WFF 添加 header X-Forwarded-Proto,这是反向代理的事实标准。

如何修改此代码以使用或不使用标准代理?

最佳答案

这是我想出的:

/// <summary>
/// Action filter to require SSL for a protected resource.
/// </summary>
/// <remarks>
/// From http://www.asp.net/web-api/overview/security/working-with-ssl-in-web-api
/// but modified to support reverse proxies such as Web Farm Framework.
/// </remarks>
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = false)]
public sealed class RequireHttpsAttribute : AuthorizationFilterAttribute
{
    [SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope", Justification = "Not possible.")]
    public override void OnAuthorization(HttpActionContext actionContext)
    {
        if (IsSecure(actionContext.Request))
        {
            base.OnAuthorization(actionContext);
        }
        else
        {
            actionContext.Response =
                new HttpResponseMessage(HttpStatusCode.Forbidden)
                    {
                        ReasonPhrase = "HTTPS Required"
                    };
        }
    }

    private static bool IsSecure(HttpRequestMessage request)
    {
        if (request.RequestUri.Scheme == Uri.UriSchemeHttps)
        {
            return true;
        }

        IEnumerable<string> headerValues;

        if (request.Headers.TryGetValues("X-Forwarded-Proto", out headerValues))
        {
            string protocol = headerValues.FirstOrDefault();

            return string.Equals(protocol, "https", StringComparison.OrdinalIgnoreCase);
        }

        return false;
    }
}

关于c# - 使用反向代理在服务中强制执行 SSL,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24659804/

相关文章:

c# - 在组合框中按对象值查找项目

ssl - JIRA Rest 客户端 SSL 认证

ssl - RabbitMQ SSL 指南...生产需要进行哪些更改?

c# - 为外部身份验证创建部分登录 cookie

c# - AUTOINCREMENT 不适用于 OleDbCommand

c# - 安装插件时 NopCommerce 4.10 错误

c# - 有没有办法在 DBSet 的查询中包含本地缓存的项目?

php - openssl_verify 和 "error:0906D06C:PEM routines:PEM_read_bio:no start line"

asp.net-mvc - 为什么这两个 API 方法会产生冲突

asp.net-mvc - aspnet-api-versioning 有什么好处?