Nginx静态服务配置---详解root和alias指令

静态文件

Nginx以其高性能著称,常用与做前端反向代理服务器。同时nginx也是一个高性能的静态文件服务器。通常都会把应用的静态文件使用nginx处理。

配置nginx的静态文件有两个指令,一个 root 和一个 alias。对于这两个指令,是否需要在路径的后面加上斜杠,经常容易让人犯晕,本文通过尝试不同的匹配规则,归纳了一个比较通用的配置方式。

基本配置

简明 Nginx Location Url 配置笔记一文关于location url配置的实验一样,本文也使用vagrant虚拟机里的nginx。其基本配置如下:

/etc/nginx/sites-enabled/pro.conf

server {
        listen 80 default_server;


        server_name localhost;

        access_log /var/log/nginx/pro/access.log;
        error_log /var/log/nginx/pro/error.log;

        error_page 404 /404.html;

        root /vagrant/pro;
        index index.html index.htm;

}

项目的目录如下:

☁  pro  tree
.
├── 403.html
├── 404.html
├── index.html
├── static
│   ├── flask
│   │   └── m.png
│   └── stc.jpg
└── upload
    └── up.png

3 directories, 6 files

分别有两个静态文件夹,一个是static,另外一个是upload。

初识root

root 是指定项目的根目录,适用与server和location。可以指定多个,如果locaiton没有指定,会往其外层的server或http中寻找继承。

访问http://192.168.33.10/static/stc.jpg 会发现图片已经返回。我们还尚未配置 location,为啥会正确的找到文件?学习root或者alias指令的时候,最好的办法是给文件拓展名加上一个字符,使得该文件在硬盘中不存在,那么就能从nginxerror.log中看到nginx寻找文件的方式。

访问 http://192.168.33.10/static/stc.jpgx,然后查看 /var/log/nginx/pro/error.log文件,可以看到如下的错误信息:

2016/09/28 07:41:48 [error] 4416#0: *70 open() "/vagrant/pro/static/stc.jpgx" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /static/stc.jpgx HTTP/1.1", host: "192.168.33.10"

/vagrant/pro/static/stc.jpgx 文件不存在。的确我们没有这个文件。如果文件名正确,就能访问,原因是由于在server中指定了root /vagrant/pro,此时的nginx就在该目录下寻找文件,而url上的地址,正好和文件的路径一致

 http://192.168.33.10  /static/stc.jpg 
 /vagrant/pro          /static/stc.jpg

由此可以猜想,nginx中root指令的地址,其实是替换了匹配后的url中的host。

root指令

为了验证上面的猜想,需要多写几个location做实验。添加一个location配置如下:

location ^~ /static {
    root /vagrant/pro/static;
}

再次访问http://192.168.33.10/static/stc.jpg,发现并不能显示图片了,查看error.log 返回如下:

2016/09/28 07:48:57 [error] 5978#0: *71 open() "/vagrant/pro/static/static/stc.jpg" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /static/stc.jpg HTTP/1.1", host: "192.168.33.10"

nginx把地址识别成/vargrant/pro/static/static/stc.jpg多了一个static,套用上面的规则,其组合为192.168.33.10 == /vagrant/pro/static ,url是/static/stc.jpg。置换可以得到/vagrant/pro/static + /static/stc.jpg。与错误的error一致。解决方案就是把root中的static去掉,马上就能访问图片了。

既然是那么把文件夹static命名为stc,其结果又会怎样?

location ^~ /static {
    root /vagrant/pro;
}

访问 http://192.168.33.10/static/stc.jpg 得到错误:

2016/09/28 07:54:46 [error] 5992#0: *73 open() "/vagrant/pro/static/stc.jpg" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /static/stc.jpg HTTP/1.1", host: "192.168.33.10"

计算路径/vagrant/pro + /static/stc.jpg, 找不到/vagrant/pro/static/stc.jpg文件,符合之前所说的规则,尝试修改location:

location ^~ /stc {
    root /vagrant/pro;
}

因为url变了,访问http://192.168.33.10/stc/stc.jpg,才能够找到图片。现在把stc文件夹变回static。

root 与 斜杠

很多人会疑惑,路径最后的斜杠/是否要加呢?location中的static后面的斜杠,和匹配后的url有关,不再赘述。root中的路径的斜杠/可以再通过实验确定。把location配置如下:

location ^~ /static/ {
   root /vagrant/pro/;
}

访问http://192.168.33.10/static/stc.jpg 一切正常,访问http://192.168.33.10/static/stc.jpg,error为找不到"/vagrant/pro/static/stc.jpgs"文件。

如果按照root替换host的规则,那么替换过程为

/vagrant/pro/ + /static/stc.jpg == /vagrant/pro//static/stc.jpg。在*nix系统中, 多个斜杠和一个斜杠是等价的,也就是 /vagrant/pro//static/stc.jpg/vagrant/pro/static/stc.jpg一样。

这样一来,root路径后面的斜杠,加与不加效果都一样。既然如此,肯定有人会想到这么配置:

location ^~ static/ {
    root /vagrant/pro;
}

如果安装之前上面的即算法,那么应该是 /vagrant/pro + static/stc.jpg,相加的应该是/vagrant/prostatic/stc.jpg,按理说应该是错误,可是实际上却能访问图片。咄咄怪事?

如果对前文nginx location的url匹配规则了解的话,应该看出来了其实 ^~ static/并不能匹配。修改 location

location ^~ static/ {
    rewrite ^ http://google.com;
   # root /vagrant/pro;
}

访问http://192.168.33.10/static/stc.jpg依然可以得到图片,没有跳转google,说明并没有匹配^~ static/

其实原理也很简单,还记得我们第一次实验,当时尚未配置location,也同样可以返回图片。没错,尽管^~ static/没有匹配,而外层的server定义了root为/vagrant/pro,因此搜索图片正常返回,再注释外层的root,再一次访问。此时会得到一个404,查看error如下:

2016/09/28 08:18:15 [error] 6227#0: *82 open() "/usr/share/nginx/html/static/stc.jpg" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /static/stc.jpg HTTP/1.1", host: "192.168.33.10"

/usr/share/nginx/html/static/stc.jpg,说明即使没有指定root,nginx默认也有一个root,/usr/share/nginx/html。当然,这个配置和 ^~ static/没有关系。

如果~ static/stc.jpgs? 那么就能命中,此时访问图片,依然能够正确的解析,因此,并不存在 /vagrant/pro + static/stc.jpg这种情况。理解这里的关键是 root替换host,并加上匹配后的url,匹配后的url当然包括前面的斜杠,匹配部分的url则不会。

对于 ~ static/stc.jpgs?模式,访问urlhttp://192.168.33.10/static/stc.jpg

  • 匹配后的url为 /static/stc.jpg
  • 匹配部分的url为 static/stc.jpg

掌握这个很重要,直接关系到后面alias指令与斜杠的关系。

对于root指令,我们可以归纳。

  1. 对于匹配后的url地址,将匹配的location中的root路径替换访问url的host即得到文件的真实地址。(多个斜杠其实等价于一个斜杠)
  2. 如果不匹配location,则寻找更外层的root做替换。
  3. root指令最后的斜杠可加可不加。

alias指令

对于root,操作上很简单,只要把root地址替换host后就是文件在硬盘路径(真实地址)。对于alise,它并不是替换匹配后的url地址,而是替换匹配部分的url。alias指令也可以有多个。

添加一个location,和root的方式几乎一样:

location ^~ /upload {
   alias /vagrant/pro;
}

访问http://192.168.33.10/upload/up.png并没有图片,查看error得到:

2016/09/28 08:36:18 [error] 6312#0: *90 open() "/vagrant/pro/up.png" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /upload/up.png HTTP/1.1", host: "192.168.33.10"

可见 alias的模式并不是/vagrant/pro + /upload/up.png,而是 /vagrant/pro + /up.png

alias这个词在计算机里很常用,字面意思是“别名”,顾名思议就是换一个名字啦。实际替换规则就是把匹配的url地址,换成alias中的路径即可。例如上述的例子替换过程可以模拟如下:

过程 模式或url
url模式 ^~ /upload
alias路径 /vagrant/pro
访问地址 http://192.168.33.10/upload/up.png
匹配部分的地址 /upload + /up.png
替换 /upload == /vagrant/pro
结果 /vagrant/pro + /up.png

为了修改图片的访问,修改locaton如下:

location ^~ /upload {
    alias /vagrant/pro/upload;
}

此时访问http://192.168.33.10/upload/up.png就能得到正确的图片啦,仿造上面的计算过程为:

过程 模式或url
url模式 ^~ /upload
alias路径 /vagrant/pro/upload
访问地址 http://192.168.33.10/upload/up.png
匹配部分的地址 /upload + /up.png
替换 /upload == /vagrant/pro/upload
结果 /vagrant/pro/upload + /up.png

从结果可以看出,正确的找到了文件路径,如果alias指令路径加上斜杠,那么计算处理的文件路径为:

/upload == /vagrant/pro/upload
/vagrant/pro/upload/ + /up.png

多个斜杠是合法的。等价于一个斜杠的情况。

下面修改locaiton如下:

location ^~ /upload/ {
   alias /vagrant/pro/upload;
}

此时匹配时的url则变成 /upload/ + up.jpg, 那么置换的结果为 /vagrant/pro/upload + up.png,而/vagrant/pro/uploadup.png的路径是非法的,从error中也能看到置换的错误:

2016/09/28 08:52:44 [error] 6452#0: *92 open() "/vagrant/pro/uploadup.png" failed (2: No such file or directory), client: 192.168.33.1, server: localhost, request: "GET /upload/up.png HTTP/1.1", host: "192.168.33.10"

解决办法也很简单,把/vagrant/pro/upload 改成 /vagrant/pro/upload/即可。由此可见,alias最后的斜杠并不像root指令那样可有可无,是否需要,取决于配合loacation的url匹配模式。

前文root模式中,考虑了没有根的斜杠(~ static/stc.jpgs?)这种情况,alias情况下会很难捕捉错误。如果locaion配置如下:

location ^~ upload/ {
      alias /vagrant/pro/upload/;
}

替换置换的文件路径应该为 /vagrant/pro/upload/up.png,可是实际测试中,这样配置alias,会一直导致一个301的重定向,如果alias目录没有打开autoindex,则会抛出一个403错误。具体情况尚未知晓,不知道是不是nginx的bug。为了避免这种情况,使用alias的时候,尽量不要配置location为 ^~ upload/的模式,并且不从根指定url,还是显得不伦不类。

alise作为别名,比起root的一大好处就是不一定要url上的路径和文件路径一样,因为alise并不是替换host,而是替换匹配部分的host。修改配置如下:

location ^~ /upload/ {
    alias /vagrant/pro/static/;
}

访问 http://192.168.33.10/upload/stc.jpg或者 http://192.168.33.10/upload/flask/m.png都能正确的访问到static目录下的文件,尽管url上是upload。

替换规则也很简单,/upload/ == /vagrant/pro/static/ 得到 /vagrant/pro/static/ + stc.jpg/vagrant/pro/static/ + flask/m.png

总结

nginx的静态文件配置中,root和alias指令都能实现。为了避免混淆,尽量不要写没有根路径的url模式,即避免 static/这样的开头,根路径斜杠需要保留,没有根路径其实也很奇怪。

root和alias的区别主要在于替换的部分,root模式中,会把root配置的路径替换匹配后的url中的host。alias则把他指定的路径,替换url中匹配的部分。指令中的斜杠对于root指令没有影响,对于alise则按照替换规则匹配即可。

root 指令

location /dir/ 
root root_path ->  http://host/dir/file.txt  -> root_path/dir/file.txt

alias 指令

location /dir
alias alias_path ->  http://host /dir /file.txt  -> alias_path/file.txt

location /dir/ 
alias alias_path/ ->  http://host /dir/ file.txt  -> alias_path/file.txt

了解了root和alise之后,通常最佳实际是配置一个项目的根root,其他的文件夹则使用alias,毕竟alias更加灵活。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,600评论 18 139
  • 1.ngnix介绍 ngnix www服务软件 俄罗斯人开发 开源 性能很高 本身是一款静态WWW软件 静态小文件...
    逗比punk阅读 2,081评论 1 6
  • Page 1:nginx 服务器安装及配置文件详解 CentOS 6.2 x86_64 安装 nginx 1.1 ...
    xiaojianxu阅读 8,527评论 1 41
  • 《老男孩Linux运维》笔记 隐藏Nginx软件版本号 一般来说,软件的漏洞都和版本有关。因此要尽量隐藏对访问用户...
    Zhang21阅读 3,626评论 0 28
  • 基本配置 为了探究nginx的url配置规则,当然需要安装nginx。我使用了vagrant创建了一个虚拟环境的u...
    人世间阅读 15,250评论 7 30