As you might know, eZ Publish user access control is pretty precise and has a very fine granularity. Most of kernel modules allow indeed to limit access to themselves thanks to security policies we can assign to a user or to a user group. This is particularly the case for content module, fundamental within the CMS, thanks to its function limitations that can be configured in the admin interface. With these limitations, one can precisely define what a contributor has the right to do regarding content and functionalities.
This tutorial is targeted to anyone who needs to implement security policy limitations for custom modules. Knowledge of eZ Publish module development is a must to fully understand the interest of this article. If you are not familiar to module and/or extension development ineZ Publish, you may read this excellent article about developing extensions. You also might read this interesting (old) tutorial about module development for eZ Publish (warning : this article was based on eZ Publish 3.x and PHP4, so read it with PHP5 style in mind). You can brush-up your knowledge of the basics of Access Control in eZ Publish by reviewing the online documentation on this subject.
It is obviously possible to define security rules, configurable in the same way, for our own modules. All we need to do is to create a list of functions for our module, in module.php.
extension/myextension/modules/mymodule/module.php :
<?php $Module = array('name' => 'mymodule'); $ViewList = array(); $ViewList['myview'] = array( 'script' => 'view.php', 'params' => array(), 'functions' => array( 'myfunction' ) ); $FunctionList['myfunction'] = array();
Here we define a function for the module mymodule and we affect it to the view myview. This function will display in the corresponding list when creating a new security policy for module mymodule in the admin interface. You will find this kind of definition in most of modules contained in the kernel, or within extensions developped by eZ Systems or by the community (this is the case for NovenINIUpdate extension).
What I described above is sufficient in most cases, but what if we need to add more complex limitations like in content module (language, section, etc...) ?
As you probably noticed in the module.php example above, variable $FunctionList['myfunction'] is an empty array, which means that the function doesn't have any limitation. We just need to fill it with right values to add some.
Example for a language level limitation :
<?php $Module = array('name' => 'mymodule'); $ViewList = array(); $ViewList['myview'] = array( 'script' => 'view.php', 'params' => array(), 'functions' => array( 'myfunction' ) ); $Language = array( 'name'=> 'Language', 'values'=> array(), 'path' => 'classes/', 'file' => 'ezcontentlanguage.php', 'class' => 'eZContentLanguage', 'function' => 'fetchLimitationList', 'parameter' => array( false ) ); $FunctionList['myfunction'] = array('Language' => $Language);
Here we tell eZ Publish that myfunction has a limitation named Language and that we need to fetch possible values list with eZContentLanguage::fetchLimitationList() method, with parameter false. This simply stands for an old school way to define a callback function, probably here since first releases of eZ Publish 3 (remind you first versions of PHP4 and all its limitations). For curious minds, the important stuffs can be found in eZPolicyLimitation::allValuesAsArrayWithNames(), from line 249 (in a 4.3.0 eZ Publish instance).
Here we call a class from the kernel to fill our limitation array, but it is of course possible to use a class from an extension. However, autoload system cannot be used here and an old good include_once is made in the backend. So, in order to use a class from an extension, we need to add another key to our $Language array :
$Language = array( 'name' => 'Language', 'values' => array(), 'extension' => 'myextension', 'path' => 'classes/', 'file' => 'myclass.php', 'class' => 'MyClass', 'function' => 'fetchLanguageLimitationList', 'parameter' => array( false ) );
This way, extension/myextension/classes/myclass.php class will be included.
What does our method MyClass:fetchLanguageLimitationList() have to return ? A quick look to eZContentLanguage::fetchLimitationList() shows us that it must be a simple array with each entry being an associative array itself containing id and name keys.
Here what should be our class MyClass :
class MyClass { /** * Fetches the array with names and IDs of the languages used on the site. This method is used by the permission system. * * @return Array with names and IDs of the languages used on the site. * @static */ public static function fetchLanguageLimitationList() { $langList = eZINI::instance( 'site.ini' )->variable( 'RegionalSettings', 'SiteLanguageList' ); $aResult = array(); foreach($langList as $lang) { if($lang) { $aResult[] = array( 'id' => $lang, 'name' => $lang ); } } return $aResult; } }
Here is below a series of screenshots showing all the process :
Now that our limitations are defined, we now need to filter access to our module depending on the rights that have been affected to a user. Indeed, this control is made in index.php for content module, and only for this one. As a consequence, we are forced to control the access ourselves. We can imagine changes in this regard in the future versions of eZ Publish.
Manual access control of those limitations using the framework can easily become a real pain as the system is pretty complex. While a little refactoring would be nice on this part, workarounds exist that will make your life easier! Developer Andrè Rømcke have implemented in eZJSCore extension a simplified access control method in the shape of a template operator. This extension being now part of eZ Publish distribution, it would be a shame not to use it ! Besides, you can find a real good article presenting this extension on the community portal.
Here is the best way to proceed :
extension/myextension/modules/mymodule/myview.php :
$userHasAccess = ezjscAccessTemplateFunctions::hasAccessToLimitation( 'mymodule', 'myfunction' ); // Returns a boolean for current user // Or if you want to check using limitations as well using ezjscore 1.2 (comes with eZ Publish 4.4) and up // In this case providing list of languages user must have access to $userHasAccess = ezjscAccessTemplateFunctions::hasAccessToLimitation( 'mymodule', 'myfunction', array( 'Language' => $languageList ) );
Unfortunately, older versions of eZJSCore then 1.1.1 / 1.2 doesn't have any method allowing us to get available limitations for current user, which could be very useful to display a combo box containing limitations granted to the user for example (available languages in our case).
To do this, we will need to write a method returning those limitations in a simplified way. Indeed, eZUser class does have hasAccessTo() method, but its result is absolutely unreadable and needs to be strongly simplified. We will thus write a complementary method returning simplified limitations.
class MyClass { /** * Shorthand method to check user access policy limitations for a given module/policy function. * Returns the same array as eZUser::hasAccessTo(), with "simplifiedLimitations". * 'simplifiedLimitations' array holds all the limitations names as defined in module.php. * If your limitation name is not defined as a key, then your user has full access to this limitation * @param string $module Name of the module * @param string $function Name of the policy function ($FunctionList element in module.php) * @return array */ public static function getSimplifiedUserAccess( $module, $function ) { $user = eZUser::currentUser(); $userAccess = $user->hasAccessTo( $module, $function ); $userAccess['simplifiedLimitations'] = array(); if( $userAccess['accessWord'] == 'limited' ) { foreach( $userAccess['policies'] as $policy ) { foreach( $policy as $limitationName => $limitationList ) { foreach( $limitationList as $limitationValue ) { $userAccess['simplifiedLimitations'][$limitationName][] = $limitationValue; } $userAccess['simplifiedLimitations'][$limitationName] = array_unique($userAccess['simplifiedLimitations'][$limitationName]); } } } return $userAccess; } }
This method returns an array containing result from eZUser::hasAccessTo(), with a new key : simplifiedLimitations. This key is also an array, containing the precious limitations.
In our example, for a user whom we would have affected a limitation Language allowing only fre-FR and eng-GB, this array would contain :
$limitations = MyClass::getSimplifiedUserAccess( 'mymodule', 'myfunction' ); print_r( $limitations['simplifiedLimitations'] ); // Result Array ( [Language] => Array ( [0] => fre-FR [1] => eng-GB ) )
As a result, in our module where we need to display a combo box with authorized languages, we would have :
extension/myextension/modules/mymodule/myview.php :
$tpl = eZTemplate::factory(); // Template init – from 4.3.0 $authorizedLang = eZINI::instance('site.ini')->variable( 'RegionalSettings', 'SiteLanguageList' ); // Default is all languages $limitations = MyClass::getSimplifiedUserAccess( 'mymodule', 'myfunction' ); if( isset( $limitations['simplifiedLimitations']['Language'] ) ) // Found limitations on language. These will be the only available in the dropdown menu $authorizedLang = $limitations['simplifiedLimitations']['Language']; $tpl->setVariable( 'languages', $authorizedLang ); $Result['content'] = $tpl->fetch( 'design:mydesignsubdir/myview.tpl' );
extension/myextension/design/standard/templates/mydesignsubdir/myview.tpl :
<select name="LanguageSelection"> {foreach $languages as $language} <option value=”{$language}”>{$language}</option> {/foreach} </select>
eZ Publish 4.3.0 introduced the ability to filter the display of tabs in the admin interface depending on users security policies. Here is how to proceed :
extension/myextension/settings/menu.ini.append.php :
<?php /* #?ini charset="utf-8"? [NavigationPart] Part[mynavigationpart]=My NavigationPart description [TopAdminMenu] Tabs[]=mytab [Topmenu_mytab] NavigationPartIdentifier= mynavigationpart Name=My Tab Tooltip=My Tooltip URL[] URL[default]=mymodule/myview Enabled[] Enabled[default]=true Enabled[browse]=false Enabled[edit]=false Shown[] Shown[default]=true Shown[navigation]=true Shown[browse]=false # Simply add access control to your module's default function PolicyList[]=mymodule/function */ ?>
Enjoy !
Here is a quick recap of each step :
As an example, we could consider NovenINIUpdate extension (environment switch for INI settings). It has indeed a GUI in the admin interface. At the time of writing, every user that has access to the main module of this extension can switch to every declared environment (ie. dev, staging, production). We could add a limitation on the environment, allowing many users to only switch from/to dev and staging environment. Production switch would be given only to admin users.
In short, building custom limitations for security policies attached to custom modules in eZ Publish is not a piece of cake. Personally, it took me several days of patience, deeply digging into the kernel to understand those mechanisms I just exposed to you, so you do not need to search for hours yourself :)
However, despite the real complexity, this mechanism allows very fine grain granularity in user access control. It is also unique and native in eZ Publish for a long time now.To be widely used, it would need to be simplified within the kernel, as it has been done thanks to eZJSCore and the helper method we used in this tutorial.
I particularly would like to thank Nicolas Pastorino, Damien Pobel and Andrè Rømcke who helped me during my research :-).
This tutorial is an english translation of my original post on my blog, Lolart.net (french) : http://www.lolart.net/blog/ez-publish/des-limitations-pour-vos-politiques-de-securite
This article is available in PDF for offline reading : Jerome Vieilledent - Adding custom security policy limitations to your modules - eZ Publish Community
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 works now 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 ).
Timing: | Jan 17 2025 23:55:22 |
Script start | |
Timing: | Jan 17 2025 23:55:22 |
Module start 'layout' | |
Timing: | Jan 17 2025 23:55:22 |
Module start 'content' | |
Timing: | Jan 17 2025 23:55:22 |
Module end 'content' | |
Timing: | Jan 17 2025 23:55:22 |
Script end |
Total runtime | 0.2989 sec |
Peak memory usage | 4,096.0000 KB |
Database Queries | 71 |
Checkpoint | Start (sec) | Duration (sec) | Memory at start (KB) | Memory used (KB) |
---|---|---|---|---|
Script start | 0.0000 | 0.0063 | 591.8203 | 152.6563 |
Module start 'layout' | 0.0063 | 0.0022 | 744.4766 | 39.5156 |
Module start 'content' | 0.0086 | 0.2882 | 783.9922 | 961.3047 |
Module end 'content' | 0.2968 | 0.0020 | 1,745.2969 | 32.2500 |
Script end | 0.2988 | 1,777.5469 |
Accumulator | Duration (sec) | Duration (%) | Count | Average (sec) |
---|---|---|---|---|
Ini load | ||||
Load cache | 0.0033 | 1.1152 | 16 | 0.0002 |
Check MTime | 0.0016 | 0.5211 | 16 | 0.0001 |
Mysql Total | ||||
Database connection | 0.0008 | 0.2723 | 1 | 0.0008 |
Mysqli_queries | 0.0798 | 26.6949 | 71 | 0.0011 |
Looping result | 0.0006 | 0.2053 | 69 | 0.0000 |
Template Total | 0.2650 | 88.7 | 2 | 0.1325 |
Template load | 0.0031 | 1.0503 | 2 | 0.0016 |
Template processing | 0.2618 | 87.5998 | 2 | 0.1309 |
Template load and register function | 0.0001 | 0.0375 | 1 | 0.0001 |
states | ||||
state_id_array | 0.0063 | 2.1220 | 10 | 0.0006 |
state_identifier_array | 0.0052 | 1.7376 | 11 | 0.0005 |
Override | ||||
Cache load | 0.0153 | 5.1038 | 226 | 0.0001 |
Sytem overhead | ||||
Fetch class attribute name | 0.0032 | 1.0568 | 17 | 0.0002 |
Fetch class attribute can translate value | 0.0001 | 0.0307 | 8 | 0.0000 |
class_abstraction | ||||
Instantiating content class attribute | 0.0000 | 0.0100 | 17 | 0.0000 |
XML | ||||
Image XML parsing | 0.0400 | 13.3825 | 8 | 0.0050 |
General | ||||
dbfile | 0.0422 | 14.1239 | 40 | 0.0011 |
String conversion | 0.0000 | 0.0034 | 4 | 0.0000 |
Note: percentages do not add up to 100% because some accumulators overlap |
Usage | Requested template | Template | Template loaded | Edit | Override |
---|---|---|---|---|---|
1 | node/view/full.tpl | full/article.tpl | extension/sevenx/design/simple/override/templates/full/article.tpl | ||
9 | content/datatype/view/ezxmltext.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tpl | ||
15 | content/datatype/view/ezxmltags/header.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/header.tpl | ||
22 | content/datatype/view/ezxmltags/emphasize.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/emphasize.tpl | ||
13 | content/datatype/view/ezxmltags/strong.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/strong.tpl | ||
50 | content/datatype/view/ezxmltags/paragraph.tpl | <No override> | extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tpl | ||
13 | content/datatype/view/ezxmltags/link.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/link.tpl | ||
11 | content/datatype/view/ezxmltags/separator.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltags/separator.tpl | ||
7 | content/datatype/view/ezxmltags/newpage.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/newpage.tpl | ||
10 | content/datatype/view/ezxmltags/literal.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/literal.tpl | ||
10 | content/datatype/view/ezxmltags/li.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/li.tpl | ||
2 | content/datatype/view/ezxmltags/ul.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/ul.tpl | ||
8 | content/datatype/view/ezxmltags/embed.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed.tpl | ||
8 | content/view/embed.tpl | embed/image.tpl | extension/sevenx/design/simple/override/templates/embed/image.tpl | ||
8 | content/datatype/view/ezimage.tpl | <No override> | extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tpl | ||
1 | content/datatype/view/ezxmltags/ol.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/ol.tpl | ||
1 | content/datatype/view/ezxmltags/line.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/line.tpl | ||
1 | content/datatype/view/ezxmltags/embed-inline.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed-inline.tpl | ||
1 | content/view/embed-inline.tpl | <No override> | design/standard/templates/content/view/embed-inline.tpl | ||
1 | print_pagelayout.tpl | <No override> | extension/community/design/community/templates/print_pagelayout.tpl | ||
Number of times templates used: 192 Number of unique templates used: 20 |
Time used to render debug report: 0.0002 secs