cybertools/zutil/jeep.py
helmutm 242c4a4921 extending composer and organize.schema to provide basis for tumsm
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@1845 fd906abe-77d9-0310-91a1-e0d9ade77398
2007-07-29 13:28:50 +00:00

132 lines
3.8 KiB
Python

#
# Copyright (c) 2007 Helmut Merz helmutm@cy55.de
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
"""
A general purpose (thus 'Jeep') class that provides most of the interfaces
of sequences and dictionaries and in addition allows attribute access to
the dictionary entries.
This is the Zope-based persistent variant of the Jeep class.
$Id$
"""
from persistent import Persistent
from persistent.list import PersistentList
from BTrees.OOBTree import OOBTree
from zope.app.container.interfaces import IContainer
from zope.interface import implements
_notfound = object()
_nodefault = object()
class Jeep(Persistent):
implements(IContainer)
_attributes = ('_sequence', '_mapping')
def __init__(self, seq=[]):
sequence = self._sequence = PersistentList()
mapping = self._mapping = OOBTree()
for attr, value in seq:
sequence.append(attr)
mapping[attr] = value
def __len__(self):
return len(self._sequence)
def __iter__(self):
for key in self._sequence:
yield self[key]
def __getattr__(self, attr, default=_nodefault):
value = self._mapping.get(attr, _notfound)
if value is _notfound:
if default is _nodefault:
raise AttributeError(attr)
else:
return default
return value
def __setattr__(self, attr, value):
if attr in self._attributes:
super(Jeep, self).__setattr__(attr, value)
else:
if getattr(self, attr, _notfound) is _notfound:
self._sequence.append(attr)
self._mapping[attr] = value
def __delattr__(self, attr):
del self._sequence[self.index(attr)]
del self._mapping[attr]
def __getitem__(self, key):
if type(key) in (int, long):
return getattr(self, self._sequence[key])
value = getattr(self, key, _notfound)
if value is _notfound:
raise KeyError(key)
return value
def __setitem__(self, key, value):
setattr(self, key, value)
def __delitem__(self, key):
delattr(self, key)
def __contains__(self, key):
return getattr(self, key, _notfound) is not _notfound
has_key = __contains__
def keys(self):
return [key for key in self._sequence]
def values(self):
return list(self)
def items(self):
return [(k, self[k]) for k in self._sequence]
def get(self, key, default=None):
return getattr(self, key, default)
def index(self, key):
return self._sequence.index(key)
def append(self, obj):
self.insert(len(self), obj)
def insert(self, idx, obj):
key = getattr(obj, '__name__', getattr(obj, 'name', _notfound))
if key is _notfound:
raise AttributeError("No name attribute present")
if key in self:
raise ValueError("Object already present")
self._sequence.insert(idx, key)
self._mapping[key] = obj
def pop(self, key=-1):
value = self[key]
if type(key) in (int, long):
key = self._sequence[key]
delattr(self, key)
return value