Nginx将非www前缀域重写为www-prefixed域

问题描述 投票:39回答:6

我看到Nginx HttpRewriteModule documentation有一个例子,将www前缀域重写为非www前缀域:

if ($host ~* www\.(.*)) {
  set $host_without_www $1;
  rewrite ^(.*)$ http://$host_without_www$1 permanent; # $1 contains '/foo', not 'www.mydomain.com/foo'
}

如何反向 - 将非www前缀域重写为www前缀域?我想也许我可以做类似下面的事情,但Nginx不喜欢嵌套的if语句。

if ($host !~* ^www\.) {                       # check if host doesn't start with www.
    if ($host ~* ([a-z0-9]+\.[a-z0-9]+)) {    # check host is of the form xxx.xxx (i.e. no subdomain)
        set $host_with_www www.$1;
        rewrite ^(.*)$ http://$host_with_www$1 permanent;
    }
}

此外,我希望这适用于任何域名,而无需明确告诉Nginx重写domain1.com - > www.domain1.com,domain2.com - > www.domain2.com等,因为我有大量的域要重写。

nginx rewrite url-rewriting
6个回答
62
投票

As noted in the Nginx documentation, you should avoid using the if directive in Nginx where possible,因为只要你的配置中有if,你的服务器就需要评估每一个请求,以决定是否匹配if

更好的解决方案是多个服务器指令。

server {
        listen 80;
        server_name website.com;
        return 301 $scheme://www.website.com$request_uri;
}

server {
        listen 80;
        server_name www.website.com;
        ... 
}

如果您尝试为启用SSL(HTTPS)的站点提供服务,则可以使用或多或少三种不同的选项。

  1. 设置多个IP地址,让每个服务器指令监听自己的IP(或者不同的端口,如果这是你的选项)。此选项需要website.com和www.website.com的SSL证书,因此您要么拥有通配符证书,要么拥有UNI证书(多个域),或者只有两个不同的证书。
  2. 在应用程序中重写。
  3. 使用可怕的if指令。

There is also an option to use SNI, but I'm not sure this is fully supported as of now


15
投票
if ($host !~* ^www\.) {
    rewrite ^(.*)$ http://www.$host$1 permanent;
}

14
投票

好吧,我想我真的不需要外部的“if”语句,因为我只是检查xxx.xxx格式的域名。以下适用于我,虽然它不健壮。如果有更好的解决方案,请告诉我。

    if ($host ~* ^([a-z0-9\-]+\.(com|net|org))$) {
        set $host_with_www www.$1;
        rewrite ^(.*)$ http://$host_with_www$1 permanent;
    }

编辑:在正则表达式中添加连字符,因为它是主机名中的有效字符。


6
投票
if ($host ~* ^[^.]+\.[^.]+$) {
    rewrite ^(.*)$ http://www.$host$1 permanent;
}

它只能获得有效的主机名,因为请求永远不会进入您的服务器,因此不需要构建自己的验证逻辑。


4
投票

nginx文档警告不要使用if进行重写。请看这里的链接:http://wiki.nginx.org/Pitfalls#Server_Name


1
投票

没有条件的HTTP和HTTPS:

server {
    listen          80;
    listen          443;
    server_name     website.com;

    return 301 $scheme://www.website.com$request_uri;
}
server {
    listen          80;
    listen          443 default_server ssl;
    server_name     www.website.com;

    # Your config goes here #
}

0
投票

多个域的解决方案,为我工作nginx 1.17:

server {
        listen                                80;
        server_name                           .example.com;

        set $host_with_www                 $host;

        if ($host !~* www\.(.*)) {
            set $host_with_www www.$host;
        }

        return                                301 https://$host_with_www$request_uri;
}

在此配置示例中,如果您不想重写,则另外在HTTPS上重写HTTP - 在return字符串中使用http://替换https://。

如果你想保持协议 - 使用$scheme变量。

© www.soinside.com 2019 - 2024. All rights reserved.