Nginnx config for Yii 2 Advanced App Template
Long story short: use the first method provided below.
The rest of the answer is a list of recommendations.
I'm going to separate my answer in two sections. In the first part, I will tell you the easiest and the fastest way to achieve your goal according to your desired URL requests, but it partly breaks the app structure, nothing serious, though.
In the second part, I will describe you where you made mistakes in your configuration file and I will show you a poorly written configuration for your needs which works.
I. Shared Hosting deployment
I highly encourage you to use this. This is an official way from Yii 2 documentation to make backend work at the same domain, although it helps to deploy a project to a shared hosting. And it doesn't require any additional nginx configuration, just a basic one for frontend root.
Let me write a simple list according to this guide:
- Move contents from
/backend/web
to/frontend/web/admin
. - Correct scripts' paths in
/frontend/web/admin/index.php
(andindex-test.php
, if you use it)
That's all, you have your backend at the same domain at /admin
URL. Additionally, read the last section of the guide regarding cookies. The advanced template was designed to use different domains for each environment, therefore the guide describes backend config for shared hosting to keep cookies from frontend and backend separate.
Of course, don't forget to modify your /environments
files for proper initialization of your project with /init
script.
II. Nginx configuration
Mistakes
I'm not a profressional nginx administrator, but I can describe what's wrong in your configuration based on my personal experience and the documentation. Unfortunately, I won't be able to provide links to the documentation, because my current rating won't allow me to post more than 2 links.
Server context root
You do not have root
directive in your server context. Thus, when ~ \.php$
location is matched, it doesn't have root at all and uses default nginx root. Try setting common root
directive in the server
context, then all locations will have it by default. For example:
server {
# Beginning of your configuration
# ...
root /srv/http/yii2/public/frontend/web;
# The rest of your configuration
# ...
}
Not having a higher context root is a common pitfall.
root
instead of alias
Secondly, when a location is matched, the uri is appended to the location's root and that's the path the server attempts to look for. Thus, your /admin
location suggests that the server search for $host_path/backend/web/admin
. In your situation, you should use alias
directive which tells the server that the matched location uri refers to alias path, not appended to root:
location /admin {
alias $host_path/backend/web;
# The rest of location
# ...
}
I recommend that you read related nginx documentation about location
, root
and alias
directives.
Working but poorly written configuration
I post this sample configuration with comments for your understanding only, not for production use, I dicourage you to apply it for your production (until you're positive it's safe and sound).
It works, but it has an annoying defect: backend cannot find Yii2 entry script if you request it directly (like /admin/index.php
), so it must be used with enablePrettyUrl
set to true
and showScriptName
set to false
, however it finds any other PHP script in the backend web root.
server {
# The beginning of your configuration
# ...
# By default we will provide frontend
root /srv/http/yii2/public/frontend/web;
index index.php;
location / {
try_files $uri $uri/ /index.php?$args;
}
location /admin {
# We use /web/index here to make backend call to php scripts
# distinct from frontend call
index /web/index.php;
alias $root_base/backend/web;
try_files $uri $uri/ /web/index.php?$args;
# Rewrite PHP requests from /admin to /web
# However, Yii2 entry script returns 404
location ~ ^/admin/.*\.php$ {
rewrite ^/admin/(.*)$ /web/$1;
}
}
location ~ ^/web/.*\.php$ {
# Make sure this location cannot be called externally
internal;
# Remember, that the uri of this location
# will be appended to this root!
root $root_base/backend;
# PHP settings for backend
}
location ~ \.php$ {
# PHP settings for frontend
}
# The rest of your configuration
# ...
}
Additionally, add baseUrl
property to the request
component in your Yii2 backend config and set it to /admin
.
I hope my answer will help you deploying your Yii2 advanced project and understanding nginx more, nevertheless your question is 6 months old.
Here's my working config, based on accepted answer. My project backend
directory is renamed to admin
# Example config for nginx
# frontend is available on yii-application.local/
# backend (admin) is available on yii-application.local/admin
# make sure that @app/frontend/config/main.php and @app/admin/config/main.php components sections are configured properly
# e.g. @app/frontend/config/main.php
# 'homeUrl' => '',
# ...
# 'components' => [
# 'request' => [
# 'baseUrl' => '',
# ],
# 'urlManager' => [
# 'enablePrettyUrl' => true,
# 'showScriptName' => false,
# ],
# ]
#
# e.g. @app/admin/config/main.php
# 'homeUrl' => '/admin',
# ...
# 'components => [
# 'request' => [
# 'baseUrl' => '/admin',
# ],
# 'urlManager' => [
# 'enablePrettyUrl' => true,
# 'showScriptName' => false,
# ],
# ]
server {
set $project_root /home/yii/apps/yii-advanced;
set $fcgi_server unix:/opt/php/var/run/php5-fpm.sock;
charset utf-8;
client_max_body_size 128M;
listen 80;
server_name yii-application.local;
root $project_root/frontend/web;
index index.php;
access_log /home/yii/apps/yii-advanced/logs/access-backend.log;
error_log /home/yii/apps/yii-advanced/logs/error-backend.log;
location / {
try_files $uri $uri/ /index.php?$args;
}
location /admin {
index /web/index.php;
alias $project_root/admin/web;
try_files $uri $uri/ /web/index.php?$args;
location ~ ^/admin/.*\.php$ {
rewrite ^/admin/(.*)$ /web/$1;
fastcgi_pass $fcgi_server;
include fastcgi.conf;
}
}
location ~ ^/web/.*\.php$ {
internal;
root $project_root/admin;
fastcgi_pass $fcgi_server;
include fastcgi.conf;
}
location ~* \.php$ {
try_files $uri =404;
fastcgi_pass $fcgi_server;
include fastcgi.conf;
}
location ~* \.(htaccess|htpasswd|svn|git) {
deny all;
}
}