added index package for multikey dictionaries
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@1506 fd906abe-77d9-0310-91a1-e0d9ade77398
This commit is contained in:
parent
589238c76f
commit
b49cc08b36
4 changed files with 173 additions and 0 deletions
56
index/README.txt
Normal file
56
index/README.txt
Normal file
|
@ -0,0 +1,56 @@
|
|||
========================================
|
||||
Indexed Collections for Various Purposes
|
||||
========================================
|
||||
|
||||
Multikey Dictionaries
|
||||
=====================
|
||||
|
||||
A MultiKeyDict is a dictionary that expects its keys to be tuples.
|
||||
|
||||
>>> from cybertools.index.multikey import MultiKeyDict
|
||||
>>> registry = MultiKeyDict()
|
||||
|
||||
>>> registry[('index.html',)] = 'global index.html'
|
||||
|
||||
>>> registry[('index.html',)]
|
||||
'global index.html'
|
||||
|
||||
So this would be nothing special - any dictionary is able to provide this
|
||||
functionality; but a MultiKeyDict has some fallback mechanisms for retrieving
|
||||
objects only partly fitting the requested key:
|
||||
|
||||
>>> registry.get(('index.html', 'topic', 'zope3', 'Custom'))
|
||||
'global index.html'
|
||||
|
||||
>>> registry[('index.html', 'topic', 'zope3', 'Custom')]
|
||||
'global index.html'
|
||||
|
||||
>>> registry[('index.html', 'topic',)] = 'index.html for type "topic"'
|
||||
|
||||
>>> registry[('index.html', 'topic', 'zope3', 'Custom')]
|
||||
'index.html for type "topic"'
|
||||
|
||||
It is also possible to keep intermediate parts of a key variable by
|
||||
setting them to None:
|
||||
|
||||
>>> registry[('index.html', None, None, 'Custom')] = 'Global index.html for Custom skin'
|
||||
|
||||
The more on the left a matching key part is the higher is its priority:
|
||||
|
||||
>>> registry[('index.html', 'topic', 'zope3', 'Custom')]
|
||||
'index.html for type "topic"'
|
||||
|
||||
>>> registry[('index.html', 'task', 'bugfixes', 'Custom')]
|
||||
'Global index.html for Custom skin'
|
||||
|
||||
|
||||
>>> registry[('edit.html', 'topic', 'zope3', 'Custom')] = 'very special edit.html'
|
||||
|
||||
>>> registry[('index.html', 'task', 'bugfixes', 'Custom')]
|
||||
'Global index.html for Custom skin'
|
||||
|
||||
>>> registry[('index.html', 'topic', 'zope3', 'Custom')]
|
||||
'index.html for type "topic"'
|
||||
|
||||
>>> registry.get(('edit.html', 'task', 'bugfixes', 'Custom'))
|
||||
|
3
index/__init__.py
Normal file
3
index/__init__.py
Normal file
|
@ -0,0 +1,3 @@
|
|||
"""
|
||||
$Id$
|
||||
"""
|
84
index/multikey.py
Normal file
84
index/multikey.py
Normal file
|
@ -0,0 +1,84 @@
|
|||
#
|
||||
# Copyright (c) 2006 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
|
||||
#
|
||||
|
||||
"""
|
||||
Dictionaries with multiple keys.
|
||||
|
||||
$Id$
|
||||
"""
|
||||
|
||||
_not_found = object()
|
||||
_default = object()
|
||||
|
||||
class MultiKeyDict(dict):
|
||||
|
||||
def __init__(self, **kw):
|
||||
super(MultiKeyDict, self).__init__(**kw)
|
||||
self.singleKeyDict = {}
|
||||
|
||||
def __setitem__(self, key, value):
|
||||
assert type(key) is tuple
|
||||
super(MultiKeyDict, self).__setitem__(key, value)
|
||||
for n, k in enumerate(key):
|
||||
if k:
|
||||
entry = self.singleKeyDict.setdefault((n, k), [])
|
||||
if value not in entry:
|
||||
entry.append((key, value))
|
||||
|
||||
def __getitem__(self, key):
|
||||
r = self.get(key, _default)
|
||||
if r is _default:
|
||||
raise KeyError(key)
|
||||
return r
|
||||
|
||||
def get(self, key, default=None):
|
||||
assert type(key) is tuple
|
||||
kl = list(key)
|
||||
while kl:
|
||||
r = super(MultiKeyDict, self).get(tuple(kl), _not_found)
|
||||
if r is not _not_found:
|
||||
return r
|
||||
kl.pop()
|
||||
return default
|
||||
|
||||
def get(self, key, default=None):
|
||||
assert type(key) is tuple
|
||||
firstTry = super(MultiKeyDict, self).get(key, _not_found)
|
||||
# fast return for full match:
|
||||
if firstTry is not _not_found:
|
||||
return firstTry
|
||||
collector = {}
|
||||
for n, k in enumerate(key):
|
||||
rList = self.singleKeyDict.get((n, k), [])
|
||||
for r in rList:
|
||||
skip = False
|
||||
for nx, kx in enumerate(r[0]):
|
||||
if kx and kx != key[nx]: # if stored key elements are present
|
||||
skip = True # they must match
|
||||
break
|
||||
if skip:
|
||||
continue
|
||||
entry = collector.setdefault(r[1], [])
|
||||
entry.append(n)
|
||||
if not collector:
|
||||
return default
|
||||
#print 'collector', collector
|
||||
results = sorted((-len(value), value, o) for o, value in collector.items())
|
||||
#print 'sorted', results
|
||||
return results[0][2]
|
||||
|
30
index/tests.py
Executable file
30
index/tests.py
Executable file
|
@ -0,0 +1,30 @@
|
|||
#! /usr/bin/python
|
||||
|
||||
"""
|
||||
Tests for the 'cybertools.index' package.
|
||||
|
||||
$Id$
|
||||
"""
|
||||
|
||||
import unittest, doctest
|
||||
from zope.testing.doctestunit import DocFileSuite
|
||||
|
||||
from cybertools.index import multikey
|
||||
|
||||
|
||||
class Test(unittest.TestCase):
|
||||
"Basic tests for the index package."
|
||||
|
||||
def testBasicStuff(self):
|
||||
pass
|
||||
|
||||
|
||||
def test_suite():
|
||||
flags = doctest.NORMALIZE_WHITESPACE | doctest.ELLIPSIS
|
||||
return unittest.TestSuite((
|
||||
DocFileSuite('README.txt', optionflags=flags),
|
||||
unittest.makeSuite(Test),
|
||||
))
|
||||
|
||||
if __name__ == '__main__':
|
||||
unittest.main(defaultTest='test_suite')
|
Loading…
Add table
Reference in a new issue