| # Copyright (c) 2005 The Regents of The University of Michigan |
| # All rights reserved. |
| # |
| # Redistribution and use in source and binary forms, with or without |
| # modification, are permitted provided that the following conditions are |
| # met: redistributions of source code must retain the above copyright |
| # notice, this list of conditions and the following disclaimer; |
| # redistributions in binary form must reproduce the above copyright |
| # notice, this list of conditions and the following disclaimer in the |
| # documentation and/or other materials provided with the distribution; |
| # neither the name of the copyright holders nor the names of its |
| # contributors may be used to endorse or promote products derived from |
| # this software without specific prior written permission. |
| # |
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| __all__ = ["multidict"] |
| |
| |
| class multidict(object): |
| def __init__(self, parent={}, **kwargs): |
| self.local = dict(**kwargs) |
| self.parent = parent |
| self.deleted = {} |
| |
| def __str__(self): |
| return str(dict(self.items())) |
| |
| def __repr__(self): |
| return repr(dict(list(self.items()))) |
| |
| def __contains__(self, key): |
| return key in self.local or key in self.parent |
| |
| def __delitem__(self, key): |
| try: |
| del self.local[key] |
| except KeyError as e: |
| if key in self.parent: |
| self.deleted[key] = True |
| else: |
| raise KeyError(e) |
| |
| def __setitem__(self, key, value): |
| self.deleted.pop(key, False) |
| self.local[key] = value |
| |
| def __getitem__(self, key): |
| try: |
| return self.local[key] |
| except KeyError as e: |
| if not self.deleted.get(key, False) and key in self.parent: |
| return self.parent[key] |
| else: |
| raise KeyError(e) |
| |
| def __len__(self): |
| return len(self.local) + len(self.parent) |
| |
| def next(self): |
| for key, value in self.local.items(): |
| yield key, value |
| |
| if self.parent: |
| for key, value in self.parent.next(): |
| if key not in self.local and key not in self.deleted: |
| yield key, value |
| |
| def has_key(self, key): |
| return key in self |
| |
| def items(self): |
| for item in self.next(): |
| yield item |
| |
| def keys(self): |
| for key, value in self.next(): |
| yield key |
| |
| def values(self): |
| for key, value in self.next(): |
| yield value |
| |
| def get(self, key, default=None): |
| try: |
| return self[key] |
| except KeyError as e: |
| return default |
| |
| def setdefault(self, key, default): |
| try: |
| return self[key] |
| except KeyError: |
| self.deleted.pop(key, False) |
| self.local[key] = default |
| return default |
| |
| def _dump(self): |
| print("multidict dump") |
| node = self |
| while isinstance(node, multidict): |
| print(" ", node.local) |
| node = node.parent |
| |
| def _dumpkey(self, key): |
| values = [] |
| node = self |
| while isinstance(node, multidict): |
| if key in node.local: |
| values.append(node.local[key]) |
| node = node.parent |
| print(key, values) |
| |
| |
| if __name__ == "__main__": |
| test1 = multidict() |
| test2 = multidict(test1) |
| test3 = multidict(test2) |
| test4 = multidict(test3) |
| |
| test1["a"] = "test1_a" |
| test1["b"] = "test1_b" |
| test1["c"] = "test1_c" |
| test1["d"] = "test1_d" |
| test1["e"] = "test1_e" |
| |
| test2["a"] = "test2_a" |
| del test2["b"] |
| test2["c"] = "test2_c" |
| del test1["a"] |
| |
| test2.setdefault("f", multidict) |
| |
| print("test1>", list(test1.items())) |
| print("test2>", list(test2.items())) |
| # print(test1['a']) |
| print(test1["b"]) |
| print(test1["c"]) |
| print(test1["d"]) |
| print(test1["e"]) |
| |
| print(test2["a"]) |
| # print(test2['b']) |
| print(test2["c"]) |
| print(test2["d"]) |
| print(test2["e"]) |
| |
| for key in test2.keys(): |
| print(key) |
| |
| test2.get("g", "foo") |
| # test2.get('b') |
| test2.get("b", "bar") |
| test2.setdefault("b", "blah") |
| print(test1) |
| print(test2) |
| print(repr(test2)) |
| |
| print(len(test2)) |
| |
| test3["a"] = [0, 1, 2, 3] |
| |
| print(test4) |