cybertools/util/jeep.py
helmutm c2df42555e work in progress: cybertools.meta package for managing general meta-information, e.g. for configuration settings and preferences
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@2516 fd906abe-77d9-0310-91a1-e0d9ade77398
2008-04-10 16:52:51 +00:00

193 lines
5.7 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.
$Id$
"""
_notfound = object()
class Jeep(object):
_attributes = ('_sequence',)
def __init__(self, seq=[]):
if isinstance(seq, Jeep):
self._sequence = list(seq.keys())
for key, value in seq.items():
object.__setattr__(self, key, value)
else:
sequence = self._sequence = []
for item in seq:
if isinstance(item, (list, tuple)) and len(item) == 2:
attr, value = item
sequence.append(attr)
object.__setattr__(self, attr, value)
else:
self.append(item)
def __len__(self):
return len(self._sequence)
def __iter__(self):
for key in self._sequence:
yield self[key]
def __setattr__(self, attr, value):
if not attr in self._attributes:
if getattr(self, attr, _notfound) is _notfound:
self._sequence.append(attr)
super(Jeep, self).__setattr__(attr, value)
def __delattr__(self, attr):
del self._sequence[self.index(attr)]
super(Jeep, self).__delattr__(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
def keys(self):
return list(self._sequence)
def values(self):
return list(self)
return [self[k] for k in 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 setdefault(self, key, value):
existing = self.get(key, default=_notfound)
if existing is _notfound:
self[key] = value
return value
return existing
def append(self, obj):
self.insert(len(self), obj)
def extend(self, sequence):
for obj in sequence:
self.append(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.keys():
raise ValueError("Object '%s' already present" % key)
self._sequence.insert(idx, key)
object.__setattr__(self, key, obj)
def update(self, mapping):
for key, value in mapping.items():
self[key] = value
def pop(self, key=-1):
value = self[key]
if type(key) in (int, long):
key = self._sequence[key]
delattr(self, key)
return value
def find(self, obj):
if isinstance(obj, basestring):
key = obj
else:
key = getattr(obj, '__name__', getattr(obj, 'name', _notfound))
if key is _notfound:
raise AttributeError("No name attribute present")
if key in self:
return self._sequence.index(key)
else:
return -1
def index(self, obj):
idx = self.find(obj)
if idx < 0:
raise ValueError('list.index(x): %r not in list' % obj)
return idx
def remove(self, *keys):
myKeys = self.keys()
for key in keys:
if key in myKeys:
del self[key]
def select(self, *keys):
myKeys = self._sequence
self._sequence = [k for k in keys if k in myKeys]
for k in myKeys:
if k not in keys:
super(Jeep, self).__delattr__(k)
def reorder(self, delta, *keys):
self._sequence = moveByDelta(self._sequence, keys, delta)
class Term(object):
""" A simple name/title association that may be put in a jeep object.
"""
def __init__(self, name, title, **kw):
self.name = self.__name__ = self.value = self.token = name
self.title = title
for k, v in kw.items():
setattr(self, k, v)
def __str__(self):
return self.title
def moveByDelta(objs, toMove, delta):
""" Return the list given by objs re-ordered in a way that the elements
of toMove (which must be in the objs list) have been moved by delta.
"""
result = [obj for obj in objs if obj not in toMove]
if delta < 0:
objs = list(reversed(objs))
result.reverse()
toMove = sorted(toMove, lambda x,y: cmp(objs.index(x), objs.index(y)))
for element in toMove:
newPos = min(len(result), objs.index(element) + abs(delta))
result.insert(newPos, element)
if delta < 0:
result.reverse()
return result