[Grok-dev] Re: Global utilities registered too early?
Philipp von Weitershausen
philipp at weitershausen.de
Tue Jun 19 13:33:36 EDT 2007
Jan-Wijbrand Kolman wrote:
> I'm trying to create/register a global utility by subclassing
> grok.GlobalUtility. If I understand correctly, exactly one instance of
> this utility class is created at registration time, right? It is this
> instance that is returned when calling getUtility, right?
Right; Right.
> Now, in my case the __init__ of my utility class makes use of other
> global utilities that are registered by a "third party" package
> (called userschema). However, when creating the instance for my
> utility class, the "third party" utilities are not registered (yet)
> and thus cannot be found.
>
> I'm not sure I fully understand why this is the case (and thus I do
> not see a way around it yet). I also tried to <include
> package="userschema" ?> just before the <grok:grok package="."/>
> directive in my package's configure.zcml, but to no avail; My global
> utility seems always to be registered before the utilities from
> userschema are registered.
>
> Maybe somebody can nudge me in the right direction with this?
In general I would say you're doing it wrong. As I asserted above,
utility's __init__ is called during registration time at which time the
state of the component registry cannot be known. I don't think you
should make any assumptions at that point. In particular, you shouldn't
make any assumptions for the rest of the Python process.
One of the key features of the Component Architecture is
overrideability. We can have it by simply registering a new utility
"over" an old one, we can do it elegantly with ZCML overrides and we can
do it locally using local sites. Point is, to ensure this
overrideability, there are certain sacrifices to make which can be, I
think, summed up in two simple rules of thumb:
* don't register during import time
* don't look up during registration time
In particular, you only want to do look-ups during request time because
the way components are acquired may be different in each individual
requests, thanks to local sites.
"I don't have local components, they're all global utilities" you might
say. Well, getUtility() isn't aware of that and I think you should never
code in a way that it implicitly has to make that assumption. So,
whenever you need those secondary utilities in your utility, get them by
doing getUtility() each time. Might not be particularly elegant from a
Python perspective, but it's the price to pay for flexibility.
--
http://worldcookery.com -- Professional Zope documentation and training
More information about the Grok-dev
mailing list