nginx+apache+单页面刷新404问题解决方案

0. 背景

一个实际的应用环境中需要同时部署多个站点,所以选择同时使用nginx和apache,使用apache web容器同时部署2个站点,使用nginx进行转发。但由于网站是使用react或vue开发的,一个使用了hash router,一个使用的是history router,中途遇到一个问题是当使用的是history路由时,如果当前而在不是index页面,浏览器刷新后会出现404错误

1. 应用运行环境

操作系统:CentOS Linux release 7.6.1810 (Core)

Web容器:Apache/2.4.6 (CentOS)

nginx:nginx/1.16.0

2. Apache多站点部署快捷方法

方法一:同域名下多个站点

本质上仍是一个网站,通过二级目录的方式划分成2个不同的站点

方法二:不同域名多个站点

本质是是通过Apache的VirtualHost实现,具体配置详细说明如下:

由于在我的应用环境中向外是通过nginx做转发,所以这里说明通过VirtualHost基本端口号实现多站点的配置过程

  • 在/etc/httpd/conf/httpd.conf中增加以下内容

    Listen 8080
    Listen 8090
    #以上将在VirtualHost配置中使用
    LoadModule rewrite_module modules/mod_rewrite.so
    #上一行配置的作用在后续解决刷新404问题中的一步
    
  • 在/etc/httpd/conf.d/下新建host.conf文件,文件在配置以下内容

    <VirtualHost localhost:8080>
            ServerAdmin root@localhost
            ServerName localhost:8091
            DocumentRoot "网站A根目录"
            DirectoryIndex index.html
            <Directory "网站A根目录">
                    Options Indexes Includes FollowSymlinks
                    AllowOverride All
                    Require all granted
            </Directory>
            ErrorLog /var/log/httpd/err_8080.log
            LogLevel warn
            CustomLog /var/log/httpd/err_8080_access.log combined
    </VirtualHost>
    <VirtualHost localhost:8090>
            ServerAdmin root@localhost
            ServerName localhost:8090
            DocumentRoot "网站B根目录"
            DirectoryIndex index.html
            <Directory "网站B根目录">
                    Options Indexes Includes FollowSymlinks
                    AllowOverride All
                    Require all granted
            </Directory>
            ErrorLog /var/log/httpd/err_8090.log
            LogLevel warn
            CustomLog /var/log/httpd/err_8090_access.log combined
    </VirtualHost>
    

3. nginx转发配置

这里假设A网站分配域名为a.me.com,B网站分配域名是b.me.com

详细配置如下:

#user  nobody;
worker_processes  1;

#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;

#pid        logs/nginx.pid;


events {
    worker_connections  1024;
}


http {
    include       mime.types;
    default_type  application/octet-stream;
    map $http_upgrade $connection_upgrade {
        default upgrade;
        ''      close;
    }

    #log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
    #                  '$status $body_bytes_sent "$http_referer" '
    #                  '"$http_user_agent" "$http_x_forwarded_for"';

    #access_log  logs/access.log  main;

    sendfile        on;
    underscores_in_headers on;
    #tcp_nopush     on;

    #keepalive_timeout  0;
    keepalive_timeout  65;

    #gzip  on;
   #将http请求转发到https
   server {
        listen 80;
        server_name a.me.com;
        return 301 https://$server_name$request_uri;
   }

   server {
        listen 80;
        server_name b.me.com;
        return 301 https://$server_name$request_uri;
   }
    # HTTPS server
    #
    server {
        listen 443 ssl;
        server_name a.me.com;
        root html;
        index index.html index.htm;
        ssl_certificate   对应域名ssl证书路径;
        ssl_certificate_key  对应域名ssl证书key路径;
        ssl_session_timeout 5m;
        ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4;
        ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
        ssl_prefer_server_ciphers on;
        location ~* / {
            client_max_body_size  100M;
            proxy_pass http://localhost:8080;
            add_header Cache-Control 'max-age=0';
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection $connection_upgrade;
        }
        location / {
            root html;
            index index.html index.htm;
        }
    }


    server {
        listen 443 ssl;
        server_name b.me.com;
        root html;
        index index.html index.htm;
        ssl_certificate   对应域名ssl证书路径;
        ssl_certificate_key  对应域名ssl证书key路径;
        ssl_session_timeout 5m;
        ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4;
        ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
        ssl_prefer_server_ciphers on;
        location ^~ /h5/ {
            proxy_pass http://localhost:8090;
            add_header Cache-Control 'max-age=0';
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        }
        location ~* / {
            client_max_body_size  100M;
            proxy_pass http://localhost:8090;
            add_header Cache-Control 'max-age=0';
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        }
        location / {
            client_max_body_size  100M;
            root html;
            index index.html index.htm;
        }
    }
}

4.怎么解决问题

在上述配置下使用React或Vue基本history的单页面路由,当当前页面不是index页面时刷新时出现404问题

  • 定位产生问题原因

    由于同时使用了nginx和apache,如果不确定问题是出在谁身上只能一点点的去试,显然是没办法的办法。其实从本质上分析nginx是做转发工作,那只要确定问题是不是出在转发层面就可以了。方法很直接:

    由于服务并没有向外开放8080和8090端口,所以可以直接在服务上通过wget打开一个非index的具体页面,如果可以打开那么问题自然是在nginx上,否则是apache配置的问题。

    经排除发现问题是在apache端

  • 问题解决办法

    第一步、在apache配置文件中增加以下配置

    LoadModule rewrite_module modules/mod_rewrite.so
    

    第二步、在网站根目录下新建.htaccess,文件内容如下

    <IfModule mod_rewrite.c>
      RewriteEngine On
      RewriteBase /
      RewriteRule ^index\.html$ - [L]
      RewriteCond %{REQUEST_FILENAME} !-f
      RewriteCond %{REQUEST_FILENAME} !-d
      RewriteRule . /index.html [L]
    </IfModule>
    
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • 大多数 Nginx 新手都会频繁遇到这样一个困惑,那就是当同一个location配置块使用了多个 Nginx 模块...
    SkTj阅读 7,657评论 0 12
  • 一、概述 二、编译安装 三、httpd服务基础 四、httpd.conf配置文件 五、httpd服务访问控制 六、...
    紫_轩阅读 947评论 0 0
  • nginx重写规则 nginx rewrite 正则表达式匹配 大小写匹配 ~ 为区分大小写匹配 ~* 为不区分大...
    桖辶殇阅读 5,562评论 0 2
  • Nginx简介 解决基于进程模型产生的C10K问题,请求时即使无状态连接如web服务都无法达到并发响应量级一万的现...
    魏镇坪阅读 1,997评论 0 9
  • AMP安装 源码包相关 源码包安装步骤:第一步(configure):我们需要检测环境;配置程序的安装内容;生成m...
    宠辱不惊丶岁月静好阅读 796评论 0 1