Blogs / Marko Zmak / When ezsi doesn't do it

When ezsi doesn't do it

Wednesday 09 February 2011 9:05:02 am

  • Currently 4 out of 5 Stars.
  • 1
  • 2
  • 3
  • 4
  • 5

By : Marko Žmak

The ezsi extension is a pretty good one, but there are situations where it doesn't do the job. It lacks a "fragmented" caching mechanism.

Last few days I have dealt with performance optimization of complex site with a lot of content, lots of visitors and a very high refresh rate. The site is using standard eZP caching mechanisms, view cache, cache blocks and static cache, but as the pageviews started to increase it needed more... As a first choice I tried the ezsi extension but it just couldn't do the job...

The problem is that there are pages (for example the frontpage) that have a lot of heavy parts that have to be refreshed very often, and all at the same time. This heavy parts have a complex template logic and produce heavy SQL queries. And since they all have to be refreshed very often and at the same time, the complete processing of the page tends to become very demanding.

So when the server is under heavy load, it happens that the execution of the queries often fails and sometimes the complete eZP generation of the heavy page also fails. This ends up in parts of page missing or in outputting a fatal error. So I started to look for a solution that would allow me to execute and cache this heavy parts separately.

I have investigated the possibility of solving this problem with ezsi extension combined with Apache SSI, but I came to the conclusion that it doesn't work for this case.

The problem with ezsi is that it doesn't give you the possibility of "fragmented" cache generation. When the ezsi cronjob does the si-blocks regeneration it actually calls the URL of the page where the block was generated. So the whole page is processed and if all the blocks have the same TTL (which is needed in may case), all of them are re-executed which leads back to the original problem and doesn't help at all.

The solution I came up with is:

  • create a special module that allows you to execute only a part of your code (i.e. a specific template) by calling a specific URL
  • put the URL-s for the most heavy parts of the code in static cache
  • include the static files of this heavy parts in your page by using SSI or ESI includes

So I did it... I've created a module that allows you to include a chosen template based of the view parameters, and without any additional processing of nodes etc. And combined with an empty page layout this gives us the possibility to execute specific parts of the code via special URLs.

So, for example an URL like this

  • sa_include/part/top_content/(period)/2_days

can be setup (via an eZ template) to output the list of top rated content within last 2 days.

Then all I had to do is put URLs like this in the static cache list, put the appropriate SSI includes in my templates, and that's it. This pretty simple idea did the trick.

Not to be misunderstood, I'm not saying that ezsi extension doesn't do a good job, but it doesn't yet have the solution for the problem I described. In fact, I would be very glad if a mechanism like the one in my solution, gets implemented in ezsi.

And here's a proposal for the implementation:

  • create a template operator (for example ezsi_include) that accepts this parameters:
    • template name
    • template parameters (only simple types allowed)
    • TTL
  • when the ezsi_include operator is executed it does the following:
    • execute the given template with the given parameters
    • save the output of the template execution in a static HTML file (besides the usual ezsi keys the filename generation should also take into account the template parameters)
    • save template name, value of template parameters and TTL in an ezsi db table
  • in the block regeneration cronjob when an expired block created via the ezsi_include operator is encountered, execute the given template just like in step 2

This is just a basic idea and is probably not bulletproof, but I believe it's worth exploring. Any feedback, opinions and counter-ideas are appreciated.

Blog Post Discussion

When ezsi doesn't do it

eZ debug

Timing: Jan 17 2025 21:18:10
Script start
Timing: Jan 17 2025 21:18:10
Module start 'content'
Timing: Jan 17 2025 21:18:10
Module end 'content'
Timing: Jan 17 2025 21:18:10
Script end

Main resources:

Total runtime0.1948 sec
Peak memory usage6,144.0000 KB
Database Queries186

Timing points:

CheckpointStart (sec)Duration (sec)Memory at start (KB)Memory used (KB)
Script start 0.00000.0059 597.4688180.8516
Module start 'content' 0.00590.0709 778.3203453.1250
Module end 'content' 0.07680.1179 1,231.4453412.8203
Script end 0.1947  1,644.2656 

Time accumulators:

 Accumulator Duration (sec) Duration (%) Count Average (sec)
Ini load
Load cache0.00371.9028210.0002
Check MTime0.00140.7361210.0001
Mysql Total
Database connection0.00080.430710.0008
Mysqli_queries0.118360.75081860.0006
Looping result0.00160.83561840.0000
Template Total0.171988.320.0860
Template load0.00180.908220.0009
Template processing0.170187.339020.0851
Template load and register function0.00010.039410.0001
states
state_id_array0.00070.343910.0007
state_identifier_array0.00060.315820.0003
Override
Cache load0.00150.7638390.0000
Sytem overhead
Fetch class attribute name0.00160.845030.0005
Fetch class attribute can translate value0.00080.389110.0008
class_abstraction
Instantiating content class attribute0.00000.004830.0000
XML
Image XML parsing0.00030.148310.0003
General
dbfile0.00190.9915220.0001
String conversion0.00000.002730.0000
Note: percentages do not add up to 100% because some accumulators overlap

CSS/JS files loaded with "ezjscPacker" during request:

CacheTypePacklevelSourceFiles
CSS0extension/community/design/community/stylesheets/ext/jquery.autocomplete.css
extension/community_design/design/suncana/stylesheets/scrollbars.css
extension/community_design/design/suncana/stylesheets/tabs.css
extension/community_design/design/suncana/stylesheets/roadmap.css
extension/community_design/design/suncana/stylesheets/content.css
extension/community_design/design/suncana/stylesheets/star-rating.css
extension/community_design/design/suncana/stylesheets/syntax_and_custom_tags.css
extension/community_design/design/suncana/stylesheets/buttons.css
extension/community_design/design/suncana/stylesheets/tweetbox.css
extension/community_design/design/suncana/stylesheets/jquery.fancybox-1.3.4.css
extension/bcsmoothgallery/design/standard/stylesheets/magnific-popup.css
extension/sevenx/design/simple/stylesheets/star_rating.css
extension/sevenx/design/simple/stylesheets/libs/fontawesome/css/all.min.css
extension/sevenx/design/simple/stylesheets/main.v02.css
extension/sevenx/design/simple/stylesheets/main.v02.res.css
JS0extension/ezjscore/design/standard/lib/yui/3.17.2/build/yui/yui-min.js
extension/ezjscore/design/standard/javascript/jquery-3.7.0.min.js
extension/community_design/design/suncana/javascript/jquery.ui.core.min.js
extension/community_design/design/suncana/javascript/jquery.ui.widget.min.js
extension/community_design/design/suncana/javascript/jquery.easing.1.3.js
extension/community_design/design/suncana/javascript/jquery.ui.tabs.js
extension/community_design/design/suncana/javascript/jquery.hoverIntent.min.js
extension/community_design/design/suncana/javascript/jquery.popmenu.js
extension/community_design/design/suncana/javascript/jScrollPane.js
extension/community_design/design/suncana/javascript/jquery.mousewheel.js
extension/community_design/design/suncana/javascript/jquery.cycle.all.js
extension/sevenx/design/simple/javascript/jquery.scrollTo.js
extension/community_design/design/suncana/javascript/jquery.cookie.js
extension/community_design/design/suncana/javascript/ezstarrating_jquery.js
extension/community_design/design/suncana/javascript/jquery.initboxes.js
extension/community_design/design/suncana/javascript/app.js
extension/community_design/design/suncana/javascript/twitterwidget.js
extension/community_design/design/suncana/javascript/community.js
extension/community_design/design/suncana/javascript/roadmap.js
extension/community_design/design/suncana/javascript/ez.js
extension/community_design/design/suncana/javascript/ezshareevents.js
extension/sevenx/design/simple/javascript/main.js

Templates used to render the page:

UsageRequested templateTemplateTemplate loadedEditOverride
1node/view/full.tplblog_entry/full.tplextension/community_design/design/suncana/override/templates/blog_entry/full.tplEdit templateOverride template
2content/datatype/view/ezxmltext.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tplEdit templateOverride template
2content/datatype/view/ezxmltags/link.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/link.tplEdit templateOverride template
6content/datatype/view/ezxmltags/paragraph.tpl<No override>extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tplEdit templateOverride template
5content/datatype/view/ezxmltags/li.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/li.tplEdit templateOverride template
5content/datatype/view/ezxmltags/ul.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/ul.tplEdit templateOverride template
1content/datatype/view/ezkeyword.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezkeyword.tplEdit templateOverride template
1pagelayout.tpl<No override>extension/sevenx/design/simple/templates/pagelayout.tplEdit templateOverride template
 Number of times templates used: 23
 Number of unique templates used: 8

Time used to render debug report: 0.0001 secs