| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 |
- # 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)
|