When developing templates in an eZ Publish project, use of fetch functions is a recurring task. You will of course mainly fetch content nodes/objects, or maybe infos about current user :
{def $myNode = fetch( 'content', 'node', hash( 'node_id', 2 ) ) $userIsLoggedIn = fetch( 'user', 'current_user' )}
But sometimes you need to fetch pieces of information that are not covered by built-in functions (consume an external webservice, rows from non-native tables in your database, ...). Fortunately, fetch mechanism is extensible and it is fairly easy to add a new custom fetch function. Let's see how to do that...
This tutorial is targeted to intermediate eZ Publish developers or beginners who has already understood template language and eZ Publish directory structure.
Basic PHP programming skills are also needed in order to understand fetch mechanism. MVC design pattern understanding can also be useful but is not required.
When working your eZ project, you may ask yourself a few questions :
Well, let's try to answer those vital questions ;-).
Well, that's right, the difference is thin as both can suit you in many situations. Here are significant differences :
Basically, you will need to develop a fetch function when you need to look for some data in your database, on your file system, in a partner's webservice... This is the main role for this feature, but not only. Indeed, as I said above, with fetch functions you can drastically reduce template code complexity.
Did you ever get bored of maintaining template code ? Have you ever dreamt of breakpoints driven debug instead of crappy attribute( 'show' ) ? Well, as a fetch function is written in PHP, you can transfer all your complex template code into a real function, get rid of template language limitations and take fully advantage of the real eZ Publish framework power. And of course, your fetch result can fully be exploited in your templates, so you can aggregate your data into an array, or even into PHP objects (but for that, you will need to implement some methods in your returned object's class; see the appendix for more explanations).
A fetch function is actually part of a module and is mainly composed of 2 files :
You can get many examples in modules from eZ Publish kernel, take a look into kernel/content/ folder for example (please note that every folders located under kernel/, except classes/ and common/, contain a module).
Basically, this file (function_definition.php) holds an associative array called $FunctionList. Keys of this array are fetch functions names and values are the definition as another associative array :
<?php // function_definition.php // Fetch definition file $FunctionList = array(); $FunctionList['fetch_function_name'] = array( 'name' => 'fetch_function_name', // Retype the name of your function 'operation_types' => 'read', // Obsolete, for BC 'call_method' => array( 'class' => 'MyModuleFunctionCollection', // Function collection class 'method' => 'getMyFetchFunctionResult' ), // Method to call 'parameter_type' => 'standard', // Obsolete, for BC 'parameters' => array( // Function parameters in the PHP function order array( 'name' => 'first_param', 'type' => 'string', 'required' => true ), array( 'name' => 'second_param', 'type' => 'boolean', 'required' => false, 'default' => false ) ) );
Above is a typical fetch definition file. You will find similar ones in the kernel or in extensions from the community (take a look into the SQLIGeoloc one).
As I said before, your PHP fetch function will be contained in a so called function collection, which is a simple PHP class with static methods. Fetch code needs to be written in one of those static methods.
The main point to remember about these methods is the argument order. It must be strictly the same than declared in the definition file. Thus, in the example above, first_param will be mapped to the method's first argument, and second_param to the second.
Each method must return an associative array, with result as key and the value to be returned as value. If an error occurs, then it must return an associative array, with error as key and the error message as value.
<?php /** * MyModule fetch functions */ class MyModuleFunctionCollection { /** * Fetch function code. * You can do everything you want here. * Just return an associative array with 'result' as key and your result as value. * If an error is raised, return an associative array with 'error' as key and the * error message as value * @param string $myFirstParam first_param declared in function_definition.php * @param string $mySecondParam second_param declared in function_definition.php */ public static function getMyFetchFunctionResult( $myFirstParam, $mySecondParam = false ) { try { /* * Do everything you want here to fetch/aggregate your data * Associative arrays can be used in templates : * {def $result = fetch( 'mymodule', 'fetch_function_name', hash( * 'first_param', 'something', * 'second_param', true() * ) )} * {$result.first_key} {* Will display "foo" *} * {$result.second_key} {* Will display "bar" *} */ $result = array( 'first_key' => 'foo', 'second_key' => 'bar' ); return array( 'result' => $result ); } catch( Exception $e ) { $errMsg = $e->getMessage(); eZDebug::writeError( $errMsg ); return array( 'error' => $errMsg ); } } }
As fetch functions must be part of a module, the only requirement is to have a declared module inside of an active extension.
The function_definition.php file will reside inside the module folder, but the function collection class doesn't need to be in it since it will be automatically loaded thanks to the built-in class autoload system. Best practices tell us to create a classes/ folder inside the extension and place our class here.
So called mymodule module need to be declared in extension/myextension/settings/module.ini.append.php :
<?php /* #?ini charset="utf-8"? [ModuleSettings] ExtensionRepositories[]=myextension ModuleList[]=mymodule */ ?>
And of course, be sure that your extension is activated in settings/override/site.ini.append.php (or in your siteaccess with ActiveAccessExtensions)
Et voilà ! You can now use your fetch function in templates :
{def $result = fetch( 'mymodule', 'fetch_function_name', hash( 'first_param', 'something', 'second_param', true() ) )} {$result.first_key} {* Will display "foo" *} {$result.second_key} {* Will display "bar" *}
One of the main advantages of fetch functions vs template operators is that they can easily be called in your PHP scripts. A handler class, eZFunctionHandler, with two shorthand methods are available for that :
<?php $myResult = eZFunctionHandler::execute( 'mymodule', 'fetch_function_name', array( 'first_param' => 'something_else', 'second_param' => true ) ); echo $myResult['first_key']; // Will display "foo" echo $myResult['second_key']; // Will display "bar"
Here is a quick recap :
Please remember that templates should remain simple. As soon as you need to add too much complexity, consider developing a fetch function; you will be hands free and your code will be much easier to maintain.
Also note that you are not forced to write all code in the same PHP static method ! You can delegate part of your code into dedicated classes (Model in MVC pattern).
This tutorial is available in PDF format for offline reading :
Jérôme Vieilledent - Understanding and developing fetch functions
Jérôme is a completely self-educated web developer. He learnt PHP all by himself and started developing with eZ Publish in 2007. He is eZ Publish certified and now works as a technical manager and expert at SQLi Agency in Paris.
This work is licensed under the Creative Commons – Share Alike license ( http://creativecommons.org/licenses/by-sa/3.0 ).
While developing templates, you might have noticed that you can access most of every kernel objects attributes directly in templates :
{* $myNode will hold an eZContentObjectTreeNode object *} {def $myNode = fetch( 'content', 'node', hash( 'node_id', 2 ) )} {* Directly access to $myNode attributes *} NodeID : {$myNode.node_id}<br /> Name : {$myNode.name} {* Inspect $myNode *} {$myNode|attribute( 'show', 2 )}
In order to be able to use a PHP object in templates such as eZContentObjectTreeNode in the example above, you will need to implement 3 methods :
So, when you need to access an object attribute, eZ Publish first checks if it does exist via hasAttribute() method. If so, then it will use attribute() accessor to retrieve the value. Last attributes() method is only used when inspecting the object with attribute template operator.
Example :
<?php class MyExampleClass { /** * Attribute holder * @var array */ private $attributes; public function __construct() { $this->attributes = array( 'first_attribute' => 'foo', 'another_attribute' => 'bar' ); } /** * Checks if $attributeName is a valid attribute * @return bool */ public function hasAttribute( $attributeName ) { return isset( $this->attributes[$attributeName] ); } /** * Attribute accessor * @return mixed */ public function attribute( $attributeName ) { return $this->attributes[$attributeName]; } /** * Returns all available attribute names * @return array */ public function attributes() { return array_keys( $this->attributes ); } }
Template code :
{* Assume that $myObject is a MyExampleClass object *} {$myObject.first_attribute}{* Will display 'foo' *}<br /> {$myObject.another_attribute}{* Will display 'bar' *} {* Inspect $myNode *} {$myNode|attribute( 'show', 2 )}
Timing: | Jan 17 2025 22:52:54 |
Script start | |
Timing: | Jan 17 2025 22:52:54 |
Module start 'layout' | |
Timing: | Jan 17 2025 22:52:54 |
Module start 'content' | |
Timing: | Jan 17 2025 22:52:54 |
Module end 'content' | |
Timing: | Jan 17 2025 22:52:54 |
Script end |
Total runtime | 0.0146 sec |
Peak memory usage | 2,048.0000 KB |
Database Queries | 3 |
Checkpoint | Start (sec) | Duration (sec) | Memory at start (KB) | Memory used (KB) |
---|---|---|---|---|
Script start | 0.0000 | 0.0053 | 588.0469 | 152.6406 |
Module start 'layout' | 0.0053 | 0.0028 | 740.6875 | 39.4844 |
Module start 'content' | 0.0081 | 0.0045 | 780.1719 | 107.5469 |
Module end 'content' | 0.0125 | 0.0020 | 887.7188 | 46.2891 |
Script end | 0.0145 | 934.0078 |
Accumulator | Duration (sec) | Duration (%) | Count | Average (sec) |
---|---|---|---|---|
Ini load | ||||
Load cache | 0.0026 | 17.9828 | 14 | 0.0002 |
Check MTime | 0.0011 | 7.4163 | 14 | 0.0001 |
Mysql Total | ||||
Database connection | 0.0009 | 6.0030 | 1 | 0.0009 |
Mysqli_queries | 0.0023 | 16.0564 | 3 | 0.0008 |
Looping result | 0.0000 | 0.1029 | 1 | 0.0000 |
Template Total | 0.0015 | 10.6 | 1 | 0.0015 |
Template load | 0.0008 | 5.5912 | 1 | 0.0008 |
Template processing | 0.0007 | 4.9540 | 1 | 0.0007 |
Override | ||||
Cache load | 0.0005 | 3.6730 | 1 | 0.0005 |
General | ||||
dbfile | 0.0002 | 1.7025 | 8 | 0.0000 |
String conversion | 0.0000 | 0.0703 | 4 | 0.0000 |
Note: percentages do not add up to 100% because some accumulators overlap |
Usage | Requested template | Template | Template loaded | Edit | Override |
---|---|---|---|---|---|
1 | print_pagelayout.tpl | <No override> | extension/community/design/community/templates/print_pagelayout.tpl | ||
Number of times templates used: 1 Number of unique templates used: 1 |
Time used to render debug report: 0.0001 secs