Forums / Suggestions / Ideas for an eZModuleInterface

Ideas for an eZModuleInterface

Author Message

Peter Rudolfsen

Tuesday 02 February 2010 1:14:03 am

First, I'd point out that I'm quite new to eZ Publish, and I only have experience with 4.2.

But one of the first things I thought of when creating my first module, was "why do I have to declare all this variables, why not just declare a class that implements a eZModuleInterface?".

I did a few changes in the eZModule class that enables a class definition in module.php instead of the normal variables (Module, ViewList and FunctionList).

The class is very simple, it implements eZModuleInterface with three public methods: getModule, getViewList and getFunctionList, which should all return arrays with the usual content, nnd the name of the class has to follow the convention modulename_ezModule.

Here's the interface:

interface eZModuleInterface
{
    public function getModule();

    public function getViewList();

    public function getFunctionList();
}

And an example class:

class my_module_eZModule implements eZModuleInterface
{

    public function getModule()
    {
        return array( 'name' => 'Module name' );
    }

    public function getViewList()
    {
        return array(
                'main' => array(
                    'functions' => array(  ),
                    'script' => 'main.php'
                ));
    }

    public function getFunctionList()
    {
        return array();
    }
}

The change in ezmodule.php is quite minor, and I can post that as well if there's any interest in it, but the important thing is that the change is and should be compatible with the normal way of defining modules.

This is just a quick outline, I'm sure a module class/interface can do a lot more than the above, but my experience is limited, and I could use some input on this.

What do you think? Any point in spending any more time on this?

Peter Rudolfsen

Tuesday 02 February 2010 1:26:18 am

Here's the diff on eZModule:

 --- a/lib/ezutils/classes/ezmodule.php
+++ b/lib/ezutils/classes/ezmodule.php
@@ -32,7 +32,6 @@
   \brief Allows execution of modules and functions
 
 */
-
 class eZModule
 {
     const STATUS_IDLE = 0;
@@ -45,6 +44,8 @@ class eZModule
     const HOOK_STATUS_CANCEL_RUN = 1;
     const HOOK_STATUS_FAILED = 2;
 
+    const CLASS_POSTFIX = '_eZModule';
+
     function eZModule( $path, $file, $moduleName, $checkFileExistence = true )
     {
         $this->initialize( $path, $file, $moduleName, $checkFileExistence);
@@ -64,7 +65,28 @@ class eZModule
             unset( $FunctionList );
             unset( $Module );
             unset( $ViewList );
-            include( $file );
+
+            $moduleClass = $moduleName . self::CLASS_POSTFIX;
+            if ( !in_array( $moduleClass, get_declared_classes() ) )
+            {
+                include( $file );
+            }
+
+            if ( in_array($moduleClass, get_declared_classes() ) )
+            {
+                $moduleObject = new $moduleClass();
+
+                if ( !$moduleObject instanceof eZModuleInterface )
+                {
+                    $this->ExitStatus = eZModule::STATUS_FAILED;
+                    return;
+                }
+
+                $Module = $moduleObject->getModule();
+                $ViewList = $moduleObject->getViewList();
+                $FunctionList = $moduleObject->getFunctionList();
+            }
+
             $this->Functions = $ViewList;
             if ( isset( $FunctionList ) and
                  is_array( $FunctionList ) and

Nicolas Pastorino

Tuesday 02 February 2010 1:34:59 am

Hallo Peter !

Welcome aboard the eZ Community, and thanks for this awesome post :)

I love this idea. This actually turns the currently 'include'-based module definitions into a proper object-oriented system, and this is not nothing. One of the immediate consequences is that it reduces the amount of required workarounds and hacks in the kernel, which leaves a larger room for maintenance-peace and tranquility: upgrades, automatic bugfixing (eZ Premium).

Have you pushed this in production yet ? Any other magical enhancements proposals like this one ?

Cheers,
PS : not sure you already did this, but check the "Track this reply" checkbox next to the "OK | "Cancel" buttons below, so that you are automatically notified when someone replies to your threads. You can also check your notification settings there : http://share.ez.no/notification/settings

--
Nicolas Pastorino
Director Community - eZ
Member of the Community Project Board

eZ Publish Community on twitter: http://twitter.com/ezcommunity

t : http://twitter.com/jeanvoye
G+ : http://plus.tl/jeanvoye

Peter Rudolfsen

Tuesday 02 February 2010 1:48:33 am

Hey Nicolas!

I haven't pushed this into production, as I wanted to post it here first to see if there was any interest in this.

There's probably even more to gain by adding support for object-oriented views as well, but that looks like a bit more work. I'll take a look at it and see if I can come up with a suggestion.

Any ideas to how that could be implemented?

I'm tracking this post now, btw :)

Matthieu Sévère

Tuesday 02 February 2010 2:08:07 am

That's a great idea !

Methods to hook easily could be great too !

--
eZ certified developer: http://ez.no/certification/verify/346216

Felix Woldt

Tuesday 02 February 2010 5:27:06 am

I like this, too.

Then it is also easier to write a global phpdoc for a module.

http://www.jac-systeme.de - Developers united in eZ Publish: http://www.cjw-network.com

CJW Newsletter 1.0.0 released: http://projects.ez.no/cjw_newsletter

Peter Rudolfsen

Tuesday 02 February 2010 5:46:16 am

Another ting... aren't the 'views' of eZ Publish more like a 'controller' if you think about the traditional sense of MVC?

So why not create some type of controller class with actions and have eZProcess execute them as objects instead? Or just leave eZProcess alone and write another class, say eZDispatch, that's used if the 'view' file contains a class.

Guess you should be careful with tampering with the terminology, but at least I think it's easier to think of the views as controllers.

I searched around a bit and found a few MVC tools in ez components. Is that something that could be used for this, or would that introduce too many unwanted dependencies to ezc?

Gaetano Giunta

Tuesday 02 February 2010 2:19:25 pm

I like this idea too.

The current interface of ezmodule is also a bit limited, as you usually end up coing the extension/module/view names a lot of times in the $Results var at the bottom of your views and in the template code.

While at it, why not removing the eztemplateautoload.php file, and other such redundant definition of stuff that can be

-neatly wrapped in php classes and methods

- can thus take advantage of autoload magic instead of having ezp scanning frantically directories to find the proper xxx-handler php class file?

I think the answer we all know, and it is: because we like to keep our binary compatibility...

Principal Consultant International Business
Member of the Community Project Board

Gaetano Giunta

Friday 23 April 2010 1:53:13 pm

@Peter I played a bit more with module introspection, and I think I might have missed the overall sense in your original post.

The first thing I want to say is that the current ezmodule class has attributes that can be used to retrieve from it the list of views; it really misses one to retrieve the list of fetch functions though!

To have a module defined via a php class, we should, imho

- rename your interface to ezmoduleimplementorInterface or ezmoduleDefinitionInterface, as it is not a module in itself you are implementing, the module being the instance of ezmodule

- maybe add a new ini setting to declare which modules use the new interface instead of module.php, instead of using the automatic 'try loading class first' approach

Principal Consultant International Business
Member of the Community Project Board

Kristof Coomans

Saturday 24 April 2010 5:01:52 am

Maybe it would be worth investigating as well if it would be beneficial to move away from PHP for this kind of (sometimes deeply nested) definition arrays and go for XML instead. A big advantage you will have is that if you provide a proper XML schema, it's easier for developers to create their module definition (xml intellisense/autocompletion provided by IDE's), as well as to validate it. IMHO the current nested array structures are very unhandy.

independent eZ Publish developer and service provider | http://blog.coomanskristof.be | http://ezpedia.org

Gaetano Giunta

Saturday 24 April 2010 10:37:58 am

@kristof: I longed a while ago for replacing module.php and also the definition of template operators and functions with ini files. Then I stumbled upon some module that dynamically created the available views and I understood the benefit of having it done in php.

I personally am not a big fan of xml - a hybrid format too hard and verbose to be read/written by humans and too complex to be efficiently parsed by machines. In 2010 you see more and more json, yaml and other simpler markup formats that are suited to represent data and less and less xml.

But moving between a php array and json/xml is a matter of 2 lines of code

What I'd like instead is

- more docs on the possible values for those arrays - most of those are hidden and never documented anywhere

- some validator script / function

Principal Consultant International Business
Member of the Community Project Board

Kristof Coomans

Saturday 24 April 2010 2:15:29 pm

Regarding the module that dynamically created the available views, can you give us some more insight in what had to be accomplished this way what could not be accomplished with view parameters instead?

The validation you would get for free if you use XML, once you agree on the grammar through a XML schema. You need a blue print of the elements that can/should be there anyway, and better to do it formally following a W3C recommendation than with (often incomplete and out to date) documentation and some custom PHP code that checks for the existence of specific array keys.

There is some sort of schema validator as well for YAML, Kwalify, but afaik only Ruby and Java versions exist.

YAML might be less verbose to create by hand, but with an IDE with good XML schema support you're able to get equal results in no time.

independent eZ Publish developer and service provider | http://blog.coomanskristof.be | http://ezpedia.org

Jérôme Vieilledent

Sunday 25 April 2010 2:20:04 am

I just LOVE your idea Peter !

Besides, you're right, module views are MVC controllers and they also should be classes (like in Symfony framework). It would be much cleaner IMHO !

@Kristof : Your idea is really interesting too ;). Writing module definition is really annoying, such as for template operators, fetch functions or operations ! From this point of view I'd rather prefer YAML vs XML, plus a PHP class (extending eZModule and implementing an interface) for module views (we really should call them controllers as it's quite confusing).

Gaetano Giunta

Sunday 25 April 2010 4:00:14 am

@kristof: I agree that validating xml is currently easier because of a wide array of tools available.

But I think that the validation part is not what is hampering development currently. The DOCS are. And no, saying that docs get outdated is not a good reason to avoid producing docs altogether.

Take for a simple example the following parameter in a module.php file:

 $Module = array( 'name' => 'So what?',

              'variable_params' => true );

raise your hands if you know what 'variable_params' does (ps: I am sure that Kristof does, but how many besides him do?).

And honestly, tell me if validating that to check that it is either true or false helps you a bit.

I'm not against using existing schema languages; it is of course faster than writing a custom php validator from scratch, but what we should achieve is rather making sure that the docs are embedded in the validation definition. And afaik, niether relax-ng nor xsd mandate that you document elements.

In my experience the best (basically the only) way to make sure that doc is always up-to-date is to

a - use meaningful names for variables, methods, classes, and a clean design, and

b - embedding comments in the doc

For point b, while sticking to the current php-based definition, we could add phpdoc comments to a "blueprint" module definition, and voilà, zero impact and a large gain. We could of course also add php classes that act as blueprint for those arrays ('struct' classes), similar to the way that ezc define options...

Principal Consultant International Business
Member of the Community Project Board

Gaetano Giunta

Sunday 25 April 2010 4:08:15 am

Ps: my fave one: json + jsonschema

Principal Consultant International Business
Member of the Community Project Board

Gaetano Giunta

Sunday 25 April 2010 4:30:44 am

@Jerome: writing stub code is a chore I agree;

But I think that copying and pasting xml would be no better than copying and pasting php code.

And even if we made a GUI wizard like the one that exists for tpl ops, it would not be much faster for the developer in the end.

What we need otoh is reverse engineering of existing php code that creates the stubs automatically.

I think it would make for a nice extension: scripts/php classes to wrap an existing php function/object into a template operator stub, a template fetch function stub, etc...

Willing to join?

Been there, done that, btw: http://phpxmlrpc.svn.sourceforge.net/viewvc/phpxmlrpc/trunk/xmlrpc/lib/xmlrpc_wrappers.inc

Principal Consultant International Business
Member of the Community Project Board

Kristof Coomans

Sunday 25 April 2010 6:21:47 am

@Gaetano: I am not talking here about copying and pasting code, but about making it possible to use auto completion in your favorite (intelligent) editor according to a defined grammar: in the case of PHP code, methods and properties of your classes and objects; in case of XML, elements and attributes defined in the schema. AFAIK there is no good auto completion support for JSON / YAML in most popular enterprise-level open source editors (just to name 2 I am using: Eclipse, Netbeans) yet. If you know existing tools to do this, let me know and I'll be happy to try them.

I would like the approach of Zeta Components' struct classes a lot more than YAML/JSON then, because of auto completion possibilities.

By the way, there is some documentation available at http://ezpedia.org/en/ez/module for a long time already, might be improved heavily language-wise though.

independent eZ Publish developer and service provider | http://blog.coomanskristof.be | http://ezpedia.org

Jérôme Vieilledent

Sunday 25 April 2010 10:04:43 am

"

Willing to join?

"

Sure ! :)

Gaetano Giunta

Monday 26 April 2010 9:05:04 am

Excellent page on ezpedia, thanks kristof!

Summing it up, to get the power of ide introspection without hacking the kernel code, we might define new 'struct' classes for module def, view def, fetch_function def, that implement the arrayaccess interface (http://php.net/manual/en/class.arrayaccess.php) and everybody's happy.

Principal Consultant International Business
Member of the Community Project Board

Kristof Coomans

Monday 26 April 2010 11:17:26 am

Yes, that sounds like a good plan.

Will it be in the core of 4.4?

independent eZ Publish developer and service provider | http://blog.coomanskristof.be | http://ezpedia.org

eZ debug

Timing: Jan 17 2025 20:52:46
Script start
Timing: Jan 17 2025 20:52:46
Module start 'content'
Timing: Jan 17 2025 20:52:47
Module end 'content'
Timing: Jan 17 2025 20:52:47
Script end

Main resources:

Total runtime1.0514 sec
Peak memory usage4,096.0000 KB
Database Queries273

Timing points:

CheckpointStart (sec)Duration (sec)Memory at start (KB)Memory used (KB)
Script start 0.00000.0055 597.1641180.8359
Module start 'content' 0.00550.9514 778.00001,032.9063
Module end 'content' 0.95700.0944 1,810.9063376.6016
Script end 1.0514  2,187.5078 

Time accumulators:

 Accumulator Duration (sec) Duration (%) Count Average (sec)
Ini load
Load cache0.00350.3331210.0002
Check MTime0.00130.1233210.0001
Mysql Total
Database connection0.00060.059910.0006
Mysqli_queries0.910686.60922730.0033
Looping result0.00220.20972710.0000
Template Total1.025497.520.5127
Template load0.00180.171920.0009
Template processing1.023697.350020.5118
Template load and register function0.00010.014110.0001
states
state_id_array0.00070.069010.0007
state_identifier_array0.00090.081020.0004
Override
Cache load0.00190.18081310.0000
Sytem overhead
Fetch class attribute can translate value0.00100.098180.0001
Fetch class attribute name0.00130.1216260.0000
XML
Image XML parsing0.00300.287780.0004
class_abstraction
Instantiating content class attribute0.00010.0057360.0000
General
dbfile0.00340.3277580.0001
String conversion0.00000.000630.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.tplfull/forum_topic.tplextension/sevenx/design/simple/override/templates/full/forum_topic.tplEdit templateOverride template
20content/datatype/view/ezxmltext.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tplEdit templateOverride template
29content/datatype/view/ezxmltags/paragraph.tpl<No override>extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tplEdit templateOverride template
4content/datatype/view/ezxmltags/literal.tpl<No override>extension/community/design/standard/templates/content/datatype/view/ezxmltags/literal.tplEdit templateOverride template
16content/datatype/view/ezimage.tpl<No override>extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tplEdit templateOverride template
3content/datatype/view/ezxmltags/strong.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/strong.tplEdit templateOverride template
1content/datatype/view/ezxmltags/line.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/line.tplEdit templateOverride template
2content/datatype/view/ezxmltags/emphasize.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/emphasize.tplEdit templateOverride template
1content/datatype/view/ezxmltags/quote.tpldatatype/ezxmltext/quote.tplextension/ezwebin/design/ezwebin/override/templates/datatype/ezxmltext/quote.tplEdit templateOverride template
1content/datatype/view/ezxmltags/sup.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/sup.tplEdit templateOverride template
1pagelayout.tpl<No override>extension/sevenx/design/simple/templates/pagelayout.tplEdit templateOverride template
 Number of times templates used: 79
 Number of unique templates used: 11

Time used to render debug report: 0.0001 secs