[Python-3000] ABC's, Roles, etc

Phillip J. Eby pje at telecommunity.com
Wed May 9 19:28:18 CEST 2007


At 09:57 PM 5/8/2007 -0600, Jeff Shell wrote:
>I must admit that I didn't read PEP 3124 in depth - most of it was
>fascinating, some of it went way over my head in complexity, and then
>suddenly I saw an Interface. It seemed quite out of place, actually,

Yep; it's really more like a "adapting abstract base class".  In a 
GF-based language like Dylan, it would probably just be called a 
"module".  In Dylan, the generic functions you export from a module 
define an interface, in precisely the same way as PEP 3124 Interfaces 
work, except they have no IFoo(bar).baz(), only IFoo.baz(bar).  (And 
with different syntax, of course.)


>[snip lots of stuff about zope.interface's specification features]

If you don't count in-house "enterprise" operations and shops like 
Google, Yahoo, et al., the development of Zope is certainly one of 
the largest (if not the very largest) Python project.  It's 
understandable that LBYL is desirable in that environment.  On the 
other hand, such large projects in Python are pretty darn rare.

Meanwhile, the subject of typing systems for Python (into which 
category zope.interface most assuredly falls) is still an open 
research area.  I've watched zope.interface and its predecessors and 
spin-offs for almost a decade now, and it's *still* not particularly 
settled.  Look, for example, at Guido's blog posts about type 
expressions and type parameterization.  Look at how quickly the 
efforts to define standard ABCs for Py3K turned back from grand 
vision to, "oh heck that doesn't quite do what we wanted".

So, my thought is that LBYL type systems for Python are still "here 
there be dragons" territory.  PEP 3124 simply doesn't try to go 
there, but neither does it block the passage of other explorers.  It 
happily coexists with other type systems, and if you want to use 
something called "roles" or "traits" as argument annotations, it will 
be OK with that.

The specifics, which I haven't spelled out in the "extension API" 
section yet, are mainly that in order to work with the default 
dispatching engine, you must register methods for the "implies()" 
generic function, such that the engine can tell whether a class 
implies the annotation, the annotation implies a class, or the 
annotation implies any other annotation.

Of course, there will also be a generic function you can register 
with in order to use a different dispatch engine when your 
annotations are encountered.  This would be the hook you'd need to 
use in order to have instance-specific checks.  Bear in mind, of 
course, that the such checks will necessarily be slow, and the 
slowdown may apply to every invocation of the function.


>'Look Before You Leap'
>brings to mind the concept of "don't dive into an empty pool" or
>"don't do a backwards flip onto the pointy rocks"; where as 'Easier to
>Ask Forgiveness Than Permission' brings to bind the concept of "sorry
>i dove head first into your empty pool and cracked my skull open Mr.
>Johnson. If I had asked I'm sure you would have said no! In any case,
>even though it's your pool and there was a fence and everything and
>you did not give me permission, my parents are going to sue" (OK,
>maybe that last bit is the result of a healthy american upbringing...
>but still!)

Well, if you run a program whose effects are that important without 
having tested it first, perhaps you *should* be sued.  :)


>Huh? I'll try to look at that. types, classes, instances... That does
>it, I'm switching to Io. (Honestly - I've recently seen the light
>about prototype based object oriented programming; in light of types
>of types and classes of classes and classes and instances and oh my,
>languages that believe in "there are only objects, and they are only
>instances" are sounding sweeter every day)

Why stop at Io?  Cecil is a prototype-based language with generic 
functions, including full predicate dispatch and even "dynamic types" 
(i.e., an object can be of several types at the same time, depending 
on its current state).  :)

For Python, though, I really don't see the need to create such 
ultra-dynamic objects.  It's so easy to just dynamically create a 
class whenever you need one, that it doesn't seem worth the bother to 
munge instances.  Zope, of course, is always a special case in this 
respect, since *persisting* dynamic classes is a PITA, compared to 
dynamic instances.  But the stdlib ain't Zope, and most Python code 
doesn't need to have its classes stored in a database.


>>Since #1 falls outside of PEP 3124's goals (i.e., it's not about
>>specification or testing), that leaves use case #2.  In my
>>experience, it has been more than sufficient to simply give these
>>object some *other* interface, such as an IViewTags interface with a
>>method to query these dynamic "tag" interfaces.  In other words, my
>>experience and opinion supports the view that use case #2 is actually
>>a coincidental abuse of interfaces for convenience, rather than the
>>"one obvious way" to handle the use case.
>
>Ugh. Yeah, there are 'marker' interfaces, but.. ugh. dynamic "tag"
>interfaces. Yuck.
>
>My experience has been otherwise.

Now you're confusing me.  How is having a way to ask for markers 
different from marker interfaces?  Both are equally "yuck", except 
that one isn't abusing interfaces to use them as markers.


>But I'm sorry that I confused that section of PEP 3124 to be about
>specification and testing. I do, however, think that is a better use
>case.

Well, write another PEP, then.  :)


>We have a dynamic data transformation framework that exists outside of
>Zope (Zope is basically used for UI). Objects are being dynamically
>composed, wrapped, decomposed, rewrapped, filtered, and split -
>constantly. Objects, not types. It's all composed of rules. I'm
>itching to be able to add rules to apply zope.interface specifications
>to the generated objects; if only to then make it much easier to add
>other filtering rules later on.

Maybe I'm getting to be like Guido in my old age, but maybe you 
should just write the program first and extract the framework second.  :)

In truth, though, I'd almost bet some serious cash that proper use of 
generic functions would evaporate your framework to virtual 
nothingness.  My observation has been that in languages with generic 
functions, the sort of thing that requires complex frameworks with 
lots of interfaces in Zope looks like a trivial little library.  In 
PEAK, I was able to cut out about 75% of the code in one 
sub-framework by switching it from interfaces+adaptation to generic 
functions, and in the process made it much more comprehensible.  With 
that kind of productivity enhancement, one can afford to write a lot 
more unit tests to do the LBYL-ing, and still come out ahead.  :)

Really, the problem of LBYL interfaces and adaptation is that they 
require you to laboriously figure out in advance where to allow 
flexibility in the framework.

Worse, they emphasize the *solution* domain rather than the problem 
domain.  They define what's required of parts that have to be plugged 
into a machine that then "solves" the problem.  So you have to design 
that machine and where various parts fit into it, which is largely a 
distraction from whatever you were trying to do in the first place!

In contrast, with generic functions, you focus simply on identifying 
the operations required by the problem domain, performing a 
functional decomposition rather than trying to create an entire 
network mesh of roles and responsibilities.  You code the *problem*, 
not the solution, so your code might even be comprehensible (or at 
least explainable) to non-programmer domain experts.  (i.e., even if 
they can't read the code, you can read it to them and confirm whether 
it matches the requirements.)

*Then*, after you have the functional decomposition (which is your 
real "specification" anyway), you can then decide what concrete 
object types you might need, and implement the lowest-level domain 
operations for those types.

And if you're following that process, interfaces really aren't 
anything but the documentation that explains what those 
problem-domain operations are supposed to do, so that if for some 
reason you need to implement new concrete types at a later time, you 
can add appropriate implementations.

IMO, that's a lot closer to being the One Obvious Way, because it 
doesn't *need* LBYL or anything like zope.interface, anywhere in that 
process.  See, if you want contract enforcement, you can just *build 
it right in* to the generic functions, using an appropriate method 
type.  See my comments here:

http://mail.python.org/pipermail/python-3000/2007-May/007444.html

Of course, that's not a *static* guarantee of correctness, but 
neither is most of what you're talking about.  Tests are still 
required, either way.



More information about the Python-3000 mailing list