I would like to display a custom error page if there is a folder missing.
My directory structure is the following:
data
defaults
error.html
nothosted.html
example.com
misc
error.html
example.net
...
So when someone accesses example.de whose folder does not exist the defaults/nothosted.html gets displayed. But if the folder exists my @error location should be used.
Currently this is my configuration which does not work.
server_name ~^(\*\.)?(?<subdomain>[a-z\d][a-z\d-]*[a-z\d]\.)(?<domain>.+)+$;
location / {
root /data/$subdomain$domain;
try_files $uri $uri/ @nothosted;
index index.html index.htm index.php;
}
location @nothosted {
root /data;
set $link /$subdomain$domain;
if (!-d $link) {
set $link /defaults/nothosted.html;
}
try_files $link @error;
}
error_page 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 421 422 423 424 425 426 428 429 431 451 500 501 502 503 504 505 506 507 508 510 511 @error;
location @error {
internal;
ssi on;
auth_basic off;
root /data;
try_files /$subdomain$domain/misc/error.html /$domain/misc/error.html /defaults/error.html =404;
}
The error i get in the nginx log is the following:
*1 directory index of "/data/" is forbidden
The error i get trying to connect to a domain whose folder does not exist is a "403 forbidden" from the @error location.
The permissions should be right (the whole /data/ folder is owned by the nginx user)
EDIT: When using the following nothosted location the correct link gets displayed on the webpage once accessed. So the problem has to be the try_files function.
location @nothosted {
root /data;
set $link /$subdomain$domain;
if (!-d $document_root/$subdomain$domain) {
set $link /defaults/nothosted.html;
}
return 200 $link;
add_header Content-Type text/plain;
}
This one can't be solved the way you are trying to solve it. The reason is that if
is evil when used in location
context. I have known this for a long time, but I didn't know it can be so evil sometimes.
Generally nginx configuration directives are declarative (for example there is no difference where will you place a directive like proxy_pass
, within the exact location it can be placed anywhere). While in common nginx directives from the rewrite module are the only ones that can be considered as imperative (see the internal implementation chapter from the module documentation), an if
directive is a very special case. After reading the aforementioned implementation description, for a long time I thought that using only directives from the rewrite module inside the if
block will made such a block completely safe one. Unfortunately it isn't true. One of the best descriptions of how this directive actually works was made by Yichun Zhang, an author of famous lua-nginx-module and the OpenResty bundle. In fact every if
directive implicitly creates a nested location which tries to inherit all the declarations from the parent one. However the try_files
directive is not gets inherited (nginx trac ticket), and such a virtual nested location, if being selected to handle the request and having a static content handler, would have a default PRECONTENT
phase handler similar to try_files $uri $uri/ =404
. Error message you received comes from the fact that implicit try_files
directive tries to use the current URI (probably the /
one) and the implicit index
directive can't find an index file inside the webroot /data
directory.
That is, the chosen method won't work. You can try the following instead:
if (!-d /data/$subdomain$domain) {
rewrite ^ /defaults/nothosted.html;
}
location / {
root /data/$subdomain$domain;
try_files $uri $uri/ =404;
index index.html index.htm index.php;
}
location = /defaults/nothosted.html {
internal;
root /data;
}
error_page 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 421 422 423 424 425 426 428 429 431 451 500 501 502 503 504 505 506 507 508 510 511 @error;
location @error {
internal;
ssi on;
auth_basic off;
root /data;
try_files /$subdomain$domain/misc/error.html /$domain/misc/error.html /defaults/error.html =404;
}
This solution has a drawback - the /defaults/nothosted.html
URI won't work for any of the hosted sites. If you find it unacceptable, you can use some kind of unique random string instead of nothosted
one (and rename that nothosted.html
file accordingly).