For a previous article I covered the various ways you can export user data using eZ Publish. In the article, I suggested how you can export all users who have logged in over the previous month using an extended attribute filter. In this article I will cover creating several different Extended Attribute Filters so you can see how they may be implemented in your code.
We’ll start by creating an extension to house our filters followed by building a basic filter to return a list of users who have forgotten their passwords. We will then create a filter to tell us how many/which users have logged in over the past month. We’ll also cover how you can add Extended Attribute Filters to both your template files and your PHP scripts.
All fetch statements in eZ Publish construct SQL statements to query the database for relevant objects. Extended attribute filters allow you to extend the built in functionality by adding completely custom clauses into the SQL statements. This is especially useful to tie in your fetch statements to other tables in the CMS, both custom tables and those not usually referenced in Fetch statements. It is also useful to query the ezcontentobject_attribute table in ways not possible with standard Fetch statements.
The Extended Attribute Filters code is available on Github for this tutorial. This and content for all future tutorials will exist in the http://github.com/davidlinnard/eztutorials repository (all content for this tutorial can be found on this commit ).
We’ll start by ensuring all of the Fetch statements we are using are showing their underlying SQL queries in the debug at the bottom of the screen. If you are comfortable doing this, and you are confortable enough creating your own extensions then I would recommend jumping to the next section .
There is a couple of key settings to change, all within your main site.ini file. We firstly need to make sure our debug is enabled for the site and we then need to make sure that all the SQL statements used to construct the page are shown in it. The following code will do this for us:
Now that the changes have been made, let’s create our extension. Go into your source code and under the extensions directory create a folder with the name you want to give your extension, I’m going to call mine onequarterenglish so rename yours accordingly in the next step (I’ll show the name of the extension in bold whenever it is used).
We now need to activate the extension, this will cause eZ Publish to look into the onequarterenglish extension directory for other settings and definition files which will store details of what functionality is provided in the extension. You can do this through the CMS or manually in your code. If you want to add it through the CMS go to Setup and then choose “Extensions” from the left hand column. If you are not using the default eZ Publish file permissions you may find you have an issue adding the extension in this way.
To add the extension manually, go into your main site.ini file again and you should see a section titled “ExtensionSettings” (in the default override it is usually at the top). Underneath this contains a list of extensions which are enabled on the site. When eZ Publish enables or disables an extension, it merely adds or removes it from this list. The other thing to note with the extensions is that the order matters. If for example you had two extensions which contained template files you may at some point have a situation where both contain overrides to a single file. In this case the extension called first in the list would be used.
Below is a sample list of extensions you should expect to see, I’ve added my one to the bottom line so just replace onequarterenglish on the last line with the name of your extension:
Now that we’ve got a working extension and the debug we need is being displayed, let’s create our first extended attribute filter. Extended attribute filters consist of two parts:
Our first step is to create a settings file so that eZ Publish can find our filter. To do this, we need to create a settings file called extendedattributefilter. This can be used to store as many as you want but for now, we’ll stick with the one. In your extensions directory, create a folder called settings (if one does not already exist). Within this create a file called extendedattributefilter.ini.append.php. Add the following to your new file (comments are supplied):
Now lets get into the real task of making our filter. We will explain how eZ Publish uses the settings information once we have created our class.
Filters work through methods within a class. As you can see from the settings above, eZ publish can translate where it needs to go by looking up the extension name and the location of the class file within this. It’s then simply a matter of going to the correct method (which is also specified in our settings file). Our class therefore is very simple to structure. We just need an empty constructor and then the method we have just specified. Below is an empty implementation of a class:
There is our basic class. Now we’ll add some custom code to pull out the users who can’t remember their passwords. For our Extended Attribute Filter to work, we have to provide eZ with the additional tables we want to use and also the extra conditions we want to place on the data returned. For our first filter, we need to use the eZ Publish table ezforgot_password . This stores the content object ID of each user who has forgotten their password. We then need to make sure in our join that we only count the content objects with an ID in the ezforgot_password table. Below is the code which will do this for us:
function forgottenPasswords( $params )//note the passing in of parameters, we'll look at these afterwards { eZDebug::writeNotice("Running forgottenPasswords Extended Attribute Filter","User Filters");//this makes sure we can easily find the related SQL in our page debug $sql_tables = ', ezforgot_password';//Make sure you start with a comma as our tables list comes after the usual list of tables in the fetch. $sql_joins = 'ezforgot_password.user_id=ezcontentobject.id AND';//ensure you add a trailing "AND" for other conditions used in the fetch statement return array( 'tables' => $sql_tables, 'joins' => $sql_joins );//return the data for eZ in an array }
There are a couple of nuances with the code I’ve noted in the comments:
As you would expect, by default user information is hidden to people viewing the site. If you want to test this functionality you need to either ensure you are logged in as a user with administrator privileges in the front of the site or ensure the template is only visible in the CMS itself.
In my next tutorial I will cover how you can add functionality such as custom templates into the CMS. In the meantime let’s add the functionality to an existing page in the front of the site.
Go into the template you want to add the fetch statement to and add the following code:
{def $fetch_params = hash( 'parent_node_id', 5, 'extended_attribute_filter',hash( 'id', 'forgottenPasswords', 'params', hash( ) ), 'depth', 2 )}{*we need to use these twice so let's store them in a variable before we use them:*} {def $users = fetch( 'content', 'list',$fetch_params)} {def $num_forgotten = fetch( 'content', 'list_count',$fetch_params)} {*we don't need to look up the number forgotten here since we are not using a limit on the fetch rows returned but extended attribute filters can be used for them using the same syntax *}
There are {$num_forgotten} who have forgotten their passwords:
We can use the extended attribute filter for both content lists and content list_counts. You should spot the fetch statements are just regular fetch statement with a slightly more complicated attribute filter. You will also notice the ‘params’ item within the extended_attribute_filter which we will look at in our next example. Please note I had to manually send a forgotten password request on my site since I am running eZ Publish locally. Also note that when you are logged out no users will be shown, they will only show up once you are logged in as an admin user (or a user account for somebody who has read rights on users). Debugging The key to successfully creating Extended Attribute Filters is being able to debug them. Unfortunately there is a lot of SQL statements to work through so being able to find your SQL statement can be tricky. For this reason the following line of code in our forgottenPasswords function is essential, especially during initial development:
eZDebug::writeNotice("Running forgottenPasswords Extended Attribute Filter","User Filters");
Luckily, the process of adding another filter is much quicker than creating your first one. We already have all of the files we need in place we just need to add additional code to a couple of files. Our function will make use of two parameters, these will store the start and end dates to give us the date range we need to check between.
Please note that eZ Publish does not track every single login by the user, only the most recent (although you can also extract when a user account was created or last updated). Therefore if you are looking at logins over a month in the distant past your script will not display users who have logged in since this time. If you do need to run this script, I would recommend running it in a cronjob so you are running it as close as possible to the dates being searched through as possible (please see this tutorial for a guide for doing this).
The first file we need to update is the extendedattributefilter.ini file we created for our first filter. Add the following after our previous filter:
#Filter for extracting the users who have logged in over user specific dates [lastLogin] ExtensionName=onequarterenglish ClassName=UserFilters MethodName=lastLogin FileName=classes/userfilters.php
extensions/onequarterenglish/classes/userfilters.php The only other file we need to update is the file containing our UserFilters PHP class with our SQL query (we’ll walk through the code next). Note the use of the $params array passed into the method:
function lastLogin( $params ) { eZDebug::writeNotice("Running lastLogin Extended Attribute Filter","User Filters");//this makes sure we can easily find the related SQL in our page debug $sql_tables= ', ezuservisit ';//the ezuservisit table contains the last login time (it also contains the number of times a user has logged in) $sql_joins = ' ezcontentobject.id = ezuservisit.user_id AND ';//the first part of our join merely links the contentobject with the database row in ezuservisit //now we need to move onto our parameters (don't forget to ensure the validity of data passed into your function): //START DATE $start_date = (int) $params['start_date'];//do not use an if statement around this so if no start date is replied we only return users have logged in at some point (users who have never logged in will have the last login time set to zero) $sql_joins .= 'ezuservisit.last_visit_timestamp > '.$start_date.' AND ';//don't forget the ANDs //If an end date has been supplied, include this in the query if ( isset( $params['end_date'] ) ) { $end_date = (int) $params['end_date']; $sql_joins .= 'ezuservisit.last_visit_timestamp < '.$end_date.' AND ';//don't forget we need an 'AND' after our join statement } return array( 'tables' => $sql_tables, 'joins' => $sql_joins ); }
Since this example is a bit more complicated, let’s step through the script. The script depends on the ezuservisit table. This stores details of failed logins, most recent login time and how many times users have been logged in. Our first task is to link the ezuservisit into our SQL statement:
eZDebug::writeNotice("Running lastLogin Extended Attribute Filter","User Filters");//ensures that we can find the SQL in the debug $sql_tables= ', ezuservisit ';//the ezuservisit table contains the last login time (it also contains the number of times a user has logged in) $sql_joins = ' ezcontentobject.id = ezuservisit.user_id AND ';//the first part of our join merely links the contentobject with the database row in ezuservisit
As we are looking at custom dates, we now need to make sure we use the parameters the user passes in. These are accessed using the parameter array passed into our function (in our case the parameters will be in the $params array). Parameters can be easily missed so we need to make sure we account for parameters which are not supplied. In our case we ignore an end date if this is not supplied but if no start date is supplied we assume we should return all users who have logged in to the site. We also ensure that the dates are sent to queries as integers to avoid invalid input:
//now we need to move onto our parameters (don't forget to ensure the validity of data passed into your function): //START DATE $start_date = (int) $params['start_date'];//do not use an if statement around this so if no start date is replied we only return users have logged in at some point (users who have never logged in will have the last login time set to zero) $sql_joins .= 'ezuservisit.last_visit_timestamp > '.$start_date.' AND ';//don't forget the ANDs //END DATE //If an end date has been supplied, include this in the query if ( isset( $params['end_date'] ) ) { $end_date = (int) $params['end_date']; $sql_joins .= 'ezuservisit.last_visit_timestamp < '.$end_date.' AND '; }
Once this has been carried out we just need to return our SQL:
return array( 'tables' => $sql_tables, 'joins' => $sql_joins );
We can now include the fetch in our templates using the following code (be sure to make a note of how the parameters are set):
{*setting our start and end dates*} {def $start_date = maketime( 0,0,0,7,1,2010)} {def $end_date = maketime( 0,0,0,7,26,2010)} {*fetch statement: *} {def $lastLogin = fetch( 'content', 'list', hash( 'parent_node_id', 5, 'extended_attribute_filter', hash( 'id', 'lastLogin', 'params', hash('start_date',$start_date, 'end_date',$end_date ) ), 'depth', 10 ) )}
The code returned from the fetch in the debug should be similar to the following, again note that our query is shown second after the debug notice (look for ezuservisit in the debug):
So now we’ve looked at using two Extended Attribute Filters in our templates but what happens if you need to add them to a script? Luckily this is straightforward. The following example assumes the logged in user has access to the user information:
//working out start and end points: $now = time();//lets use today so there will be some info $one_month_ago = strtotime ( '-1 month' , $first_of_month );//our start time is one month ago. //our extended attribute filter: $extendedAttributeFilter = array( 'id', 'lastLogin', array('start_date'=> $one_month_ago,'end_date'=>$now)); $includeClasses = array('user'); $sortBy = array("name",true);//let's sort by name //parameters to pass into the fetch: $params = array('SortBy'=>$sortBy, 'ClassFilterType'=>'include', 'ClassFilterArray'=>$includeClasses, 'ExtendedAttributeFilter'=>$extendedAttributeFilter); //Limiting the area eZ has to look to just be in 'Members' $parent_node = eZContentObjectTreeNode::fetchByURLPath('users/members');//note the lowercase. Use underscores rather than hyphens if spaces are included in the path $parent_node_id = $parent_node->NodeID; $monthly_logged_in = eZContentObjectTreeNode::subTreeByNodeID( $params,$parent_node_id ); $monthly_logged_in_count = eZContentObjectTreeNode::subTreeCountByNodeID( $params,$parent_node_id ); $cli->output('Users Logged in over the past month: '.$monthly_logged_in_count);
Let’s step through the script to see what’s happening:
//working out start and end points: $now = time();//lets use today so there will be some info $one_month_ago = strtotime ( '-1 month' , $first_of_month );//our start time is one month ago. //our extended attribute filter: $extendedAttributeFilter = array( 'id', 'lastLogin', array('start_date'=> $one_month_ago,'end_date'=>$now)); $includeClasses = array('user'); $sortBy = array("name",true);//let's sort by name //parameters to pass into the fetch: $params = array('SortBy'=>$sortBy, 'ClassFilterType'=>'include', 'ClassFilterArray'=>$includeClasses, 'ExtendedAttributeFilter'=>$extendedAttributeFilter);
First we establish the values we need to pass into the Extended Attribute Filter (in our case we are looking over the previous month). We then store this in an array with the ID of the Extended Attribute Filter so eZ Publish knows which filter to use. We can then store the other limitations we are going to add to our fetch, namely our sort order and our classes to include. Finally we store these details in one single array so they are ready to be passed into the fetch.
//getting the ID of where the users sit in the cms (limiting the area eZ has to search for the objects): $parent_node = eZContentObjectTreeNode::fetchByURLPath('users/members');//note the lowercase. Use underscores rather than hyphens if spaces are included in the path $parent_node_id = $parent_node->NodeID;
This part of the script is important as it limits where eZ needs to look for the users. In this case, we have limited it to the members section of the users tab (so do not be surprised if the admin user does not show up when you run the query). If you wanted to move the query up to be all users, make sure to add a depth filter to the parameters array we just created to ensure eZ goes to a greater depth than it’s default of one.
$monthly_logged_in = eZContentObjectTreeNode::subTreeByNodeID( $params,$parent_node_id );
$monthly_logged_in_count = eZContentObjectTreeNode::subTreeCountByNodeID( $params,$parent_node_id );
$cli->output(‘Users Logged in over the past month: ‘.$monthly_logged_in_count);
The final part of our script then merely runs the fetch statements. All we do with the information is to output the number of users. Please see my previous tutorial on Fetching User Information with eZ Publish if you wish to print off more user information.
In this tutorial we have covered how you can create and use your own Extended Attribute Filters, both in your templates and in your scripts. If you do need to create your own, you may want to check if what you’re after already exists by checking out those created by eZ Publish community . My next tutorial will cover how you can create your own templates within the eZ Publish CMS so that the user information extracted here is only presented in templates available to admin users.