mirror of https://github.com/jumpserver/jumpserver
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
106 lines
3.1 KiB
106 lines
3.1 KiB
# -*- coding: utf-8 -*- |
|
# |
|
from itertools import chain |
|
from .utils import lazyproperty |
|
|
|
|
|
class Stack(list): |
|
def is_empty(self): |
|
return len(self) == 0 |
|
|
|
@property |
|
def top(self): |
|
if self.is_empty(): |
|
return None |
|
return self[-1] |
|
|
|
@property |
|
def bottom(self): |
|
if self.is_empty(): |
|
return None |
|
return self[0] |
|
|
|
def size(self): |
|
return len(self) |
|
|
|
def push(self, item): |
|
self.append(item) |
|
|
|
|
|
class QuerySetChain: |
|
def __init__(self, querysets): |
|
self.querysets = querysets |
|
|
|
@lazyproperty |
|
def querysets_counts(self): |
|
counts = [s.count() for s in self.querysets] |
|
return counts |
|
|
|
def count(self): |
|
return self.total_count |
|
|
|
@lazyproperty |
|
def total_count(self): |
|
return sum(self.querysets_counts) |
|
|
|
def __iter__(self): |
|
self._chain = chain(*self.querysets) |
|
return self |
|
|
|
def __next__(self): |
|
return next(self._chain) |
|
|
|
def __getitem__(self, ndx): |
|
querysets_count_zip = zip(self.querysets, self.querysets_counts) |
|
length = 0 # 加上本数组后的大数组长度 |
|
pre_length = 0 # 不包含本数组的大数组长度 |
|
items = [] # 返回的值 |
|
loop = 0 |
|
|
|
if isinstance(ndx, slice): |
|
ndx_start = ndx.start or 0 |
|
ndx_stop = ndx.stop or self.total_count |
|
ndx_step = ndx.step or 1 |
|
else: |
|
ndx_start = ndx |
|
ndx_stop, ndx_step = None, None |
|
|
|
for queryset, count in querysets_count_zip: |
|
length += count |
|
loop += 1 |
|
# 取当前数组的start角标, 存在3中情况 |
|
# 1. start角标在当前数组 |
|
if length > ndx_start >= pre_length: |
|
start = ndx_start - pre_length |
|
# print("[loop {}] Start is: {}".format(loop, start)) |
|
if ndx_step is None: |
|
return queryset[start] |
|
# 2. 不包含当前数组,因为起始已经超过了当前数组的长度 |
|
elif ndx_start >= length: |
|
pre_length += count |
|
continue |
|
# 3. 不在当前数组,但是应该从当前数组0开始计算 |
|
else: |
|
start = 0 |
|
|
|
# 可能取单个值, ndx_stop 为None, 不应该再找 |
|
if ndx_stop is None: |
|
pre_length += count |
|
continue |
|
|
|
# 取当前数组的stop角标, 存在2中情况 |
|
# 不存在第3中情况是因为找到了会提交结束循环 |
|
# 1. 结束角标小于length代表 结束位在当前数组上 |
|
if ndx_stop < length: |
|
stop = ndx_stop - pre_length |
|
# 2. 结束位置包含改数组到了最后 |
|
else: |
|
stop = count |
|
# print("[loop {}] Slice: {} {} {}".format(loop, start, stop, ndx_step)) |
|
items.extend(list(queryset[slice(start, stop, ndx_step)])) |
|
pre_length += count |
|
|
|
# 如果结束再当前数组,则结束循环 |
|
if ndx_stop < length: |
|
break |
|
return items
|
|
|