|
|
|
# ~*~ coding: utf-8 ~*~
|
|
|
|
#
|
|
|
|
from treelib import Tree
|
|
|
|
from treelib.exceptions import NodeIDAbsentError
|
|
|
|
from collections import defaultdict
|
|
|
|
from copy import deepcopy
|
|
|
|
|
|
|
|
from common.utils import get_logger, timeit, lazyproperty
|
|
|
|
from .models import Asset, Node
|
|
|
|
|
|
|
|
|
|
|
|
logger = get_logger(__file__)
|
|
|
|
|
|
|
|
|
|
|
|
class TreeService(Tree):
|
|
|
|
tag_sep = ' / '
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@timeit
|
|
|
|
def get_nodes_assets_map():
|
|
|
|
nodes_assets_map = defaultdict(set)
|
|
|
|
asset_node_list = Node.assets.through.objects.values_list(
|
|
|
|
'asset', 'node__key'
|
|
|
|
)
|
|
|
|
for asset_id, key in asset_node_list:
|
|
|
|
nodes_assets_map[key].add(asset_id)
|
|
|
|
return nodes_assets_map
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
@timeit
|
|
|
|
def new(cls):
|
|
|
|
from .models import Node
|
|
|
|
all_nodes = list(Node.objects.all().values("key", "value"))
|
|
|
|
all_nodes.sort(key=lambda x: len(x["key"].split(":")))
|
|
|
|
tree = cls()
|
|
|
|
tree.create_node(tag='', identifier='', data={})
|
|
|
|
for node in all_nodes:
|
|
|
|
key = node["key"]
|
|
|
|
value = node["value"]
|
|
|
|
parent_key = ":".join(key.split(":")[:-1])
|
|
|
|
tree.safe_create_node(
|
|
|
|
tag=value, identifier=key,
|
|
|
|
parent=parent_key,
|
|
|
|
)
|
|
|
|
tree.init_assets()
|
|
|
|
return tree
|
|
|
|
|
|
|
|
def init_assets(self):
|
|
|
|
node_assets_map = self.get_nodes_assets_map()
|
|
|
|
for node in self.all_nodes_itr():
|
|
|
|
key = node.identifier
|
|
|
|
assets = node_assets_map.get(key, set())
|
|
|
|
data = {"assets": assets, "all_assets": None}
|
|
|
|
node.data = data
|
|
|
|
|
|
|
|
def safe_create_node(self, **kwargs):
|
|
|
|
parent = kwargs.get("parent")
|
|
|
|
if not self.contains(parent):
|
|
|
|
kwargs['parent'] = self.root
|
|
|
|
self.create_node(**kwargs)
|
|
|
|
|
|
|
|
def all_children_ids(self, nid, with_self=True):
|
|
|
|
children_ids = self.expand_tree(nid)
|
|
|
|
if not with_self:
|
|
|
|
next(children_ids)
|
|
|
|
return list(children_ids)
|
|
|
|
|
|
|
|
def all_children(self, nid, with_self=True, deep=False):
|
|
|
|
children_ids = self.all_children_ids(nid, with_self=with_self)
|
|
|
|
return [self.get_node(i, deep=deep) for i in children_ids]
|
|
|
|
|
|
|
|
def ancestors_ids(self, nid, with_self=True):
|
|
|
|
ancestor_ids = list(self.rsearch(nid))
|
|
|
|
ancestor_ids.pop()
|
|
|
|
if not with_self:
|
|
|
|
ancestor_ids.pop(0)
|
|
|
|
return ancestor_ids
|
|
|
|
|
|
|
|
def ancestors(self, nid, with_self=False, deep=False):
|
|
|
|
ancestor_ids = self.ancestors_ids(nid, with_self=with_self)
|
|
|
|
ancestors = [self.get_node(i, deep=deep) for i in ancestor_ids]
|
|
|
|
return ancestors
|
|
|
|
|
|
|
|
def get_node_full_tag(self, nid):
|
|
|
|
ancestors = self.ancestors(nid, with_self=True)
|
|
|
|
ancestors.reverse()
|
|
|
|
return self.tag_sep.join([n.tag for n in ancestors])
|
|
|
|
|
|
|
|
def get_family(self, nid, deep=False):
|
|
|
|
ancestors = self.ancestors(nid, with_self=False, deep=deep)
|
|
|
|
children = self.all_children(nid, with_self=False)
|
|
|
|
return ancestors + [self[nid]] + children
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def is_parent(child, parent):
|
|
|
|
parent_id = child.bpointer
|
|
|
|
return parent_id == parent.identifier
|
|
|
|
|
|
|
|
def root_node(self):
|
|
|
|
return self.get_node(self.root)
|
|
|
|
|
|
|
|
def get_node(self, nid, deep=False):
|
|
|
|
node = super().get_node(nid)
|
|
|
|
if deep:
|
|
|
|
node = self.copy_node(node)
|
|
|
|
node.data = {}
|
|
|
|
return node
|
|
|
|
|
|
|
|
def parent(self, nid, deep=False):
|
|
|
|
parent = super().parent(nid)
|
|
|
|
if deep:
|
|
|
|
parent = self.copy_node(parent)
|
|
|
|
return parent
|
|
|
|
|
|
|
|
@lazyproperty
|
|
|
|
def invalid_assets(self):
|
|
|
|
assets = Asset.objects.filter(is_active=False).values_list('id', flat=True)
|
|
|
|
return assets
|
|
|
|
|
|
|
|
def set_assets(self, nid, assets):
|
|
|
|
node = self.get_node(nid)
|
|
|
|
if node.data is None:
|
|
|
|
node.data = {}
|
|
|
|
node.data["assets"] = assets
|
|
|
|
|
|
|
|
def assets(self, nid):
|
|
|
|
node = self.get_node(nid)
|
|
|
|
return node.data.get("assets", set())
|
|
|
|
|
|
|
|
def valid_assets(self, nid):
|
|
|
|
return set(self.assets(nid)) - set(self.invalid_assets)
|
|
|
|
|
|
|
|
def all_assets(self, nid):
|
|
|
|
node = self.get_node(nid)
|
|
|
|
if node.data is None:
|
|
|
|
node.data = {}
|
|
|
|
all_assets = node.data.get("all_assets")
|
|
|
|
if all_assets is not None:
|
|
|
|
return all_assets
|
|
|
|
all_assets = set(self.assets(nid))
|
|
|
|
try:
|
|
|
|
children = self.children(nid)
|
|
|
|
except NodeIDAbsentError:
|
|
|
|
children = []
|
|
|
|
for child in children:
|
|
|
|
all_assets.update(self.all_assets(child.identifier))
|
|
|
|
node.data["all_assets"] = all_assets
|
|
|
|
return all_assets
|
|
|
|
|
|
|
|
def all_valid_assets(self, nid):
|
|
|
|
return set(self.all_assets(nid)) - set(self.invalid_assets)
|
|
|
|
|
|
|
|
def assets_amount(self, nid):
|
|
|
|
return len(self.all_assets(nid))
|
|
|
|
|
|
|
|
def valid_assets_amount(self, nid):
|
|
|
|
return len(self.all_valid_assets(nid))
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def copy_node(node):
|
|
|
|
new_node = deepcopy(node)
|
|
|
|
new_node.fpointer = None
|
|
|
|
return new_node
|
|
|
|
|
|
|
|
def safe_add_ancestors(self, node, ancestors):
|
|
|
|
# 如果没有祖先节点,那么添加该节点, 父节点是root node
|
|
|
|
if len(ancestors) == 0:
|
|
|
|
parent = self.root_node()
|
|
|
|
else:
|
|
|
|
parent = ancestors[0]
|
|
|
|
|
|
|
|
# 如果当前节点已再树中,则移动当前节点到父节点中
|
|
|
|
# 这个是由于 当前节点放到了二级节点中
|
|
|
|
if not self.contains(parent.identifier):
|
|
|
|
# logger.debug('Add parent: {}'.format(parent.identifier))
|
|
|
|
self.safe_add_ancestors(parent, ancestors[1:])
|
|
|
|
|
|
|
|
if self.contains(node.identifier):
|
|
|
|
# msg = 'Move node to parent: {} => {}'.format(
|
|
|
|
# node.identifier, parent.identifier
|
|
|
|
# )
|
|
|
|
# logger.debug(msg)
|
|
|
|
self.move_node(node.identifier, parent.identifier)
|
|
|
|
else:
|
|
|
|
# logger.debug('Add node: {}'.format(node.identifier))
|
|
|
|
self.add_node(node, parent)
|
|
|
|
#
|
|
|
|
# def __getstate__(self):
|
|
|
|
# self.mutex = None
|
|
|
|
# self.all_nodes_assets_map = {}
|
|
|
|
# self.nodes_assets_map = {}
|
|
|
|
# return self.__dict__
|
|
|
|
|
|
|
|
# def __setstate__(self, state):
|
|
|
|
# self.__dict__ = state
|