nginx proxy_pass导致404 Not Found页面 [英] nginx proxy_pass leads to 404 Not Found page
问题描述
我有一个在已安装nginx的docker ubuntu映像中运行的角度应用程序.我想将此映像部署到Kubernetes,并使用Nginx代理将所有对/api的调用重定向到Kubernetes中的后端服务.
I have a angular application running in a docker ubuntu image that has nginx installed. I want to deploy this image to Kubernetes and use a nginx proxy to redirect all calls to /api to my backend service in Kubernetes.
我的静态Web资源位于/var/www/html中,并将以下配置添加到/etc/nginx/conf.d:
My static web resources lie in /var/www/html and I add the following config to /etc/nginx/conf.d:
upstream backend-service {
server backend-service:8080;
}
server {
listen 80;
location / {
try_files $uri $uri/ /index.html;
}
location ^~ /api {
proxy_pass http://backend-service;
}
}
访问/或/#/dashboard上的前端服务会返回我的Angular页面的预期组件,但是对/api/v1/data的调用仅显示默认的Nginx 404 Not Found页面.
Accessing the frontend service on / or /#/dashboard returns the expected component of my Angular page, but a call to /api/v1/data only shows the default nginx 404 Not Found page.
我需要修改什么才能将后端呼叫重定向到后端?
What do I need to modify to have my backend calls redirected to my backend?
我在ubuntu 16.04上使用nginx 1.10.3 ,并且我的前端Dockerfile如下所示:
I use nginx 1.10.3 on ubuntu 16.04 and my frontend Dockerfile looks like this:
FROM ubuntu:16.04
# Install curl, nodejs and nginx
RUN apt-get update && \
apt-get install -y curl && \
curl -sL https://deb.nodesource.com/setup_8.x | bash - && \
apt-get install -y nodejs nginx && \
rm -rf /var/lib/apt/lists/*
# Create directory
RUN mkdir -p /usr/src/app
WORKDIR /usr/src/app
# Copy and build rest of the app
COPY . /usr/src/app
RUN npm install
RUN node_modules/@angular/cli/bin/ng build --prod
RUN cp -a dist/. /var/www/html
# Configure and start nginx
COPY frontend.conf /etc/nginx/conf.d
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]
有关后端服务的信息
后端服务监听获取/发布对/api/v1/data的请求,并且可以通过名为backend-service的服务在Kubernetes中进行访问.
The backend service listens to get and post requests on /api/v1/data and is reachable in Kubernetes via a Service named backend-service.
Edit2:Nginx access.log
https://gist.github.com/Steffen911/a56e3175bf12e511048d01359a475724
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET / HTTP/1.1" 200 380 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /styles.d41d8cd98f00b204e980.bundle.css HTTP/1.1" 200 0 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /inline.c9a1a6b995c65c13f605.bundle.js HTTP/1.1" 200 1447 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /polyfills.117078cae3e3d00fc376.bundle.js HTTP/1.1" 200 97253 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /main.3e9a37b4dd0f3bf2465f.bundle.js HTTP/1.1" 200 64481 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /vendor.146173c1a99cc2172a5f.bundle.js HTTP/1.1" 200 661261 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /api/v1/data/ HTTP/1.1" 404 209 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /assets/home.jpg HTTP/1.1" 200 2608 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /assets/busy.gif HTTP/1.1" 200 48552 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /assets/background_light.png HTTP/1.1" 200 170599 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /assets/google.svg HTTP/1.1" 200 2232 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /assets/email.svg HTTP/1.1" 200 1596 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:40 +0000] "GET /favicon.ico HTTP/1.1" 200 198 "http://192.168.99.100:30497/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
172.17.0.1 - - [13/Aug/2017:13:11:44 +0000] "GET /api/v1/data/ HTTP/1.1" 404 209 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36"
error.log文件为空.
The error.log file is empty.
Edit3:新的Nginx版本和其他SO线程
我还尝试了nginx 1.12.1,它显示了相同的行为.这个问题的答案也没有帮助: nginx proxy_pass 404错误,不明白为什么
I also tried nginx 1.12.1 and it shows the same behaviour. The answers to this question also haven't helped: nginx proxy_pass 404 error, don't understand why
Edit4:我上传了一个最小的示例,该示例在GitHub上重现了我的问题
https://github.com/Steffen911/nginx-sample
推荐答案
在对nginx进行故障排除时,看来您拥有的nginx配置文件实际上没有任何作用-您报告除索引以外的所有内容均出现404 Not Found
错误页面,从location /
中的try_files
到更具体的location ^~ /api
中的proxy_pass
和return 200 test
的所有指令,均无效.
From your troubleshooting of nginx, it appears that the nginx configuration file you have has effectively no effect — you report getting 404 Not Found
errors for everything other than the index page, with all the directives, from try_files
in location /
, to proxy_pass
and return 200 test
in a more specific location ^~ /api
, having no effect.
因此,问题似乎出在Dockerfile
中-似乎大多数其他NGINX + Docker教程都删除了默认配置(例如,使用RUN rm /etc/nginx/conf.d/default.conf
),而您的文件缺少任何此类删除.
As such, the problem appears to be in the Dockerfile
— it appears that most other NGINX + Docker tutorials remove default configurations (e.g., with RUN rm /etc/nginx/conf.d/default.conf
), whereas your file is missing any such removal.
实际上, Debian / /etc/nginx/sites-enabled
,默认情况下,必须包含 default
文件带有冒昧的listen 80 default_server
,实际上优先于其他任何 listen
没有更具体的server_name
的同一端口.
In fact, Debian/Ubuntu appear to have the non-standard directories of questionable utility called /etc/nginx/sites-available
and /etc/nginx/sites-enabled
, which, by default, must contain a default
file with a presumptuous listen 80 default_server
, effectively taking precedence over any other listen
of the same port in the absence of a more specific server_name
.
因此,有多种独立的解决方案:
As such, there are multiple independent solutions:
-
不要使用像Debian/Ubuntu提供的那些基本损坏的软件包. 我的配置不起作用,只是注意到甚至
emacs
之类的备份文件,如test.conf~
通过Debian的默认include /etc/nginx/sites-enabled/*;
包含在Debian中. 启用站点的网站是邪恶的.
Do not use fundamentally broken packages like those offered by Debian/Ubuntu. I once spent a good amount of time pulling my hair trying to figure out why my configs don't work, only to notice that even the backup files from
emacs
liketest.conf~
get included in Debian through Debian's defaultinclude /etc/nginx/sites-enabled/*;
. Sites-Enabled Is Evil.
请注意, NGINX为大多数发行版提供了官方二进制软件包,问题,因为它没有尝试在其/etc/nginx/conf.d/default.conf
中定义default_server
,而是使用server_name localhost;
进行了listen 80;
,在大多数情况下会自动摆脱困境.
Note that NGINX provides official binary packages for most distributions, which wouldn't have had this issue, as it doesn't try to define a default_server
in its /etc/nginx/conf.d/default.conf
, instead doing a listen 80;
with server_name localhost;
, getting out of your way automatically by itself in most circumstances.
例如,将FROM ubuntu:16.04
替换为Dockerfile
,以使用NGINX官方映像.
E.g., replace FROM ubuntu:16.04
with FROM nginx
in your Dockerfile
to be using NGINX official image.
- 如果仍在使用Debian/Ubuntu中的nginx ,请确保在您的
Dockerfile
中RUN rm /etc/nginx/sites-enabled/default
>删除default_server
listen
.
- If still using nginx from Debian/Ubuntu, make sure to
RUN rm /etc/nginx/sites-enabled/default
in yourDockerfile
to remove thedefault_server
listen
.
-
使用
server_name
指令定义基于主机名的服务器 ,可能与listen
指令以及default_server
参数.
请注意,重复的server_name
规范会导致配置警告(严重性为[warn]
),但是重复的default_server
是配置错误(严重性为[emerg]
),这可能有助于更早地解决问题.
Note that a duplicate server_name
specification results in a configuration warning (with the [warn]
severity), but a duplicate default_server
is a configuration error ([emerg]
severity), which might help troubleshoot the issue earlier.
这篇关于nginx proxy_pass导致404 Not Found页面的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!