-
Notifications
You must be signed in to change notification settings - Fork 177
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
0 parents
commit e4f1550
Showing
3 changed files
with
285 additions
and
0 deletions.
There are no files selected for viewing
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,264 @@ | ||
""" bunch provides Bunch, a subclass of dictionary with attribute-style access, | ||
and un/bunchify(), two utility functions for dictionary conversion. Bunches | ||
can also be converted Bunch.to/fromDict(). | ||
>>> b = Bunch() | ||
>>> b.hello = 'world' | ||
>>> b.hello | ||
'world' | ||
>>> b['hello'] += "!" | ||
>>> b.hello | ||
'world!' | ||
>>> b.foo = Bunch(lol=True) | ||
>>> b.foo.lol | ||
True | ||
>>> b.foo is b['foo'] | ||
True | ||
""" | ||
|
||
__all__ = ('Bunch', 'bunchify',) | ||
|
||
|
||
class Bunch(dict): | ||
""" A dictionary that provides attribute-style access. | ||
>>> b = Bunch() | ||
>>> b.hello = 'world' | ||
>>> b.hello | ||
'world' | ||
>>> b['hello'] += "!" | ||
>>> b.hello | ||
'world!' | ||
>>> b.foo = Bunch(lol=True) | ||
>>> b.foo.lol | ||
True | ||
>>> b.foo is b['foo'] | ||
True | ||
A Bunch is a subclass of dict; it supports all the methods a dict does... | ||
>>> b.keys() | ||
['foo', 'hello'] | ||
Including update()... | ||
>>> b.update({ 'ponies': 'are pretty!' }, hello=42) | ||
>>> print repr(b) | ||
Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
As well as iteration... | ||
>>> [ (k,b[k]) for k in b ] | ||
[('ponies', 'are pretty!'), ('foo', Bunch(lol=True)), ('hello', 42)] | ||
And "splats". | ||
>>> "The {knights} who say {ni}!".format(**Bunch(knights='lolcats', ni='can haz')) | ||
'The lolcats who say can haz!' | ||
See Bunch.toDict, Bunch.fromDict, and bunchify for notes about conversion. | ||
""" | ||
|
||
def __contains__(self, k): | ||
""" >>> b = Bunch(ponies='are pretty!') | ||
>>> 'ponies' in b | ||
True | ||
>>> 'foo' in b | ||
False | ||
>>> b['foo'] = 42 | ||
>>> 'foo' in b | ||
True | ||
>>> b.hello = 'hai' | ||
>>> 'hello' in b | ||
True | ||
""" | ||
try: | ||
return hasattr(self, k) or dict.__contains__(self, k) | ||
except: | ||
return False | ||
|
||
# only called if k not found in normal places | ||
def __getattr__(self, k): | ||
""" Gets key if it exists, otherwise throws AttributeError. | ||
nb. __getattr__ is only called if key is not found in normal places. | ||
>>> b = Bunch(bar='baz', lol={}) | ||
>>> b.foo | ||
Traceback (most recent call last): | ||
... | ||
AttributeError: foo | ||
>>> b.bar | ||
'baz' | ||
>>> getattr(b, 'bar') | ||
'baz' | ||
>>> b['bar'] | ||
'baz' | ||
>>> b.lol is b['lol'] | ||
True | ||
>>> b.lol is getattr(b, 'lol') | ||
True | ||
""" | ||
try: | ||
return self[k] | ||
except KeyError: | ||
raise AttributeError(k) | ||
|
||
def __setattr__(self, k, v): | ||
""" Sets attribute k if it exists, otherwise sets key k. A KeyError | ||
raised by set-item (only likely if you subclass Bunch) will | ||
propagate as an AttributeError instead. | ||
>>> b = Bunch(foo='bar', this_is='useful when subclassing') | ||
>>> b.values #doctest: +ELLIPSIS | ||
<built-in method values of Bunch object at 0x...> | ||
>>> b.values = 'uh oh' | ||
>>> b.values | ||
'uh oh' | ||
>>> b['values'] | ||
Traceback (most recent call last): | ||
... | ||
KeyError: 'values' | ||
""" | ||
try: | ||
# Throws exception if not in prototype chain | ||
object.__getattribute__(self, k) | ||
except AttributeError: | ||
try: | ||
self[k] = v | ||
except: | ||
raise AttributeError(k) | ||
else: | ||
object.__setattr__(self, k, v) | ||
|
||
def __delattr__(self, k): | ||
""" Deletes attribute k if it exists, otherwise deletes key k. A KeyError | ||
raised by deleting the key--such as when the key is missing--will | ||
propagate as an AttributeError instead. | ||
>>> b = Bunch(lol=42) | ||
>>> del b.values | ||
Traceback (most recent call last): | ||
... | ||
AttributeError: 'Bunch' object attribute 'values' is read-only | ||
>>> del b.lol | ||
>>> b.lol | ||
Traceback (most recent call last): | ||
... | ||
AttributeError: lol | ||
""" | ||
try: | ||
# Throws exception if not in prototype chain | ||
object.__getattribute__(self, k) | ||
except AttributeError: | ||
try: | ||
del self[k] | ||
except KeyError: | ||
raise AttributeError(k) | ||
else: | ||
object.__delattr__(self, k) | ||
|
||
def toDict(self): | ||
""" Recursively converts a bunch back into a dictionary. | ||
>>> b = Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
>>> b.toDict() | ||
{'ponies': 'are pretty!', 'foo': {'lol': True}, 'hello': 42} | ||
See unbunchify for more info. | ||
""" | ||
return unbunchify(self) | ||
|
||
def __repr__(self): | ||
""" Invertible* string-form of a Bunch. | ||
>>> b = Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
>>> print repr(b) | ||
Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
>>> eval(repr(b)) | ||
Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
(*) Invertible so long as collection contents are each repr-invertible. | ||
""" | ||
keys = self.keys() | ||
keys.sort() | ||
args = ', '.join(['%s=%r' % (key, self[key]) for key in keys]) | ||
return '%s(%s)' % (self.__class__.__name__, args) | ||
|
||
@staticmethod | ||
def fromDict(d): | ||
""" Recursively transforms a dictionary into a Bunch via copy. | ||
>>> b = Bunch.fromDict({'urmom': {'sez': {'what': 'what'}}}) | ||
>>> b.urmom.sez.what | ||
'what' | ||
See bunchify for more info. | ||
""" | ||
return bunchify(d) | ||
|
||
|
||
|
||
# While we could convert abstract types like Mapping or Iterable, I think | ||
# bunchify is more likely to "do what you mean" if it is conservative about | ||
# casting (ex: isinstance(str,Iterable) == True ). | ||
# | ||
# Should you disagree, it is not difficult to duplicate this function with | ||
# more aggressive coercion to suit your own purposes. | ||
|
||
def bunchify(x): | ||
""" Recursively transforms a dictionary into a Bunch via copy. | ||
>>> b = bunchify({'urmom': {'sez': {'what': 'what'}}}) | ||
>>> b.urmom.sez.what | ||
'what' | ||
bunchify can handle intermediary dicts, lists and tuples (as well as | ||
their subclasses), but ymmv on custom datatypes. | ||
>>> b = bunchify({ 'lol': ('cats', {'hah':'i win again'}), 'hello': [{'french':'salut', 'german':'hallo'}] }) | ||
>>> b.hello[0].french | ||
'salut' | ||
>>> b.lol[1].hah | ||
'i win again' | ||
nb. As dicts are not hashable, they cannot be nested in sets/frozensets. | ||
""" | ||
if isinstance(x, dict): | ||
return Bunch( (k, bunchify(v)) for k,v in x.iteritems() ) | ||
elif isinstance(x, (list, tuple)): | ||
return type(x)( bunchify(v) for v in x ) | ||
else: | ||
return x | ||
|
||
def unbunchify(x): | ||
""" Recursively converts a Bunch into a dictionary. | ||
>>> b = Bunch(foo=Bunch(lol=True), hello=42, ponies='are pretty!') | ||
>>> unbunchify(b) | ||
{'ponies': 'are pretty!', 'foo': {'lol': True}, 'hello': 42} | ||
unbunchify will handle intermediary dicts, lists and tuples (as well as | ||
their subclasses), but ymmv on custom datatypes. | ||
>>> b = Bunch(foo=['bar', Bunch(lol=True)], hello=42, ponies=('are pretty!', Bunch(lies='are trouble!'))) | ||
>>> unbunchify(b) | ||
{'ponies': ('are pretty!', {'lies': 'are trouble!'}), 'foo': ['bar', {'lol': True}], 'hello': 42} | ||
nb. As dicts are not hashable, they cannot be nested in sets/frozensets. | ||
""" | ||
if isinstance(x, dict): | ||
return dict( (k, unbunchify(v)) for k,v in x.iteritems() ) | ||
elif isinstance(x, (list, tuple)): | ||
return type(x)( unbunchify(v) for v in x ) | ||
else: | ||
return x | ||
|
||
|
||
|
||
|
||
if __name__ == "__main__": | ||
import doctest | ||
doctest.testmod() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
#! python | ||
from setuptools import setup, find_packages | ||
|
||
setup( | ||
name = "bunch", | ||
description = "A dot-accessible dictionary (a la JavaScript objects)", | ||
version = "1.0.0", | ||
license = 'MIT', | ||
author = "David Schoonover", | ||
author_email = "[email protected]", | ||
long_description = """ | ||
A dot-accessible dictionary (a la JavaScript objects). | ||
""", | ||
url = "http://tire.less.ly/hacking/bunch", | ||
platform = 'Any', | ||
|
||
packages=['bunch',], | ||
zip_safe = True, | ||
classifiers = [ | ||
] | ||
) |