Index: third_party/logilab/common/tree.py |
diff --git a/third_party/logilab/common/tree.py b/third_party/logilab/common/tree.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..885eb0faa417f68441d5dad118b694cf9393dcb7 |
--- /dev/null |
+++ b/third_party/logilab/common/tree.py |
@@ -0,0 +1,369 @@ |
+# copyright 2003-2011 LOGILAB S.A. (Paris, FRANCE), all rights reserved. |
+# contact http://www.logilab.fr/ -- mailto:contact@logilab.fr |
+# |
+# This file is part of logilab-common. |
+# |
+# logilab-common is free software: you can redistribute it and/or modify it under |
+# the terms of the GNU Lesser General Public License as published by the Free |
+# Software Foundation, either version 2.1 of the License, or (at your option) any |
+# later version. |
+# |
+# logilab-common 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 Lesser General Public License for more |
+# details. |
+# |
+# You should have received a copy of the GNU Lesser General Public License along |
+# with logilab-common. If not, see <http://www.gnu.org/licenses/>. |
+"""Base class to represent a tree structure. |
+ |
+ |
+ |
+ |
+""" |
+__docformat__ = "restructuredtext en" |
+ |
+import sys |
+ |
+from logilab.common import flatten |
+from logilab.common.visitor import VisitedMixIn, FilteredIterator, no_filter |
+ |
+## Exceptions ################################################################# |
+ |
+class NodeNotFound(Exception): |
+ """raised when a node has not been found""" |
+ |
+EX_SIBLING_NOT_FOUND = "No such sibling as '%s'" |
+EX_CHILD_NOT_FOUND = "No such child as '%s'" |
+EX_NODE_NOT_FOUND = "No such node as '%s'" |
+ |
+ |
+# Base node ################################################################### |
+ |
+class Node(object): |
+ """a basic tree node, characterized by an id""" |
+ |
+ def __init__(self, nid=None) : |
+ self.id = nid |
+ # navigation |
+ self.parent = None |
+ self.children = [] |
+ |
+ def __iter__(self): |
+ return iter(self.children) |
+ |
+ def __str__(self, indent=0): |
+ s = ['%s%s %s' % (' '*indent, self.__class__.__name__, self.id)] |
+ indent += 2 |
+ for child in self.children: |
+ try: |
+ s.append(child.__str__(indent)) |
+ except TypeError: |
+ s.append(child.__str__()) |
+ return '\n'.join(s) |
+ |
+ def is_leaf(self): |
+ return not self.children |
+ |
+ def append(self, child): |
+ """add a node to children""" |
+ self.children.append(child) |
+ child.parent = self |
+ |
+ def remove(self, child): |
+ """remove a child node""" |
+ self.children.remove(child) |
+ child.parent = None |
+ |
+ def insert(self, index, child): |
+ """insert a child node""" |
+ self.children.insert(index, child) |
+ child.parent = self |
+ |
+ def replace(self, old_child, new_child): |
+ """replace a child node with another""" |
+ i = self.children.index(old_child) |
+ self.children.pop(i) |
+ self.children.insert(i, new_child) |
+ new_child.parent = self |
+ |
+ def get_sibling(self, nid): |
+ """return the sibling node that has given id""" |
+ try: |
+ return self.parent.get_child_by_id(nid) |
+ except NodeNotFound : |
+ raise NodeNotFound(EX_SIBLING_NOT_FOUND % nid) |
+ |
+ def next_sibling(self): |
+ """ |
+ return the next sibling for this node if any |
+ """ |
+ parent = self.parent |
+ if parent is None: |
+ # root node has no sibling |
+ return None |
+ index = parent.children.index(self) |
+ try: |
+ return parent.children[index+1] |
+ except IndexError: |
+ return None |
+ |
+ def previous_sibling(self): |
+ """ |
+ return the previous sibling for this node if any |
+ """ |
+ parent = self.parent |
+ if parent is None: |
+ # root node has no sibling |
+ return None |
+ index = parent.children.index(self) |
+ if index > 0: |
+ return parent.children[index-1] |
+ return None |
+ |
+ def get_node_by_id(self, nid): |
+ """ |
+ return node in whole hierarchy that has given id |
+ """ |
+ root = self.root() |
+ try: |
+ return root.get_child_by_id(nid, 1) |
+ except NodeNotFound : |
+ raise NodeNotFound(EX_NODE_NOT_FOUND % nid) |
+ |
+ def get_child_by_id(self, nid, recurse=None): |
+ """ |
+ return child of given id |
+ """ |
+ if self.id == nid: |
+ return self |
+ for c in self.children : |
+ if recurse: |
+ try: |
+ return c.get_child_by_id(nid, 1) |
+ except NodeNotFound : |
+ continue |
+ if c.id == nid : |
+ return c |
+ raise NodeNotFound(EX_CHILD_NOT_FOUND % nid) |
+ |
+ def get_child_by_path(self, path): |
+ """ |
+ return child of given path (path is a list of ids) |
+ """ |
+ if len(path) > 0 and path[0] == self.id: |
+ if len(path) == 1 : |
+ return self |
+ else : |
+ for c in self.children : |
+ try: |
+ return c.get_child_by_path(path[1:]) |
+ except NodeNotFound : |
+ pass |
+ raise NodeNotFound(EX_CHILD_NOT_FOUND % path) |
+ |
+ def depth(self): |
+ """ |
+ return depth of this node in the tree |
+ """ |
+ if self.parent is not None: |
+ return 1 + self.parent.depth() |
+ else : |
+ return 0 |
+ |
+ def depth_down(self): |
+ """ |
+ return depth of the tree from this node |
+ """ |
+ if self.children: |
+ return 1 + max([c.depth_down() for c in self.children]) |
+ return 1 |
+ |
+ def width(self): |
+ """ |
+ return the width of the tree from this node |
+ """ |
+ return len(self.leaves()) |
+ |
+ def root(self): |
+ """ |
+ return the root node of the tree |
+ """ |
+ if self.parent is not None: |
+ return self.parent.root() |
+ return self |
+ |
+ def leaves(self): |
+ """ |
+ return a list with all the leaves nodes descendant from this node |
+ """ |
+ leaves = [] |
+ if self.children: |
+ for child in self.children: |
+ leaves += child.leaves() |
+ return leaves |
+ else: |
+ return [self] |
+ |
+ def flatten(self, _list=None): |
+ """ |
+ return a list with all the nodes descendant from this node |
+ """ |
+ if _list is None: |
+ _list = [] |
+ _list.append(self) |
+ for c in self.children: |
+ c.flatten(_list) |
+ return _list |
+ |
+ def lineage(self): |
+ """ |
+ return list of parents up to root node |
+ """ |
+ lst = [self] |
+ if self.parent is not None: |
+ lst.extend(self.parent.lineage()) |
+ return lst |
+ |
+class VNode(Node, VisitedMixIn): |
+ """a visitable node |
+ """ |
+ pass |
+ |
+ |
+class BinaryNode(VNode): |
+ """a binary node (i.e. only two children |
+ """ |
+ def __init__(self, lhs=None, rhs=None) : |
+ VNode.__init__(self) |
+ if lhs is not None or rhs is not None: |
+ assert lhs and rhs |
+ self.append(lhs) |
+ self.append(rhs) |
+ |
+ def remove(self, child): |
+ """remove the child and replace this node with the other child |
+ """ |
+ self.children.remove(child) |
+ self.parent.replace(self, self.children[0]) |
+ |
+ def get_parts(self): |
+ """ |
+ return the left hand side and the right hand side of this node |
+ """ |
+ return self.children[0], self.children[1] |
+ |
+ |
+ |
+if sys.version_info[0:2] >= (2, 2): |
+ list_class = list |
+else: |
+ from UserList import UserList |
+ list_class = UserList |
+ |
+class ListNode(VNode, list_class): |
+ """Used to manipulate Nodes as Lists |
+ """ |
+ def __init__(self): |
+ list_class.__init__(self) |
+ VNode.__init__(self) |
+ self.children = self |
+ |
+ def __str__(self, indent=0): |
+ return '%s%s %s' % (indent*' ', self.__class__.__name__, |
+ ', '.join([str(v) for v in self])) |
+ |
+ def append(self, child): |
+ """add a node to children""" |
+ list_class.append(self, child) |
+ child.parent = self |
+ |
+ def insert(self, index, child): |
+ """add a node to children""" |
+ list_class.insert(self, index, child) |
+ child.parent = self |
+ |
+ def remove(self, child): |
+ """add a node to children""" |
+ list_class.remove(self, child) |
+ child.parent = None |
+ |
+ def pop(self, index): |
+ """add a node to children""" |
+ child = list_class.pop(self, index) |
+ child.parent = None |
+ |
+ def __iter__(self): |
+ return list_class.__iter__(self) |
+ |
+# construct list from tree #################################################### |
+ |
+def post_order_list(node, filter_func=no_filter): |
+ """ |
+ create a list with tree nodes for which the <filter> function returned true |
+ in a post order fashion |
+ """ |
+ l, stack = [], [] |
+ poped, index = 0, 0 |
+ while node: |
+ if filter_func(node): |
+ if node.children and not poped: |
+ stack.append((node, index)) |
+ index = 0 |
+ node = node.children[0] |
+ else: |
+ l.append(node) |
+ index += 1 |
+ try: |
+ node = stack[-1][0].children[index] |
+ except IndexError: |
+ node = None |
+ else: |
+ node = None |
+ poped = 0 |
+ if node is None and stack: |
+ node, index = stack.pop() |
+ poped = 1 |
+ return l |
+ |
+def pre_order_list(node, filter_func=no_filter): |
+ """ |
+ create a list with tree nodes for which the <filter> function returned true |
+ in a pre order fashion |
+ """ |
+ l, stack = [], [] |
+ poped, index = 0, 0 |
+ while node: |
+ if filter_func(node): |
+ if not poped: |
+ l.append(node) |
+ if node.children and not poped: |
+ stack.append((node, index)) |
+ index = 0 |
+ node = node.children[0] |
+ else: |
+ index += 1 |
+ try: |
+ node = stack[-1][0].children[index] |
+ except IndexError: |
+ node = None |
+ else: |
+ node = None |
+ poped = 0 |
+ if node is None and len(stack) > 1: |
+ node, index = stack.pop() |
+ poped = 1 |
+ return l |
+ |
+class PostfixedDepthFirstIterator(FilteredIterator): |
+ """a postfixed depth first iterator, designed to be used with visitors |
+ """ |
+ def __init__(self, node, filter_func=None): |
+ FilteredIterator.__init__(self, node, post_order_list, filter_func) |
+ |
+class PrefixedDepthFirstIterator(FilteredIterator): |
+ """a prefixed depth first iterator, designed to be used with visitors |
+ """ |
+ def __init__(self, node, filter_func=None): |
+ FilteredIterator.__init__(self, node, pre_order_list, filter_func) |
+ |