Slide 1

Slide 1 text

Scaling(WordPress(using(W3TotalCache Thijs&Feryn Evangelist +32&(0)9&218&79&06 [email protected] WPUG&Belgium&UG&meeJng Wednesday&June&20th&2012 Gent,&Belgium

Slide 2

Slide 2 text

Evangelist

Slide 3

Slide 3 text

No content

Slide 4

Slide 4 text

No content

Slide 5

Slide 5 text

@ThijsFeryn

Slide 6

Slide 6 text

Please( rate(my(talk(on( Joind.in https://joind.in/6698

Slide 7

Slide 7 text

No content

Slide 8

Slide 8 text

No content

Slide 9

Slide 9 text

PHP(isn’t(*that*(fast

Slide 10

Slide 10 text

Lexing Source( to(tokens Tokenizing InterpreIng( tokens Compiling Generate( byte(code ExecuIon Running( byte(code

Slide 11

Slide 11 text

Overhead

Slide 12

Slide 12 text

External(resources

Slide 13

Slide 13 text

External(resources Database Webservice Feed

Slide 14

Slide 14 text

Typical(setup Linux Apache PHP(/(mod_php MySQL

Slide 15

Slide 15 text

What(about(W3(Total(Cache?

Slide 16

Slide 16 text

What(about(W3(Total(Cache? ✓Browser(cache ✓Minify ✓MySQL(cache ✓Object(cache ✓Page(cache ✓CDN ✓Varnish(+(purging ✓Cloudflare

Slide 17

Slide 17 text

APC

Slide 18

Slide 18 text

You(can(also(use(Memcached

Slide 19

Slide 19 text

You(can(also(use(Memcached memcached.sess_prefix = "memc.sess.key." session.save_path="127.0.0.1:11211" session.save_handler = memcached Also& for&session& handling

Slide 20

Slide 20 text

No content

Slide 21

Slide 21 text

Request vcl_recv In cache? vcl_hash Cacheable? vcl_hit() vcl_miss() vcl_deliver() vcl_fetch() No Yes No Yes Response

Slide 22

Slide 22 text

backend default { .host = "127.0.0.1"; .port = "8080"; } acl purge { "localhost"; "127.0.0.1"; "some.host.name.com"; "1.2.3.4"; }

Slide 23

Slide 23 text

sub vcl_recv { if (req.request == "PURGE") { if (!client.ip ~ purge) { error 405 "Not allowed: " + client.ip; } return(lookup); } if (req.request != "GET" && req.request != "HEAD" && req.request != "PUT" && req.request != "POST" && req.request != "TRACE" && req.request != "OPTIONS" && req.request != "DELETE") { return (pipe); }

Slide 24

Slide 24 text

if (req.request != "GET" && req.request != "HEAD") { return (pass); } if (req.http.Accept-Encoding) { if (req.url ~ "\.(gif|jpg|jpeg|swf|flv|mp3|mp4| pdf|ico|png|gz|tgz|bz2)(\?.*|)$") { remove req.http.Accept-Encoding; } else if (req.http.Accept-Encoding ~ "gzip") { set req.http.Accept-Encoding = "gzip"; } else if (req.http.Accept-Encoding ~ "deflate"){ set req.http.Accept-Encoding = "deflate"; } else { remove req.http.Accept-Encoding; } }

Slide 25

Slide 25 text

if (req.url ~ "\.(gif|jpg|jpeg|swf|css|js|flv|mp3| mp4|pdf|ico|png)(\?.*|)$") { unset req.http.cookie; set req.url = regsub(req.url, "\?.*$", ""); return (lookup); } if (req.url ~ "\?(utm_(campaign|medium|source|term)| adParams|client|cx|eid|fbid|feed|ref(id|src)?|v(er| iew))=") { set req.url = regsub(req.url, "\?.*$", ""); } if (req.http.cookie ~ "^ *$") { unset req.http.cookie; }

Slide 26

Slide 26 text

sub vcl_fetch { if (req.url ~ "wp-(login|admin)" || req.url ~ "preview=true" || req.url ~ "xmlrpc.php") { return (hit_for_pass); } if ( (!(req.url ~ "(wp-(login|admin)|login)"))) { unset beresp.http.set-cookie; set beresp.ttl = 1h; } if (req.url ~ "\.(gif|jpg|jpeg|swf|css|js|flv|mp3| mp4|pdf|ico|png)(\?.*|)$") { set beresp.ttl = 365d; } }

Slide 27

Slide 27 text

sub vcl_deliver { if (obj.hits > 0) { set resp.http.X-Cache = "HIT"; } else { set resp.http.X-Cache = "MISS"; } } sub vcl_hit { if (req.request == "PURGE") { purge; set obj.ttl = 0s; error 200 "OK"; } } sub vcl_miss { if (req.request == "PURGE") { purge; error 200 "OK"; } }

Slide 28

Slide 28 text

Linux Varnish Apache PHP(+(APC Memcached MySQL

Slide 29

Slide 29 text

FPM h\p://www.php.net/manual/en/install.fpm.configuraIon.php

Slide 30

Slide 30 text

[www] listen = 9000 user = www-data group = www-data pm = dynamic pm.max_children = 10 pm.start_servers = 4 pm.max_spare_servers = 6 pm.max_requests = 500 php_admin_value[sendmail_path] = /usr/sbin/sendmail -t -i -f [email protected] php_flag[display_errors] = off php_admin_value[error_log] = /var/log/fpm-php.www.log php_admin_flag[log_errors] = on php_admin_value[memory_limit] = 32M

Slide 31

Slide 31 text

No content

Slide 32

Slide 32 text

upstream fpm { server php1.server.com:9000; server php2.server.com:9000; } upstream memcached { server memcached1.server.com:11211; server memcached2.server.com:11211; } server { root /var/www; index index.php index.html index.htm; server_name nginx.server.com; location / { try_files $uri $uri/ /index.php; } location @php { fastcgi_pass fpm; fastcgi_index index.php; include fastcgi_params; }

Slide 33

Slide 33 text

location ~ \.php$ { set $memcached_key $request_uri; memcached_pass memcached; memcached_next_upstream not_found; default_type text/html; error_page 404 405 502 = @php; } location ~* \.(?:ico|css|js|gif|jpe?g|png)$ { expires max; add_header Pragma public; add_header Cache-Control "public, must- revalidate, proxy-revalidate"; } }

Slide 34

Slide 34 text

location ~ \.php$ { set $memcached_key $request_uri; memcached_pass memcached; memcached_next_upstream not_found; default_type text/html; error_page 404 405 502 = @php; } location ~* \.(?:ico|css|js|gif|jpe?g|png)$ { expires max; add_header Pragma public; add_header Cache-Control "public, must- revalidate, proxy-revalidate"; } }

Slide 35

Slide 35 text

Linux Varnish Nginx PHPFPM(+(APC Memcached MySQL

Slide 36

Slide 36 text

No content

Slide 37

Slide 37 text

Content Delivery Network

Slide 38

Slide 38 text

Linux Varnish Nginx PHPFPM(+(APC Memcached MySQL CDN

Slide 39

Slide 39 text

Show(Ime(!

Slide 40

Slide 40 text

Thanks(!

Slide 41

Slide 41 text

No content