我在Rackspace云上使用nginx,遵循一个教程,搜索了网络,到目前为止还不能得到这个排序。

我想要www.mysite.example去mysite。例如,正常的。htaccess为SEO和其他原因。

My /etc/nginx/sites-available/www.example.com.vhost config:

server {
       listen 80;
       server_name www.example.com example.com;
       root /var/www/www.example.com/web;

       if ($http_host != "www.example.com") {
                 rewrite ^ http://example.com$request_uri permanent;
       }

我也试过

server {
       listen 80;
       server_name example.com;
       root /var/www/www.example.com/web;

       if ($http_host != "www.example.com") {
                 rewrite ^ http://example.com$request_uri permanent;
       }

我也试过。第二次尝试都给出重定向循环错误。

if ($host = 'www.example.com' ) {
rewrite ^ http://example.com$uri permanent;
}

我的DNS设置为标准:

site.example 192.192.6.8 A type at 300 seconds
www.site.example 192.192.6.8 A type at 300 seconds

(示例ip和文件夹已用于示例和帮助人们在未来)。我使用Ubuntu 11。


当前回答

下面是如何为多个www到no-www服务器名做这件事(我用这个子域名):

server {
        server_name 
             "~^www\.(sub1.example.com)$"
             "~^www\.(sub2.example.com)$"
             "~^www\.(sub3.example.com)$";
         return 301 $scheme://$1$request_uri ;
}

其他回答

不确定是否有人注意到它可能是正确的返回301,但浏览器阻塞它做

rewrite ^(.*)$ https://yoursite.example$1;

比:

return 301 $scheme://yoursite.example$request_uri;

最佳实践:使用硬编码的server_name分离服务器

nginx的最佳实践是使用一个单独的服务器进行重定向(不与主配置的服务器共享),硬编码所有内容,并且根本不使用正则表达式。

如果您使用HTTPS,可能还需要硬编码域,因为您必须事先知道您将提供哪些证书。

server {
    server_name www.example.com;
    return  301 $scheme://example.com$request_uri;
}
server {
    server_name www.example.org;
    return  301 $scheme://example.org$request_uri;
}
server {
    server_name example.com example.org;
    # real configuration goes here
}

在server_name中使用正则表达式

如果你有很多站点,并且不关心最优的性能,但是希望它们中的每一个都有相同的关于www的策略。前缀,那么就可以使用正则表达式了。使用独立服务器的最佳实践仍然有效。

请注意,如果您使用https,这个解决方案会很棘手,因为如果您想要正常工作,那么您必须有一个单独的证书来覆盖所有的域名。


非WWW到WWW w/ regex在专用的单个服务器上为所有网站:

server {
    server_name ~^(?!www\.)(?<domain>.+)$;
    return  301 $scheme://www.$domain$request_uri;
}

WWW到非WWW w/ regex在专用的单个服务器上为所有网站:

server {
    server_name ~^www\.(?<domain>.+)$;
    return  301 $scheme://$domain$request_uri;
}

WWW到非WWW w/ regex在专用服务器仅为某些网站:

可能有必要限制正则表达式只覆盖几个域,然后你可以使用这样的方法只匹配www.example.org, www.example.com和www.subdomain.example.net:

server {
    server_name ~^www\.(?<domain>(?:example\.org|example\.com|subdomain\.example\.net))$;
    return  301 $scheme://$domain$request_uri;
}

使用nginx测试正则表达式

你可以在你的系统上用pcretest测试正则表达式是否如预期的那样工作,这与你的nginx将用于正则表达式的pcretest库完全相同:

% pcretest 
PCRE version 8.35 2014-04-04

  re> #^www\.(?<domain>(?:example\.org|example\.com|subdomain\.example\.net))$#
data> test
No match
data> www.example.org
 0: www.example.org
 1: example.org
data> www.test.example.org
No match
data> www.example.com
 0: www.example.com
 1: example.com
data> www.subdomain.example.net
 0: www.subdomain.example.net
 1: subdomain.example.net
data> subdomain.example.net
No match
data> www.subdomain.example.net.
No match
data> 

请注意,你不必担心后面的点或大小写,因为nginx已经照顾它,根据nginx服务器名称regex当“主机”头有一个结尾点。


如果在现有的服务器/ HTTPS:

这个最终的解决方案通常不被认为是最佳实践,然而,它仍然有效并能完成工作。

事实上,如果您正在使用HTTPS,那么这个最终的解决方案可能会更容易维护,因为您不必在不同的服务器定义之间复制粘贴一大堆ssl指令,而是可以将代码片段只放在所需的服务器中,从而更容易调试和维护您的站点。


非www到www:

if ($host ~ ^(?!www\.)(?<domain>.+)$) {
    return  301 $scheme://www.$domain$request_uri;
}

WWW到非WWW:

if ($host ~ ^www\.(?<domain>.+)$) {
    return  301 $scheme://$domain$request_uri;
}

硬编码单个首选域

如果你想要更多的性能,以及单个服务器可能使用的多个域之间的一致性,显式硬编码单个首选域可能仍然是有意义的:

if ($host != "example.com") {
    return  301 $scheme://example.com$request_uri;
}

引用:

http://nginx.org/r/server_name http://nginx.org/r/return http://nginx.org/en/docs/http/server_names.html

我结合了所有简单答案中最好的,没有硬编码的域。

301永久重定向从非www到www (HTTP或HTTPS):

server {
    if ($host !~ ^www\.) {
        rewrite ^ $scheme://www.$host$request_uri permanent;
    }

    # Regular location configs...
}

如果你更喜欢非HTTPS、非www而不是HTTPS, www同时重定向:

server {
    listen 80;

    if ($host !~ ^www\.) {
        rewrite ^ https://www.$host$request_uri permanent;
    }

    rewrite ^ https://$host$request_uri permanent;
}

实际上你甚至不需要重写。

server {
    #listen 80 is default
    server_name www.example.com;
    return 301 $scheme://example.com$request_uri;
}

server {
    #listen 80 is default
    server_name example.com;
    ## here goes the rest of your conf...
}

因为我的答案得到越来越多的支持,但上面也一样。永远不要在这种情况下使用重写。为什么?因为nginx必须处理并开始搜索。如果你使用return(应该在任何nginx版本中都可用),它会直接停止执行。这在任何上下文中都是首选。

将非SSL和SSL都重定向到它们的非www对应对象:

server {
    listen               80;
    listen               443 ssl;
    server_name          www.example.com;
    ssl_certificate      path/to/cert;
    ssl_certificate_key  path/to/key;

    return 301 $scheme://example.com$request_uri;
}

server {
    listen               80;
    listen               443 ssl;
    server_name          example.com;
    ssl_certificate      path/to/cert;
    ssl_certificate_key  path/to/key;

    # rest goes here...
}

如果您的服务器只在端口80(默认)上监听,并且listen选项不包含ssl关键字,$scheme变量将只包含http。不使用该变量将不会为您带来任何性能。

注意,如果使用HSTS,则需要更多的服务器块,因为HSTS头不应该通过非加密连接发送。因此,您需要带重定向的未加密服务器块和带重定向和HSTS标头的加密服务器块。

重定向所有内容到SSL(个人配置UNIX与IPv4, IPv6, SPDY,…):

#
# Redirect all www to non-www
#
server {
    server_name          www.example.com;
    ssl_certificate      ssl/example.com/crt;
    ssl_certificate_key  ssl/example.com/key;
    listen               *:80;
    listen               *:443 ssl spdy;
    listen               [::]:80 ipv6only=on;
    listen               [::]:443 ssl spdy ipv6only=on;

    return 301 https://example.com$request_uri;
}

#
# Redirect all non-encrypted to encrypted
#
server {
    server_name          example.com;
    listen               *:80;
    listen               [::]:80;

    return 301 https://example.com$request_uri;
}

#
# There we go!
#
server {
    server_name          example.com;
    ssl_certificate      ssl/example.com/crt;
    ssl_certificate_key  ssl/example.com/key;
    listen               *:443 ssl spdy;
    listen               [::]:443 ssl spdy;

    # rest goes here...
}

我猜你们现在可以自己想象其他化合物的这种模式。

更多我的配置?去这里和这里。

如果在此工作时遇到困难,则可能需要添加服务器的IP地址。例如:

server {
listen XXX.XXX.XXX.XXX:80;
listen XXX.XXX.XXX.XXX:443 ssl;
ssl_certificate /var/www/example.com/web/ssl/example.com.crt;
ssl_certificate_key /var/www/example.com/web/ssl/example.com.key;
server_name www.example.com;
return 301 $scheme://example.com$request_uri;
}

其中XXX.XXX.XXX.XXX是IP地址(显然)。

注意:必须定义ssl crt和key location来正确重定向https请求

不要忘记在修改后重新启动nginx:

service nginx restart