Skip to content

Commit

Permalink
feature: 订阅支持业务集 (closed #1724)
Browse files Browse the repository at this point in the history
  • Loading branch information
CohleRustW committed Oct 11, 2023
1 parent dbe5cff commit 42953a6
Show file tree
Hide file tree
Showing 12 changed files with 232 additions and 16 deletions.
16 changes: 16 additions & 0 deletions apps/backend/subscription/errors.py
Original file line number Diff line number Diff line change
Expand Up @@ -169,3 +169,19 @@ class SubscriptionIncludeGrayBizError(AppBaseException):
ERROR_CODE = 19
MESSAGE = _("订阅任务包含Gse2.0灰度业务,任务将暂缓执行无需重复点击")
MESSAGE_TPL = _("订阅任务包含Gse2.0灰度业务,任务将暂缓执行无需重复点击")


class BizSetNullError(AppBaseException):
"""订阅范围中的业务集包含的业务为空"""

ERROR_CODE = 20
MESSAGE = _("订阅范围中的业务集包含的业务为空")
MESSAGE_TPL = _("{订阅范围中的业务集包含的业务为空}")


class BizSetNotExsitError(AppBaseException):
"""业务集不存在"""

ERROR_CODE = 21
MESSAGE = _("业务集不存在")
MESSAGE_TPL = _("{业务集[id: {biz_set_id}] 不存在}")
11 changes: 8 additions & 3 deletions apps/backend/subscription/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,10 @@
from apps.node_man import constants, models, tools
from apps.node_man.models import ProcessStatus
from apps.node_man.serializers import policy
from apps.node_man.serializers.base import SubScopeInstSelectorSerializer
from apps.node_man.serializers.base import (
ScopeTypeSerializer,
SubScopeInstSelectorSerializer,
)
from apps.utils import basic


Expand All @@ -30,6 +33,8 @@ class GatewaySerializer(serializers.Serializer):

class ScopeSerializer(SubScopeInstSelectorSerializer):
bk_biz_id = serializers.IntegerField(required=False, default=None)
scope_type = serializers.CharField(required=False, default=None)
scope_id = serializers.IntegerField(required=False, default=None)
# TODO: 是否取消掉这个范围内的scope
bk_biz_scope = serializers.ListField(required=False)
object_type = serializers.ChoiceField(choices=models.Subscription.OBJECT_TYPE_CHOICES, label="对象类型")
Expand Down Expand Up @@ -125,7 +130,7 @@ class GetSubscriptionSerializer(GatewaySerializer):


class UpdateSubscriptionSerializer(GatewaySerializer):
class UpdateScopeSerializer(SubScopeInstSelectorSerializer):
class UpdateScopeSerializer(SubScopeInstSelectorSerializer, ScopeTypeSerializer):
node_type = serializers.ChoiceField(choices=models.Subscription.NODE_TYPE_CHOICES)
nodes = serializers.ListField()
bk_biz_id = serializers.IntegerField(required=False, default=None)
Expand Down Expand Up @@ -158,7 +163,7 @@ class SwitchSubscriptionSerializer(GatewaySerializer):


class RunSubscriptionSerializer(GatewaySerializer):
class RunScopeSerializer(SubScopeInstSelectorSerializer):
class RunScopeSerializer(SubScopeInstSelectorSerializer, ScopeTypeSerializer):
node_type = serializers.ChoiceField(choices=models.Subscription.NODE_TYPE_CHOICES, label="节点类型")
nodes = serializers.ListField(child=serializers.DictField(), label="拓扑节点列表")

Expand Down
2 changes: 2 additions & 0 deletions apps/backend/subscription/tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -502,6 +502,8 @@ def run_subscription_task_and_create_instance(
else:
scope["object_type"] = subscription.object_type
scope["bk_biz_id"] = subscription.bk_biz_id
scope["scope_type"] = subscription.scope_type
scope["scope_id"] = subscription.scope_id

# 获取订阅范围内全部实例
instances = tools.get_instances_by_scope(scope)
Expand Down
65 changes: 60 additions & 5 deletions apps/backend/subscription/tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
from concurrent.futures import ThreadPoolExecutor, as_completed
from functools import wraps
from itertools import groupby
from typing import Any, Dict, List, Union
from typing import Any, Dict, List, Set, Union

from django.conf import settings
from django.db.models import Q
Expand All @@ -28,6 +28,7 @@
from apps.backend.subscription.commons import get_host_by_inst, list_biz_hosts
from apps.backend.subscription.constants import SUBSCRIPTION_SCOPE_CACHE_TIME
from apps.backend.subscription.errors import (
BizSetNullError,
ConfigRenderFailed,
MultipleObjectError,
PipelineTreeParseError,
Expand Down Expand Up @@ -317,7 +318,7 @@ def get_modules_by_inst_list(inst_list, module_to_topo):


def get_service_instance_by_inst(bk_biz_id, inst_list, module_to_topo):
module_ids, no_module_inst_list = get_modules_by_inst_list(inst_list, module_to_topo)
module_ids, __ = get_modules_by_inst_list(inst_list, module_to_topo)
params = {"bk_biz_id": int(bk_biz_id), "with_name": True}

service_instances = batch_request(client_v2.cc.list_service_instance_detail, params, sort="id")
Expand Down Expand Up @@ -602,7 +603,7 @@ def check_instances_object_type(nodes):
:param nodes: 实例
:return: object_type 集合
"""
bk_obj_id_set = {node["bk_obj_id"] for node in nodes}
bk_obj_id_set = {node.get("bk_obj_id", "") for node in nodes}

if len(bk_obj_id_set) > 1:
raise MultipleObjectError
Expand Down Expand Up @@ -664,10 +665,12 @@ def support_multi_biz(get_instances_by_scope_func):

@wraps(get_instances_by_scope_func)
def wrapper(scope: Dict[str, Union[Dict, Any]], *args, **kwargs) -> Dict[str, Dict[str, Union[Dict, Any]]]:
if scope.get("bk_biz_id") is not None:
if scope.get("scope_type") == models.Subscription.ScopeType.BIZ_SET:
covert_biz_set_scope_to_scope(biz_set_scope=scope, biz_set_id=scope.get("scope_id"))
elif scope.get("bk_biz_id") is not None:
return get_instances_by_scope_func(scope, **kwargs)
# 兼容只传bk_host_id的情况
if (
elif (
scope["object_type"] == models.Subscription.ObjectType.HOST
and scope["node_type"] == models.Subscription.NodeType.INSTANCE
):
Expand Down Expand Up @@ -697,6 +700,58 @@ def wrapper(scope: Dict[str, Union[Dict, Any]], *args, **kwargs) -> Dict[str, Di
return wrapper


def covert_biz_set_scope_to_scope(biz_set_scope: Dict[str, Union[Dict, int, str]], biz_set_id: int = None):
"""
对于整个业务集范围的类型转换为多业务范围
对于其他类型只过滤掉所有不在当前业务集中的 node
{
"nodes": [
{
"bk_obj_id": "biz_set",
"bk_inst_id": 10
}
}
转换为
{
"nodes": [
{
"bk_obj_id": "biz",
"bk_inst_id": 2,
"bk_biz_id": 2
},
{
"bk_obj_id": "biz",
"bk_inst_id": 3,
"bk_biz_id": 3
}
]
}
"""
from apps.node_man.handlers.cmdb import CmdbHandler

bk_obj_id_set: Set[str] = check_instances_object_type(biz_set_scope["nodes"])
# 如果 node 层级上面指定了业务集, 则只对改业务集进行下发, 否则多业务集都转换为业务 ID
if biz_set_id is not None:
biz_set_ids: List[int] = [biz_set_id]
else:
biz_set_ids: List[int] = list(set([node["bk_inst_id"] for node in biz_set_scope["nodes"]]))

bk_biz_ids: List[int] = CmdbHandler.list_biz_ids_in_biz_set(biz_set_ids=biz_set_ids)
if not bk_biz_ids:
raise BizSetNullError

if list(bk_obj_id_set)[0] == "biz_set":
# 业务集转换为多业务
biz_set_scope["nodes"] = [
{"bk_obj_id": "biz", "bk_inst_id": bk_biz_id, "bk_biz_id": bk_biz_id} for bk_biz_id in bk_biz_ids
]
else:
# 不包括在业务集中的业务和未指定具体业务的 node 将被剔除
biz_set_scope["nodes"] = [node for node in biz_set_scope["nodes"] if node.get("bk_biz_id") in bk_biz_ids]


@support_multi_biz
@func_cache_decorator(cache_time=SUBSCRIPTION_SCOPE_CACHE_TIME)
def get_instances_by_scope(scope: Dict[str, Union[Dict, int, Any]]) -> Dict[str, Dict[str, Union[Dict, Any]]]:
Expand Down
4 changes: 4 additions & 0 deletions apps/backend/subscription/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,8 @@ def create_subscription(self, request):
bk_biz_id=scope["bk_biz_id"],
object_type=scope["object_type"],
node_type=scope["node_type"],
scope_type=scope["scope_type"],
scope_id=scope["scope_id"],
nodes=scope["nodes"],
instance_selector=scope.get("instance_selector"),
target_hosts=params.get("target_hosts"),
Expand Down Expand Up @@ -160,6 +162,8 @@ def info(self, request):
"bk_biz_id": subscription.bk_biz_id,
"object_type": subscription.object_type,
"node_type": subscription.node_type,
"scope_type": subscription.scope_type,
"scope_id": subscription.scope_id,
"nodes": subscription.nodes,
},
"pid": subscription.pid,
Expand Down
44 changes: 43 additions & 1 deletion apps/node_man/handlers/cmdb.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
from django.core.cache import cache
from django.utils.translation import ugettext as _

from apps.backend.subscription.errors import BizSetNotExsitError
from apps.component.esbclient import client_v2
from apps.exceptions import ComponentCallError
from apps.iam import Permission
Expand All @@ -35,7 +36,7 @@
cache_all_biz_topo_delay_task,
get_and_cache_format_biz_topo,
)
from apps.utils import APIModel
from apps.utils import APIModel, basic
from apps.utils.batch_request import batch_request, request_multi_thread
from apps.utils.local import get_request_username
from common.log import logger
Expand Down Expand Up @@ -582,3 +583,44 @@ def find_host_service_template(bk_host_ids: List[int]) -> List[Dict]:
@staticmethod
def get_biz_service_template(bk_biz_id: int) -> List[Dict]:
return batch_request(client_v2.cc.list_service_template, {"bk_biz_id": bk_biz_id})

@staticmethod
def list_biz_ids_in_biz_set(biz_set_ids: List[int]):
group_bk_biz_ids: List[int] = []
for biz_set_id in biz_set_ids:
# 先查询业务集是否存在
set_count: int = client_v2.cc.list_business_set(
{
"bk_biz_set_filter": {
"condition": "AND",
"rules": [{"field": "bk_biz_set_id", "operator": "equal", "value": biz_set_id}],
},
"page": {"enable_count": True},
}
).get("count", 0)
if set_count <= 0:
raise BizSetNotExsitError(biz_set_id=biz_set_id)
# 将所有的业务集包括的业务 ID 查询出来后聚合去重复
biz_count: int = int(
client_v2.cc.list_business_in_business_set(
{"bk_biz_set_id": biz_set_id, "page": {"enable_count": True}}
)["count"]
)
prams_list = [
{
"fields": ["bk_biz_id"],
"bk_biz_set_id": biz_set_id,
"page": {
"start": index * constants.QUERY_CMDB_LIMIT,
"limit": count,
"enable_count": False,
"sort": "bk_biz_id",
},
}
for index, count in enumerate(basic.number_slice(biz_count, constants.QUERY_CMDB_LIMIT))
]
set__biz_info = request_multi_thread(
client_v2.cc.list_business_in_business_set, prams_list, get_data=lambda x: x["info"]
)
group_bk_biz_ids.extend([x["bk_biz_id"] for x in set__biz_info])
return list(set(group_bk_biz_ids))
29 changes: 29 additions & 0 deletions apps/node_man/migrations/0076_auto_20230926_1745.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
# Generated by Django 3.2.4 on 2023-09-26 09:45

from django.db import migrations, models


class Migration(migrations.Migration):

dependencies = [
("node_man", "0075_pluginconfigtemplate_variables"),
]

operations = [
migrations.AddField(
model_name="subscription",
name="scope_id",
field=models.IntegerField(db_index=True, null=True, verbose_name="订阅集合类型ID"),
),
migrations.AddField(
model_name="subscription",
name="scope_type",
field=models.CharField(
choices=[("HOST", "主机"), ("SERVICE", "服务")],
db_index=True,
max_length=20,
null=True,
verbose_name="订阅集合类型",
),
),
]
9 changes: 9 additions & 0 deletions apps/node_man/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -1750,6 +1750,11 @@ def __str__(self):
class Subscription(export_subscription_prometheus_mixin(), orm.SoftDeleteModel):
"""订阅"""

class ScopeType(object):
BIZ_SET = "BIZ_SET"

SCOPE_TYPE_CHOICES = (ScopeType.BIZ_SET, _("业务集"))

class ObjectType(object):
HOST = "HOST"
SERVICE = "SERVICE"
Expand Down Expand Up @@ -1787,6 +1792,8 @@ class CategoryType(object):

name = models.CharField(_("任务名称"), max_length=64, null=True, blank=True)
bk_biz_id = models.IntegerField(_("业务ID"), db_index=True, null=True)
scope_type = models.CharField(_("订阅集合类型"), db_index=True, null=True, choices=OBJECT_TYPE_CHOICES, max_length=20)
scope_id = models.IntegerField(_("订阅集合类型ID"), db_index=True, null=True)
object_type = models.CharField(_("对象类型"), max_length=20, choices=OBJECT_TYPE_CHOICES, db_index=True)
node_type = models.CharField(_("节点类型"), max_length=20, choices=NODE_TYPE_CHOICES, db_index=True)
nodes = JSONField(_("节点"), default=list)
Expand Down Expand Up @@ -1830,6 +1837,8 @@ def scope(self):
need_register = True
return {
"bk_biz_id": self.bk_biz_id,
"scope_type": self.scope_type,
"scope_id": self.scope_id,
"object_type": self.object_type,
"node_type": self.node_type,
"nodes": self.nodes,
Expand Down
11 changes: 6 additions & 5 deletions apps/node_man/serializers/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,11 +17,12 @@

# 放在后台会导致循坏导入
class SubScopeInstSelectorSerializer(serializers.Serializer):
instance_selector = serializers.ListField(
child=serializers.DictField(),
required=False,
label="实例筛选器"
)
instance_selector = serializers.ListField(child=serializers.DictField(), required=False, label="实例筛选器")


class ScopeTypeSerializer(serializers.Serializer):
scope_type = serializers.CharField(required=False, default=None)
scope_id = serializers.IntegerField(required=False, default=None)


# 安装插件配置
Expand Down
13 changes: 13 additions & 0 deletions apps/utils/basic.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
"""
import ipaddress
import json
import math
from collections import Counter, namedtuple
from copy import deepcopy
from functools import partial
Expand Down Expand Up @@ -140,6 +141,18 @@ def list_slice(lst: List[Any], limit: int) -> List[List[Any]]:
return slice_list


def number_slice(number: int, limit: int) -> List[int]:
num_chunks = math.ceil(number / limit)
chunks = []
for i in range(num_chunks):
if i == num_chunks - 1:
# 如果是最后一个分片,加入余数
chunks.append(number % limit)
else:
chunks.append(limit)
return chunks


def to_int_or_default(val: Any, default: Any = None) -> Union[int, Any, None]:
try:
return int(val)
Expand Down
12 changes: 12 additions & 0 deletions blueking/component/apis/cc.py
Original file line number Diff line number Diff line change
Expand Up @@ -779,3 +779,15 @@ def __init__(self, client):
path="/api/c/compapi{bk_api_ver}/cc/list_service_template/",
description="查询业务服务模板列表",
)
self.list_business_in_business_set = ComponentAPI(
client=self.client,
method="POST",
path="/api/c/compapi{bk_api_ver}/cc/list_business_in_business_set/",
description="查询业务集中的业务列表",
)
self.list_business_set = ComponentAPI(
client=self.client,
method="POST",
path="/api/c/compapi{bk_api_ver}/cc/list_business_set/",
description="查询业务集",
)
Loading

0 comments on commit 42953a6

Please sign in to comment.