Skip to content

Commit

Permalink
feat: 订阅接口支持指定用户操作进程 (closed TencentBlueKing#2297)
Browse files Browse the repository at this point in the history
  • Loading branch information
Huayeaaa committed Jul 29, 2024
1 parent 107d77d commit f2bae5a
Show file tree
Hide file tree
Showing 8 changed files with 58 additions and 9 deletions.
3 changes: 2 additions & 1 deletion apps/backend/components/collections/plugin.py
Original file line number Diff line number Diff line change
Expand Up @@ -1163,6 +1163,7 @@ def request_gse_or_finish_schedule(self, proc_operate_req: List, data, common_da
def _execute(self, data, parent_data, common_data: PluginCommonData):
op_type = data.get_one_of_inputs("op_type")
gse_version = data.get_one_of_inputs("meta", {}).get("GSE_VERSION")
user = data.get_one_of_inputs("meta", {}).get("user")
policy_step_adapter = common_data.policy_step_adapter
process_statuses = common_data.process_statuses
plugin = policy_step_adapter.plugin_desc
Expand Down Expand Up @@ -1204,7 +1205,7 @@ def _execute(self, data, parent_data, common_data: PluginCommonData):
"proc_name": package_control.process_name or plugin.name,
"setup_path": process_status.setup_path,
"pid_path": process_status.pid_path,
"user": constants.ACCOUNT_MAP.get(host.os_type, "root"),
"user": user or constants.ACCOUNT_MAP.get(host.os_type, "root"),
},
"control": gse_control,
"resource": host_id__resource_policy_map[bk_host_id]["resource"],
Expand Down
13 changes: 13 additions & 0 deletions apps/backend/subscription/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,17 @@ def validate(self, attrs):
raise ValidationError("目前机器参数必须要有 bk_host_id 或者 (ip/bk_host_innerip + bk_cloud_id)")


class OperateInfoSerializer(serializers.Serializer):
bk_host_id = serializers.IntegerField(required=False, label="主机ID")
operate_user = serializers.CharField(required=False, label="操作用户")
system_account = serializers.DictField(required=False, label="系统对应账户")

def validate(self, attrs):
if attrs.get("bk_host_id") and attrs.get("system_account"):
raise ValidationError(_("仅支持一种方式:实例 or 操作系统 指定操作用户"))
return attrs


class CreateSubscriptionSerializer(GatewaySerializer):
class CreateStepSerializer(serializers.Serializer):
id = serializers.CharField(label="步骤标识符", validators=[])
Expand All @@ -83,6 +94,7 @@ class CreateStepSerializer(serializers.Serializer):
target_hosts = TargetHostSerializer(many=True, label="下发的目标机器列表", required=False, allow_empty=False)
run_immediately = serializers.BooleanField(required=False, default=False, label="是否立即执行")
is_main = serializers.BooleanField(required=False, default=False, label="是否为主配置")
operate_info = serializers.ListField(required=False, child=OperateInfoSerializer(), label="操作信息")

# 策略新参数
plugin_name = serializers.CharField(required=False, label="插件名")
Expand Down Expand Up @@ -147,6 +159,7 @@ class UpdateStepSerializer(serializers.Serializer):
scope = UpdateScopeSerializer()
steps = serializers.ListField(child=UpdateStepSerializer())
run_immediately = serializers.BooleanField(required=False, default=False)
operate_info = serializers.ListField(required=False, child=OperateInfoSerializer(), label="操作信息")

# 策略新参数
plugin_name = serializers.CharField(required=False)
Expand Down
15 changes: 12 additions & 3 deletions apps/backend/subscription/tasks.py
Original file line number Diff line number Diff line change
Expand Up @@ -322,10 +322,12 @@ def create_task(
instances: Dict[str, Dict[str, Union[Dict, Any]]],
instance_actions: Dict[str, Dict[str, str]],
preview_only: bool = False,
operate_info: List[Dict[str, Union[Dict, Any]]] = None,
):
"""
创建执行任务
:param preview_only: 是否仅预览
:param operate_info: 操作信息
:param subscription: Subscription
:param subscription_task: SubscriptionTask
:param instances: dict
Expand All @@ -338,7 +340,7 @@ def create_task(
:return: SubscriptionTask
"""
# 兜底注入 Meta,此处注入是覆盖面最全的(包含历史被移除实例)
GrayTools().inject_meta_to_instances(instances)
GrayTools().inject_meta_to_instances(instances, operate_info=operate_info)
logger.info(
"[sub_lifecycle<sub(%s), task(%s)>][create_task] inject meta to instances[num=%s] successfully",
subscription.id,
Expand Down Expand Up @@ -571,10 +573,12 @@ def run_subscription_task_and_create_instance(
scope: Optional[Dict] = None,
actions: Optional[Dict] = None,
preview_only: bool = False,
operate_info: List[Dict[str, Union[Dict, Any]]] = None,
):
"""
自动检查实例及配置的变更,执行相应动作
:param preview_only: 是否仅预览,若为true则不做任何保存或执行动作
:param operate_info:操作信息
:param subscription: Subscription
:param subscription_task: SubscriptionTask
:param scope
Expand Down Expand Up @@ -629,7 +633,7 @@ def run_subscription_task_and_create_instance(
return

# 预注入 Meta,用于变更计算(仅覆盖当前订阅范围,移除场景通过 create_task 兜底注入)
GrayTools().inject_meta_to_instances(instances)
GrayTools().inject_meta_to_instances(instances, operate_info=operate_info)
logger.info(
"[sub_lifecycle<sub(%s), task(%s)>][run_subscription_task_and_create_instance] "
"pre-inject meta to instances[num=%s] successfully",
Expand Down Expand Up @@ -768,7 +772,12 @@ def run_subscription_task_and_create_instance(
instances.update(deleted_instance_info)

create_task_result = create_task(
subscription, subscription_task, instances, instance_actions, preview_only=preview_only
subscription,
subscription_task,
instances,
instance_actions,
preview_only=preview_only,
operate_info=operate_info,
)

return {
Expand Down
6 changes: 4 additions & 2 deletions apps/backend/subscription/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ def create_subscription(self, request):
params = self.validated_data
scope = params["scope"]
run_immediately = params["run_immediately"]
operate_info = params.get("operate_info")

category = params.get("category")
enable = params.get("enable") or False
Expand Down Expand Up @@ -121,7 +122,7 @@ def create_subscription(self, request):
subscription_id=subscription.id, scope=subscription.scope, actions={}
)
tasks.run_subscription_task_and_create_instance.delay(
subscription, subscription_task, language=get_language()
subscription, subscription_task, operate_info=operate_info, language=get_language()
)
result["task_id"] = subscription_task.id

Expand Down Expand Up @@ -189,6 +190,7 @@ def update_subscription(self, request):
params = self.validated_data
scope = params["scope"]
run_immediately = params["run_immediately"]
operate_info = params.get("operate_info")
with transaction.atomic():
try:
subscription = models.Subscription.objects.get(id=params["subscription_id"], is_deleted=False)
Expand Down Expand Up @@ -278,7 +280,7 @@ def update_subscription(self, request):
subscription_id=subscription.id, scope=subscription.scope, actions={}
)
tasks.run_subscription_task_and_create_instance.delay(
subscription, subscription_task, language=get_language()
subscription, subscription_task, operate_info=operate_info, language=get_language()
)
result["task_id"] = subscription_task.id

Expand Down
19 changes: 17 additions & 2 deletions apps/core/gray/tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,11 +67,14 @@ def get_host_ap_gse_version(self, bk_biz_id: typing.Any, ap_id: int, is_install_
return gse_version

def inject_meta_to_instances(
self, instances: typing.Dict[str, typing.Dict[str, typing.Union[typing.Dict, typing.Any]]]
self,
instances: typing.Dict[str, typing.Dict[str, typing.Union[typing.Dict, typing.Any]]],
operate_info: typing.List[typing.Dict[str, typing.Union[typing.Dict, typing.Any]]] = None,
):
"""
在 instances 中注入 Meta 信息
:param instances:
:param instances:实例信息
:param operate_info:操作信息
:return:
"""
bk_host_ids: typing.Set[int] = {
Expand All @@ -87,6 +90,11 @@ def inject_meta_to_instances(
key=node_man_models.GlobalSettings.KeyEnum.JOB_TASK_POLICY.value, default={}
)
biz_ids_list: typing.List[int] = job_task_policy.get(node_man_constants.BkJobScopeType.BIZ.value, [])
host_id__user = {}
system__account = {}
if operate_info:
host_id__user: typing.Dict[int, str] = {op.get("bk_host_id"): op.get("operate_user") for op in operate_info}
system__account: typing.Dict[str, str] = operate_info[0].get("system_account", {})

for host_info in host_infos:
host_id__ap_id_map[host_info["bk_host_id"]] = host_info["ap_id"]
Expand All @@ -103,6 +111,13 @@ def inject_meta_to_instances(
if bk_biz_id in biz_ids_list:
meta["SCOPE_ID"] = bk_biz_id
meta["SCOPE_TYPE"] = node_man_constants.BkJobScopeType.BIZ.value
bk_host_id = host_info.get("bk_host_id")
bk_os_type = host_info.get("bk_os_type")
os_type = node_man_constants.OS_TYPE.get(bk_os_type)
if bk_host_id and host_id__user.get(bk_host_id):
meta["user"] = host_id__user[bk_host_id]
if os_type and system__account.get(os_type):
meta["user"] = system__account[os_type]
gse_version: str = self.get_host_ap_gse_version(
bk_biz_id=bk_biz_id,
ap_id=ap_id,
Expand Down
4 changes: 3 additions & 1 deletion apps/node_man/handlers/plugin_v2.py
Original file line number Diff line number Diff line change
Expand Up @@ -152,7 +152,7 @@ def history(query_params: Dict):
return packages

@staticmethod
def operate(job_type: str, plugin_name: str, scope: Dict, steps: List[Dict]):
def operate(job_type: str, plugin_name: str, scope: Dict, steps: List[Dict], operate_info: List[Dict] = None):
bk_biz_scope = list(set([node["bk_biz_id"] for node in scope["nodes"]]))

CmdbHandler().check_biz_permission(bk_biz_scope, IamActionType.plugin_operate)
Expand All @@ -166,6 +166,8 @@ def operate(job_type: str, plugin_name: str, scope: Dict, steps: List[Dict]):
"scope": scope,
"bk_biz_scope": bk_biz_scope,
}
if operate_info:
base_create_kwargs["operate_info"] = operate_info

if job_type == constants.JobType.MAIN_INSTALL_PLUGIN:
create_data = {**base_create_kwargs, "steps": steps}
Expand Down
6 changes: 6 additions & 0 deletions apps/node_man/serializers/plugin_v2.py
Original file line number Diff line number Diff line change
Expand Up @@ -189,13 +189,19 @@ def validate(self, data):
return data


class OperateInfoSerializer(serializers.Serializer):
bk_host_id = serializers.IntegerField(label=_("主机ID"), required=False)
operate_user = serializers.CharField(label=_("操作用户"), required=False)


class PluginOperateSerializer(serializers.Serializer):
job_type = serializers.ChoiceField(label=_("任务类型"), choices=list(constants.PLUGIN_JOB_TUPLE))
plugin_name = serializers.CharField(label=_("插件名称"))
# 一次性任务范围
scope = base.ScopeSerializer()
# 参数配置
steps = serializers.ListField(child=base.StepSerializer(), required=False, default=[])
operate_info = serializers.ListField(label=_("操作信息"), child=OperateInfoSerializer(), required=False)

def validate(self, data):
if models.GsePluginDesc.objects.filter(name=data["plugin_name"]).first() is None:
Expand Down
1 change: 1 addition & 0 deletions apps/node_man/views/plugin_v2.py
Original file line number Diff line number Diff line change
Expand Up @@ -749,6 +749,7 @@ def operate(self, request):
plugin_name=params["plugin_name"],
scope=params["scope"],
steps=params.get("steps"),
operate_info=params.get("operate_info"),
)
)

Expand Down

0 comments on commit f2bae5a

Please sign in to comment.