I've had a chance to look a bit more at the sourceof dictobject.c, and I now
believe that PyDict_Items will not be of any help (it appears to use many of
the same structures that do us no good in PyDict_Next). I now think
something like the following pseudo code would be right:
dict = PyDict_New();
list = PyObject_CallMethod(dict, "items", NULL);
for tuple in list {
k = tuple[0];
v = tuple[1];
PyDict_SetItem(dict, k, v);
}
Of course, this is some ugly pseudo code, but hopefully the idea is clear.
Jason
> -----Original Message-----
> From: quixote-users-bounces+jsibre=chironsys.com@mems-exchange.org
> [mailto:quixote-users-bounces+jsibre=chironsys.com@mems-exchange.org]On
> Behalf Of Jason Sibre
> Sent: Sunday, March 07, 2004 7:15 PM
> To: quixote-users@mems-exchange.org
> Subject: [Quixote-users] Inconsistency between _c_htmltext.c
> and_py_htmltext.py
>
>
> I have been playing around with the form frameworks (form and
> form2), and I
> tried to create a class that wrapped the form2.Form, and allowed
> dictionary-style access to the Widget's render() method and the
> WidgetRow's
> title attribute, when I discovered a bug in _c_htmltext.c.
> Rather, after a
> few hours of debugging, I'm pretty sure it's a bug.
>
> The problem is that an instance of htmltext that came from the c version
> will not accept a mapping object other than a dict, or a subclass of one,
> while the python version of htmltext will allow you to provide any
> 'mapping-like' object. Specifically, the python version requires that the
> mapping implement items().
>
> In a nutshell, the line that produces the error looks like this (in a
> modified form2.Form object):
> return htmltext(self.body_template) % FormTemplateDict(self)
>
> I know those names won't mean much to you folks, as they aren't
> part of the
> standard form2, so I'll explain briefly. body_template is a string, along
> the lines of:
> """\
>
>
> %(titleof_testInitValue)s |
> %(testInitValue)s |
> Named |
> %(testName)s |
>
>
> %(submit)s |
>
>
"""
>
> And FormTemplateDict (my fake dictionary wrapper) wraps self (the Form
> instance) providing __getitem__ (items(), values(), etc...) access to the
> widgets by via the format codes in body_template. For example,
> FormTemplateDict(self)['titleof_testInitValue'] would return the .title of
> the WidgetRow containing the widget named 'testInitValue', while
> FormTemplateDict(self)['testInitValue'] would return the results
> of calling
> that Widget's render() method.
>
>
> Ok. So, If I do:
> return self.body_template % FormTemplateDict(self)
> it works.
>
> If I do:
> return htmltext(self.body_template) % FormTemplateDict(self)
> with _c_htmltext.so missing or renamed, causing _py_htmltext.py to be used
> instead, it works.
>
> But if I try the desired call:
> return htmltext(self.body_template) % FormTemplateDict(self)
> while using the _c_htmltext.so file. It blows up with a KeyError.
>
>
> I'm not much of a C programmer, and I have NO experience with doing python
> extensions in C, but I spent some time trying to narrow down the problem
> (with the hope of providing a patch), and the best I could
> conclude is that
> the following loop
>
> _c_htmltext.c
> 323 while (PyDict_Next(args, &pos, &key, &value)) {
> 324 PyObject *wvalue = wrap_arg(value);
> 325 if (wvalue == NULL) {
> 326 Py_DECREF(wargs);
> 327 return NULL;
> 328 }
> 329 if (PyDict_SetItem(wargs, key, wvalue) < 0) {
> 330 Py_DECREF(wargs);
> 331 return NULL;
> 332 }
> 333 Py_DECREF(wvalue);
> 334 }
>
> requires a REAL dict (or subclass). It does not call "items()" as the
> python implementaion in _py_htmltext.py does. In fact, I provided a
> subclass of a dict instead, with values in it, and then put a
> __getattribute__ method to log all accesses, and it worked fine, but no
> accesses were logged at all (except for the update() in my constructor,
> where I initialized the object to match a dictionary passed to the
> constructor). My suspicion is that the PyDict_Next loop should
> be rewritten
> to use PyDict_Items, but with my current level of knowledge of writing
> Python extensions in C (None), anything I do to make that change would
> probably be *very* buggy.
>
> I wonder if a similar bug would be discovered if I was emulating
> a List type
> instead of a dictionary...
>
> In the meantime, I have two ideas for working around this, so
> it's not that
> big of a deal:
> - Passing in a real dictionary
> - htmlescaping the args inside of my fake
> dictionary, so that I can apply the %
> operator to a string instance instead
> of an htmltext instance
>
> But I still wanted to point this out to those who may be able to
> fix it with
> a reasonable amount of effort.
>
> If anyone wants a test case, I've attached a fairly thorough one that
> demonstrates behavior with a dict, a subclass of dict, and something
> pretending to be a dict. Everything in the script should work fine until
> line 125, at which point you'll get a KeyError (because the dict
> produced in
> _c_htmltext to be passed to the PyString_Format function is
> empty, since the
> PyDict_Next produced no entries.)
>
> 125: print c_htmltext % dict_emulator # <--- This will bomb out on a
> KeyError!
>
> I don't thinks it's relevant, but for the record, I'm using
> Python 2.3.3 on
> Linux for this.
>
> Jason Sibre
>
>
>