eZ Publish 4.2011 Community (4.5) with Nginx and PHP-FPM on Debian 6 (“Squeeze”)

Introduction

Nginx is an event-driven, lightweight and high-performance HTTP server and reverse proxy/load balancer. With the recent release of Nginx 1.0.0, PHP-FPM 5.3.6 and improved documentation, it is now simpler to install and configure Nginx to serve eZ Publish sites with the same features as Apache HTTP Server while receiving the performance and memory consumption benefits of Nginx.

The tutorial covers the setup and basic configuration of Nginx, PHP/MySQL and eZ Publish 4.2011 Community Edition (4.5 Enterprise) on Debian 6. The steps can also be used as an outline for performing the same setup on other versions or distributions.

 

Pre-requisites and target population

Technical pre-requisites are a basic knowledge of web servers, Debian package installation using apt or similar, and some experience with installing or moving/upgrading eZ Publish.

The reference system for this tutorial is a clean network install of Debian 6 with admin tools only, running in a virtual machine with two 2-Ghz processors, 2 GB RAM and 16 GB hard disk space.

NOTE: All operations are assumed to be run as the root user unless otherwise specified.

 

Step 1: Update Debian package lists

Nginx 1.0.0 and PHP-FPM (FastCGI Process Manager) 5.3.6 is available through dotdeb, and your apt sources file should be updated accordingly.

Add the following to your /etc/apt/sources.list file:

deb http://packages.dotdeb.org stable all
deb-src http://packages.dotdeb.org stable all
 

Update the dotdeb key:

wget http://www.dotdeb.org/dotdeb.gpg 
cat dotdeb.gpg | apt-key add –
 

Run:

apt-get update
 

Step 2: Install required packages

apt-get install nginx
apt-get install mysql-server
apt-get install php5-cli php5-common php5-suhosin
apt-get install php5-fpm php5-cgi php5-curl php5-gd php5-imagick php5-mysql php5-apc imagemagick
 

Step 3: Configure PHP-FPM

Update the files listed below. You need to substitute local variables where noted.

In /etc/php5/fpm/pool.d/www.conf:

listen = /tmp/php-fpm.sock
 

Since this is a single-server installation, we will configure PHP-FPM to listen on a unix socket as opposed to a tcp/ip socket, thus bypassing the tcp/ip stack.

The socket file can be placed anywhere you like – this is just an example for convenience. Just make sure it’s in a writable directory, and remember that Nginx has to be pointed to the same location in Step 4.

In /etc/php5/fpm/php.ini:

date.timezone = "YOUR TIME ZONE, such as Europe/Oslo"
short_open_tag = On
memory_limit = 256M
max_execution_time = 120
cgi.fix_pathinfo=0

NOTE: the last setting is very important as it prevents a security issue when running PHP in cgi mode.

 

In /etc/php5/cli/php.ini:

date.timezone = "YOUR TIME ZONE, such as Europe/Oslo"
max_execution_time = 120
 

Step 4: Configure Nginx

Now we need to configure Nginx to run with PHP over FastCGI.

Update the files listed below. You need to substitute local variables where noted.

In /etc/nginx/nginx.conf:

user www-data;
worker_processes 4;
pid /var/run/nginx.pid;
events {
    worker_connections 2048;
    # multi_accept on;
}
http {
##
# Basic Settings
##
sendfile on;
tcp_nopush on;
tcp_nodelay on;
keepalive_timeout 65;
types_hash_max_size 2048;
server_tokens off;
include /etc/nginx/mime.types;
default_type application/octet-stream;
##
# Logging Settings
##
access_log /var/log/nginx/access.log;
error_log /var/log/nginx/error.log;
##
# Gzip Settings
##
gzip on;
gzip_vary on;
gzip_disable "msie6";
gzip_proxied any;
gzip_comp_level 5;
gzip_buffers 32 8k;
gzip_http_version 1.1;
gzip_types text/plain text/css application/json application/x-javascript text/xml application/xml application/xml+rss text/javascript;
##
# Virtual Host Configs
##
include /etc/nginx/conf.d/*.conf;
include /etc/nginx/sites-enabled/*;
}
 

In /etc/nginx/fastcgi_params:

fastcgi_connect_timeout 60;
fastcgi_send_timeout 180;
fastcgi_read_timeout 180;
fastcgi_buffer_size 128k;
fastcgi_buffers 4 256k;
fastcgi_busy_buffers_size 256k;
fastcgi_temp_file_write_size 256k;
fastcgi_intercept_errors on;
 

The above are standard settings to prevent PHP-FPM from choking, and you may want to experiment with tuning these to your particular environment. They serve as a good starting point for most installations.

In /etc/nginx/sites-available/default:

server {
server_name YOUR.INTERNET.HOSTNAME;
root YOUR_EZPUBLISH_ROOT;
index index.php;
location / {

rewrite "^/var/storage/(.*)$" "/var/storage/$1" break;
rewrite "^/var/([^/]+)/storage/(.*)$" "/var/$1/storage/$2" break;
rewrite "^/var/(([^/]+/)?)cache/(texttoimage|public)/(.*)$" "/var/$1cache/$3/$4" break;
rewrite "^/design/([^/]+)/(stylesheets|images|javascript)/(.*)$" "/design/$1/$2/$3" break;
rewrite "^/share/icons/(.*)$" "/share/icons/$1" break;
rewrite "^/extension/([^/]+)/design/([^/]+)/(stylesheets|images|javascripts|javascript|flash|lib?)/(.*)$" "/extension/$1/design/$2/$3/$4" break;
rewrite "^/packages/styles/(.+)/(stylesheets|images|javascript)/([^/]+)/(.*)$" "/packages/styles/$1/$2/$3/$4" break;
rewrite "^/packages/styles/(.+)/thumbnail/(.*)$" "/packages/styles/$1/thumbnail/$2" break;
rewrite "^/favicon\.ico$" "/favicon.ico" break;
rewrite "^/robots\.txt$" "/robots.txt" break;
rewrite "^/var/cache/debug.html(.*)$" "/var/cache/debug.html$1" break;
rewrite "^/var/(([^/]+/)?)cache/public/(.*)$" "/var/$1cache/public/$3" break;
rewrite "^/var/([^/]+)/cache/debug\.html(.*)$" "/var/$1/cache/debug.html$2" break;
rewrite "content/treemenu/?$" "/index_treemenu.php" break;
rewrite "ezjscore/call/?$" "/index_ajax.php" break;
rewrite "^(.*)$" "/index.php?$1" last;
}
location ~* ^.+.(jpg|jpeg|gif|css|png|js|ico|swf|flv|eot|ttf|woff|svg)$ {
       access_log        off;
       expires           30d;

}
location ~ "^/[^/]*\.php$" {
    set $script "index.php";
    if ( $uri ~ "^/(.*\.php)" ) {
       set $script $1;
    }
    fastcgi_pass   unix:/tmp/php-fpm.sock;
    fastcgi_index  index.php;
    fastcgi_param  SCRIPT_FILENAME $document_root$script;
    include        fastcgi_params;
}
}
 

Replace YOUR.INTERNET.HOSTNAME and YOUR_EZPUBLISH_ROOT as necessary.

You can also enter these settings in a file with any name in /etc/nginx/sites-available and create a symlink to it in /etc/nginx/sites-enabled if you want to run multiple vhosts. The principles are the same as under Apache 2.

Special thanks to Boris Huisgen for providing the basic rewrite rules for eZ Publish under Nginx.

 

Step 5: Reload PHP and Nginx

/etc/init.d/php5-fpm restart
/etc/init.d/nginx restart
 

Step 6: Install or move eZ Publish

Download and unpack the eZ Publish distribution to the chosen web root directory (YOUR_EZPUBLISH_ROOT) and install as normal. You can also move an existing installation, but it should be upgraded to eZ Publish 4.5 or 4.2011 Community Edition because of issues with fastcgi in earlier versions.

See http://doc.ez.no for instructions on how to install or upgrade eZ Publish.

 

Step 7: Update eZ Publish site settings

eZ Publish doesn't automatically recognise virtual host mode on Nginx as opposed to Apache.

When you are done installing/moving, in settings/override/site.ini.append.php, add the following under the [SiteAccessSettings] block:

ForceVirtualHost=true
 

Step 8: Flush caches

su - www-data
cd YOUR_EZ_PUBLISH_ROOT
php bin/php/ezcache.php --clear-all --purge<span> 
</span>
 

Conclusion

At this point you should have a working eZ Publish installation under Nginx, behaving exactly the same as under Apache but with even more speed and stability, especially under high-load situations. The next natural steps are of course to experiment with the settings to optimise them for your own environments, and carry out some stress testing.

This is just the tip of the iceberg of Nginx and server constellations, and in future tutorials I hope to provide examples of clustered setups, load balancing and proxying as well as load tests.

 

Resources

This tutorial is available in PDF format : eZ Publish 4.2011 Community (4.5) with Nginx and PHP-FPM on Debian 6 ("Squeeze") - PDF Version

 

About the author : Daniel Arnrup-Øien

Daniel Arnrup-Øien is Board Director and developer at Open Concept SA, a Norwegian cooperative corporation dedicated to providing complete open source web solutions with quality and performance. He has been developing in eZ Publish since 2005. He can be reached at daniel.oien@openconcept.no.

 

License choice

This work is licensed under the Creative Commons Attribution-ShareAlike (CC BY-SA) license.

Powered by eZ Publish™ CMS Open Source Web Content Management. Copyright © 1999-2014 eZ Systems AS (except where otherwise noted). All rights reserved.

eZ debug

Timing: Jan 17 2025 21:09:58
Script start
Timing: Jan 17 2025 21:09:58
Module start 'layout'
Timing: Jan 17 2025 21:09:58
Module start 'content'
Timing: Jan 17 2025 21:09:58
Module end 'content'
Timing: Jan 17 2025 21:09:58
Script end

Main resources:

Total runtime0.1441 sec
Peak memory usage4,096.0000 KB
Database Queries52

Timing points:

CheckpointStart (sec)Duration (sec)Memory at start (KB)Memory used (KB)
Script start 0.00000.0085 596.7422152.1875
Module start 'layout' 0.00850.0030 748.929739.5547
Module start 'content' 0.01150.1315 788.4844886.4141
Module end 'content' 0.14300.0012 1,674.898420.8281
Script end 0.1441  1,695.7266 

Time accumulators:

 Accumulator Duration (sec) Duration (%) Count Average (sec)
Ini load
Load cache0.00342.3382160.0002
Check MTime0.00130.9185160.0001
Mysql Total
Database connection0.00140.948210.0014
Mysqli_queries0.043530.1474520.0008
Looping result0.00040.2445500.0000
Template Total0.113478.720.0567
Template load0.00211.485320.0011
Template processing0.111277.164620.0556
Template load and register function0.00010.097810.0001
states
state_id_array0.00302.073950.0006
state_identifier_array0.00191.321460.0003
Override
Cache load0.00271.88981540.0000
Sytem overhead
Fetch class attribute name0.00130.876940.0003
Fetch class attribute can translate value0.00000.023530.0000
class_abstraction
Instantiating content class attribute0.00000.006140.0000
XML
Image XML parsing0.00241.679530.0008
General
dbfile0.00221.5149220.0001
String conversion0.00000.019540.0000
Note: percentages do not add up to 100% because some accumulators overlap

Templates used to render the page:

UsageRequested templateTemplateTemplate loadedEditOverride
1node/view/full.tplfull/article.tplextension/sevenx/design/simple/override/templates/full/article.tplEdit templateOverride template
1content/datatype/view/ezxmltext.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tplEdit templateOverride template
3content/datatype/view/ezxmltags/embed.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/embed.tplEdit templateOverride template
3content/view/embed.tplembed/image.tplextension/sevenx/design/simple/override/templates/embed/image.tplEdit templateOverride template
3content/datatype/view/ezimage.tpl<No override>extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tplEdit templateOverride template
14content/datatype/view/ezxmltags/header.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/header.tplEdit templateOverride template
46content/datatype/view/ezxmltags/paragraph.tpl<No override>extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tplEdit templateOverride template
6content/datatype/view/ezxmltags/link.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/link.tplEdit templateOverride template
20content/datatype/view/ezxmltags/separator.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezxmltags/separator.tplEdit templateOverride template
3content/datatype/view/ezxmltags/emphasize.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/emphasize.tplEdit templateOverride template
5content/datatype/view/ezxmltags/newpage.tpl<No override>extension/community/design/standard/templates/content/datatype/view/ezxmltags/newpage.tplEdit templateOverride template
12content/datatype/view/ezxmltags/strong.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/strong.tplEdit templateOverride template
13content/datatype/view/ezxmltags/literal.tpl<No override>extension/community/design/standard/templates/content/datatype/view/ezxmltags/literal.tplEdit templateOverride template
3content/datatype/view/ezxmltags/li.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/li.tplEdit templateOverride template
1content/datatype/view/ezxmltags/ul.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/ul.tplEdit templateOverride template
1content/datatype/view/ezxmltags/embed-inline.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/embed-inline.tplEdit templateOverride template
1content/view/embed-inline.tpl<No override>design/standard/templates/content/view/embed-inline.tplEdit templateOverride template
1print_pagelayout.tpl<No override>extension/community/design/community/templates/print_pagelayout.tplEdit templateOverride template
 Number of times templates used: 137
 Number of unique templates used: 18

Time used to render debug report: 0.0001 secs