我想让我的nginx代理只在客户端还没有通过身份验证的情况下执行一个子请求。条件部分是我卡住的地方。如何创建一个配置,使客户端在每个会话中只经过一次身份验证?
我能够成功地向Apache执行auth_request,并拉回我想传递给后端的头,但这会发生在每个请求上,而且代价很高。
在这里的示例中,我的目标是只在“Authorization”头丢失或为空,或者包含令牌的cookie时执行auth_request
# DEFAULT BACKEND
location / {
proxy_pass_request_body off;
if ($http_authorization ~* '')
{
rewrite ^(.*)$ /__login;
}
if ($user !~* "([aa-zZ]+)@example.com")
{
}
if ($http_cookie !~* "(auth_cookie=([aa-zZ]+)@example.com)")
{
add_header Set-Cookie "auth_cookie=$user;domain=.example.com;Max-Age=3000";
}
proxy_pass_header x-webauth-user;
proxy_pass_header Set-Cookie;
proxy_pass http://example.com:6762/;
}
字符串
location /__login { internal;
auth_request /auth;
auth_request_set $user $upstream_http_x_webauth_user;
set $xuser $user;
add_header Auth-User $user;
proxy_set_header User-Name $user;
proxy_set_header Authorization $http_authorization;
#proxy_pass_header x-webauth-user;
#proxy_pass_header Set-Cookie;
proxy_pass http://example:6762/;
access_log /etc/nginx/login_debug.log;
}
location = /auth{
internal;
proxy_pass http://example.com:81/;
proxy_pass_request_body off;
proxy_set_header Content-Length "";
proxy_set_header Host $host;
proxy_set_header X-Forwarded-for $proxy_add_x_forwarded_for;
#proxy_pass_header Set-Cookie;
#proxy_pass_header x-webauth-user;
}
型
Auth-User头在第一个请求之后的所有请求中都会丢失,并且cookie似乎从未设置过,除此之外,页面实际上似乎不会在浏览器中呈现。很明显我做错了什么,请帮我弄清楚这一点。
3条答案
按热度按时间l7wslrjt1#
在Nginx wiki warns中,
if
在location
内部可能会给予意外的结果,但rewrite ... last;
是安全的。下面是一个示例:字符串
具体有两种情况:Cookie:用户名是否存在。如果存在,则执行第一个
proxy_pass
。否则,使用/__login
。请注意,$uri
被传递,因此它可以被发送到backend-app。对于更高级的条件,可以使用map而不是
if
。不过,请注意,如果不对每个请求进行身份验证,则会有接受带有“伪造”cookie/头的请求的风险。
agyaoht72#
请查看NJS(https://nginx.org/en/docs/njs/)模块。它真的很简单,肯定可以做你想做的事情。以下是示例解决方案:
文件:/etc/nginx/conf.d/default.conf:
字符串
以及nginx.conf文件的示例来展示如何启用NJS模块:
型
最后是auth.js文件中的main函数:
型
请把它当作一个例子。好吧,也许它看起来很复杂,但它确实很强大,你可以在万维网上找到更多的例子。
t5zmwmid3#
也许太迟了但我找到了另一个解决办法。使用proxy_cache缓存auth_request。
字符串