[PEAK] Prototype generic functions now available in CVS

Phillip J. Eby pje at telecommunity.com
Tue Jul 13 19:06:06 EDT 2004


The CVS trunk version of PyProtocols and PEAK now supports generic 
functions using 'when()', e.g.:

     from protocols.dispatch import when

     [when("x in int and y in str")]
     def do_something(x,y):
         # body here will be executed when 'x'
         # is an integer and 'y' is a string

If 'do_something' is an existing generic function, the function body is 
added to it, for execution when the condition is true.  If there is 
currently no 'do_something' generic function, it is created.  So, you can 
in one module define a few initial cases for a generic function, and then 
extend it in another, like so:

     from other_module import do_something

     [when("some new condition here")]
     def do_something(x,y):
         # etc.

Note that this *modifies* the imported generic function, so the new case 
will be used whenever the original generic function is called, as long as 
the condition is the most-specific applicable one.

This is an extremely early release, and there's still a long to-do 
list.  For example, you can't refer to varargs or keyword args in your 
when() conditions.  There's no way to make "real" variables (i.e. not have 
names looked up at function definition time), except by having them not be 
defined yet.  Everything is pure Python, and probably rather slow.  There's 
no real method combining yet.  Lamdbas and list comprehensions aren't 
allowed in 'when()' expressions.  And so on.

Based on the total implementation size, (~2700 lines), this actually 
doubles the current size of PyProtocols, making it probably sensible to 
give it its own package.  :)  It's indeed half the size of the pure-Python 
'compiler' package, which I guess makes sense since it is indeed a compiler 
of sorts for about half the Python language.

One caveat: ordering of operations among subexpressions is not currently 
guaranteed.  If you write something like:

     [when("y<>0 and x/y>23")]

it's potentially possible that these conditions will be evaluated in a 
different order, making it possible to get a ZeroDivisionError.  Fixing 
this is one of the next things up on the to-do list, though, and it's 
pretty much just the writing of some rather tedious tests and 
code.  (Specifically, to ensure that Signature operations are always 
order-preserving, and to have GenericFunction objects keep an acyclic 
constraint graph between expressions that share common variables, using it 
to restrict its choice of the "best expression to test next".)

Oh, and don't forget what I said in a recent e-mail: the 'when()' 
expression language doesn't execute exactly the way Python does, even 
though it's Python syntax.  Anything that *can* be computed when the 
expression is parsed, *will* be.  If you say something like "x > 
foo.bar[baz]", and 'foo' and 'baz' are not arguments to the generic 
function, then 'foo.bar[baz]' will be computed at *parse time*.  If 'baz' 
is an argument, then 'foo.bar' will be computed at parse time.  Only 
subexpressions involving function arguments will be computed when the 
generic function is invoked.

In general, that's a really good thing, but it basically means you should 
make sure that anything you reference in the condition already has the 
value you want, at the time of the 'when()' call.  Alternatively, you can 
make sure the variable(s) are *not* defined, in which case it will force 
calculation at runtime.

(Hmmm...  which means that if you typo, you won't find out about it until a 
case that needs to compute that expression happens.  So maybe there should 
be some kind of warning unless you specifically identify variable portions 
of the expression.  It seems worth thinking about, anyway.)




More information about the PEAK mailing list