From b8ca2f98decc08d0905904308de812fd9847d0a5 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Mon, 9 Dec 2024 16:07:26 +0800 Subject: [PATCH 01/11] flow: fix bug (instance operate auth error) --- .../middlewares/flow/src/dto/flow_inst_dto.rs | 3 ++ .../flow/src/serv/flow_inst_serv.rs | 32 ++++++++++++------- 2 files changed, 23 insertions(+), 12 deletions(-) diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index f66037e3..23df27a1 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -213,6 +213,7 @@ pub struct FLowInstStateApprovalConf { #[derive(Serialize, Deserialize, PartialEq, Clone, Debug, Default, poem_openapi::Object, sea_orm::FromJsonQueryResult)] pub struct FlowInstArtifacts { pub guard_conf: FlowGuardConf, // 当前操作人权限 + pub prohibit_guard_by_spec_account_ids: Option>, // 禁止操作的指定用户ID pub approval_result: HashMap>>, // 当前审批结果 pub form_state_map: HashMap>, // 录入节点映射 key为节点ID,对应的value为节点中的录入的参数 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID @@ -223,6 +224,8 @@ pub struct FlowInstArtifacts { #[derive(Serialize, Deserialize, PartialEq, Clone, Debug, Default, sea_orm::FromJsonQueryResult)] pub struct FlowInstArtifactsModifyReq { pub guard_conf: Option, // 当前操作人权限 + pub add_prohibit_guard_conf_account_id: Option, // 增加禁止操作人ID + pub delete_prohibit_guard_conf_account_id: Option, // 删除禁止操作人ID pub add_guard_conf_account_id: Option, // 增加操作人ID pub delete_guard_conf_account_id: Option, // 删除操作人ID pub add_approval_result: Option<(String, FlowApprovalResultKind)>, // 增加审批结果 diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 785d372d..3539e86b 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -381,6 +381,7 @@ impl FlowInstServ { (flow_inst::Entity, flow_inst::Column::Transitions), (flow_inst::Entity, flow_inst::Column::OwnPaths), (flow_inst::Entity, flow_inst::Column::Artifacts), + (flow_inst::Entity, flow_inst::Column::Comments), ]) .expr_as(Expr::col((rel_state_table.clone(), NAME_FIELD.clone())).if_null(""), Alias::new("current_state_name")) .expr_as(Expr::col((flow_state_table.clone(), Alias::new("color"))).if_null(""), Alias::new("current_state_color")) @@ -1601,6 +1602,18 @@ impl FlowInstServ { if let Some(guard_conf) = &modify_artifacts.guard_conf { inst_artifacts.guard_conf = guard_conf.clone(); } + if let Some(add_prohibit_guard_conf_account_id) = &modify_artifacts.add_prohibit_guard_conf_account_id { + let mut prohibit_guard_by_spec_account_ids = inst_artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default(); + if !prohibit_guard_by_spec_account_ids.contains(add_prohibit_guard_conf_account_id) { + prohibit_guard_by_spec_account_ids.push(add_prohibit_guard_conf_account_id.clone()); + } + inst_artifacts.prohibit_guard_by_spec_account_ids = Some(prohibit_guard_by_spec_account_ids); + } + if let Some(delete_prohibit_guard_conf_account_id) = &modify_artifacts.delete_prohibit_guard_conf_account_id { + let mut prohibit_guard_by_spec_account_ids = inst_artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default(); + prohibit_guard_by_spec_account_ids = prohibit_guard_by_spec_account_ids.into_iter().filter(|account_id| account_id != delete_prohibit_guard_conf_account_id).collect_vec(); + inst_artifacts.prohibit_guard_by_spec_account_ids = Some(prohibit_guard_by_spec_account_ids); + } if let Some(add_guard_conf_account_id) = &modify_artifacts.add_guard_conf_account_id { if !inst_artifacts.guard_conf.guard_by_spec_account_ids.contains(add_guard_conf_account_id) { inst_artifacts.guard_conf.guard_by_spec_account_ids.push(add_guard_conf_account_id.clone()); @@ -1651,7 +1664,8 @@ impl FlowInstServ { match state_kind { FlowStateKind::Form => kind_conf.form.as_ref().map(|form| { let mut operators = HashMap::new(); - if artifacts.clone().unwrap_or_default().guard_conf.check(ctx) { + let artifacts = artifacts.clone().unwrap_or_default(); + if artifacts.guard_conf.check(ctx) && artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { operators.insert(FlowStateOperatorKind::Submit, form.submit_btn_name.clone()); if form.referral { if let Some(referral_guard_custom_conf) = &form.referral_guard_custom_conf { @@ -1673,8 +1687,8 @@ impl FlowInstServ { }), FlowStateKind::Approval => kind_conf.approval.as_ref().map(|approval| { let mut operators = HashMap::new(); - let approval_conf = artifacts.clone().unwrap_or_default(); - if approval_conf.guard_conf.check(ctx) { + let artifacts = artifacts.clone().unwrap_or_default(); + if artifacts.guard_conf.check(ctx) && artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { operators.insert(FlowStateOperatorKind::Pass, approval.pass_btn_name.clone()); operators.insert(FlowStateOperatorKind::Overrule, approval.overrule_btn_name.clone()); operators.insert(FlowStateOperatorKind::Back, approval.back_btn_name.clone()); @@ -1686,22 +1700,15 @@ impl FlowInstServ { operators.insert(FlowStateOperatorKind::Referral, "".to_string()); } } - if ctx.own_paths == approval_conf.prev_non_auto_account_id.unwrap_or_default() { + if ctx.own_paths == artifacts.prev_non_auto_account_id.unwrap_or_default() { operators.insert(FlowStateOperatorKind::Revoke, "".to_string()); } - let operators = HashMap::from([ - (FlowStateOperatorKind::Referral, "".to_string()), - (FlowStateOperatorKind::Revoke, "".to_string()), - (FlowStateOperatorKind::Pass, approval.pass_btn_name.clone()), - (FlowStateOperatorKind::Overrule, approval.overrule_btn_name.clone()), - (FlowStateOperatorKind::Back, approval.back_btn_name.clone()), - ]); FLowInstStateConf { operators, form_conf: None, approval_conf: Some(FLowInstStateApprovalConf { approval_vars_collect_conf: Some(approval.vars_collect.clone()), - form_vars_collect: artifacts.unwrap_or_default().form_state_map.get(state_id).cloned().unwrap_or_default(), + form_vars_collect: artifacts.form_state_map.get(state_id).cloned().unwrap_or_default(), }), } }), @@ -1721,6 +1728,7 @@ impl FlowInstServ { &FlowInstArtifactsModifyReq { add_guard_conf_account_id: operate_req.operator.clone(), delete_guard_conf_account_id: Some(ctx.owner.clone()), + add_prohibit_guard_conf_account_id: Some(ctx.owner.clone()), ..Default::default() }, funs, From 122d6e44f0036feb771390174a61955f3af68368 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Mon, 9 Dec 2024 18:42:45 +0800 Subject: [PATCH 02/11] flow: fix bug (instance add comment error) --- .../flow/src/api/ci/flow_ci_model_api.rs | 3 +- .../middlewares/flow/src/dto/flow_inst_dto.rs | 8 ++-- .../flow/src/serv/flow_model_serv.rs | 38 ++++++++++--------- 3 files changed, 26 insertions(+), 23 deletions(-) diff --git a/backend/middlewares/flow/src/api/ci/flow_ci_model_api.rs b/backend/middlewares/flow/src/api/ci/flow_ci_model_api.rs index 4d04889e..5c419993 100644 --- a/backend/middlewares/flow/src/api/ci/flow_ci_model_api.rs +++ b/backend/middlewares/flow/src/api/ci/flow_ci_model_api.rs @@ -137,11 +137,12 @@ impl FlowCiModelApi { &FlowModelFilterReq { basic: RbumBasicFilterReq { ids: Some(rel_model_ids), + enabled: Some(true), own_paths: Some("".to_string()), with_sub_own_paths: true, ..Default::default() }, - main: Some(true), + // main: Some(true), ..Default::default() }, None, diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index 23df27a1..5afc3432 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -509,8 +509,8 @@ pub struct FlowInstCommentInfo { pub output_message: String, /// 评价人上下文 pub owner: String, - pub parent_comment_id: String, - pub parent_owner: String, + pub parent_comment_id: Option, + pub parent_owner: Option, /// 评论时间 pub create_time: DateTime, } @@ -520,8 +520,8 @@ pub struct FlowInstCommentInfo { pub struct FlowInstCommentReq { /// 输出信息 pub output_message: String, - pub parent_comment_id: String, - pub parent_owner: String, + pub parent_comment_id: Option, + pub parent_owner: Option, } #[derive(poem_openapi::Object, Serialize, Deserialize, Debug)] diff --git a/backend/middlewares/flow/src/serv/flow_model_serv.rs b/backend/middlewares/flow/src/serv/flow_model_serv.rs index a7ac21d8..aa3c1d7c 100644 --- a/backend/middlewares/flow/src/serv/flow_model_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_model_serv.rs @@ -604,26 +604,23 @@ impl RbumItemCrudOperation TardisResult> { let mut res = Self::do_find_items(filter, desc_sort_by_create, desc_sort_by_update, funs, ctx).await?; for item in res.iter_mut() { - let version = FlowModelVersionServ::get_item( - &item.current_version_id, - &FlowModelVersionFilterReq { - basic: RbumBasicFilterReq { - ids: None, - ..filter.basic.clone() + if !item.current_version_id.is_empty() { + let version = FlowModelVersionServ::get_item( + &item.current_version_id, + &FlowModelVersionFilterReq { + basic: RbumBasicFilterReq { + ids: None, + ..filter.basic.clone() + }, + ..Default::default() }, - ..Default::default() - }, - funs, - ctx, - ) - .await?; - item.init_state_id = version.init_state_id; - - let rel_transition = - FlowRelServ::find_from_simple_rels(&FlowRelKind::FlowModelTransition, &item.id, None, None, funs, ctx).await?.into_iter().map(|rel| rel.ext).collect_vec().pop(); - item.rel_transition = rel_transition.map(|rel_transition| TardisFuns::json.obj_to_json(&rel_transition).unwrap_or_default()); + funs, + ctx, + ) + .await?; + item.init_state_id = version.init_state_id; - let states = FlowRelServ::find_from_simple_rels(&FlowRelKind::FlowModelState, &item.current_version_id, None, None, funs, ctx) + let states = FlowRelServ::find_from_simple_rels(&FlowRelKind::FlowModelState, &item.current_version_id, None, None, funs, ctx) .await? .into_iter() .map(|rel| FLowStateIdAndName { @@ -632,6 +629,11 @@ impl RbumItemCrudOperation Date: Tue, 10 Dec 2024 09:25:14 +0800 Subject: [PATCH 03/11] flow: fix bug (copy model error) --- backend/middlewares/flow/src/dto/flow_model_dto.rs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/backend/middlewares/flow/src/dto/flow_model_dto.rs b/backend/middlewares/flow/src/dto/flow_model_dto.rs index 46af736a..45fefc89 100644 --- a/backend/middlewares/flow/src/dto/flow_model_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_model_dto.rs @@ -75,13 +75,14 @@ impl From for FlowModelAddReq { is_init: value.init_state_id == state.id, }) .collect_vec(); + let rel_transition_ids = value.rel_transition().clone().map(|rel_transition| vec![rel_transition.id]); Self { name: value.name.as_str().into(), icon: Some(value.icon.clone()), info: Some(value.info.clone()), kind: value.kind, status: value.status, - rel_transition_ids: None, + rel_transition_ids, rel_template_ids: Some(value.rel_template_ids.clone()), add_version: Some(FlowModelVersionAddReq { name: value.name.as_str().into(), From fd720c3d805152f5f9aefdc33d807f3686a24b41 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Tue, 10 Dec 2024 10:02:21 +0800 Subject: [PATCH 04/11] flow: fix bug (instance button show failed) --- backend/middlewares/flow/src/serv/flow_inst_serv.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 3539e86b..80bf7273 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -1665,7 +1665,7 @@ impl FlowInstServ { FlowStateKind::Form => kind_conf.form.as_ref().map(|form| { let mut operators = HashMap::new(); let artifacts = artifacts.clone().unwrap_or_default(); - if artifacts.guard_conf.check(ctx) && artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { + if artifacts.guard_conf.check(ctx) && !artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { operators.insert(FlowStateOperatorKind::Submit, form.submit_btn_name.clone()); if form.referral { if let Some(referral_guard_custom_conf) = &form.referral_guard_custom_conf { @@ -1688,7 +1688,7 @@ impl FlowInstServ { FlowStateKind::Approval => kind_conf.approval.as_ref().map(|approval| { let mut operators = HashMap::new(); let artifacts = artifacts.clone().unwrap_or_default(); - if artifacts.guard_conf.check(ctx) && artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { + if artifacts.guard_conf.check(ctx) && !artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { operators.insert(FlowStateOperatorKind::Pass, approval.pass_btn_name.clone()); operators.insert(FlowStateOperatorKind::Overrule, approval.overrule_btn_name.clone()); operators.insert(FlowStateOperatorKind::Back, approval.back_btn_name.clone()); From f7789728083a049520e18eb932da0fda45d53297 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Tue, 10 Dec 2024 17:30:42 +0800 Subject: [PATCH 05/11] flow: fix bug (instance update failed) --- .../flow/src/api/cc/flow_cc_model_api.rs | 4 ++-- .../flow/src/dto/flow_model_dto.rs | 2 +- .../flow/src/dto/flow_model_version_dto.rs | 2 +- .../flow/src/serv/flow_inst_serv.rs | 19 ++++++++++++++----- .../flow/src/serv/flow_model_serv.rs | 18 ++++++++++++++---- .../flow/src/serv/flow_model_version_serv.rs | 13 +++++++------ .../flow/tests/test_flow_scenes_fsm1.rs | 4 ++-- 7 files changed, 41 insertions(+), 21 deletions(-) diff --git a/backend/middlewares/flow/src/api/cc/flow_cc_model_api.rs b/backend/middlewares/flow/src/api/cc/flow_cc_model_api.rs index 39333d42..22ddd4bf 100644 --- a/backend/middlewares/flow/src/api/cc/flow_cc_model_api.rs +++ b/backend/middlewares/flow/src/api/cc/flow_cc_model_api.rs @@ -287,7 +287,7 @@ impl FlowCcModelApi { .sort_states .into_iter() .map(|state| FlowModelVersionModifyState { - id: state.state_id.clone(), + id: Some(state.state_id.clone()), modify_rel: Some(FlowStateRelModelModifyReq { id: state.state_id, sort: Some(state.sort), @@ -362,7 +362,7 @@ impl FlowCcModelApi { &mut FlowModelModifyReq { modify_version: Some(FlowModelVersionModifyReq { modify_states: Some(vec![FlowModelVersionModifyState { - id: req.0.id.clone(), + id: Some(req.0.id.clone()), modify_rel: Some(req.0), modify_state: None, add_transitions: None, diff --git a/backend/middlewares/flow/src/dto/flow_model_dto.rs b/backend/middlewares/flow/src/dto/flow_model_dto.rs index 45fefc89..e15a5a78 100644 --- a/backend/middlewares/flow/src/dto/flow_model_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_model_dto.rs @@ -239,7 +239,7 @@ impl FlowModelDetailResp { } pub fn rel_transition(&self) -> Option { - self.rel_transition.clone().map(|rel_transition| TardisFuns::json.json_to_obj(rel_transition.clone()).unwrap()) + self.rel_transition.clone().map(|rel_transition| TardisFuns::json.json_to_obj(rel_transition).unwrap_or_default()) } } diff --git a/backend/middlewares/flow/src/dto/flow_model_version_dto.rs b/backend/middlewares/flow/src/dto/flow_model_version_dto.rs index fd06484e..ea7f15bd 100644 --- a/backend/middlewares/flow/src/dto/flow_model_version_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_model_version_dto.rs @@ -71,7 +71,7 @@ pub struct FlowModelVersionBindState { #[derive(Clone, Serialize, Deserialize, Debug, Default, poem_openapi::Object)] pub struct FlowModelVersionModifyState { /// 若存在则表示,绑定已有状态节点 - pub id: String, + pub id: Option, pub modify_state: Option, pub modify_rel: Option, /// 添加动作 diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 80bf7273..41cf398c 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -309,6 +309,11 @@ impl FlowInstServ { ..Default::default() }; funs.db().update_one(flow_inst, ctx).await?; + + let flow_inst_detail = Self::get(flow_inst_id, funs, ctx).await?; + if !flow_inst_detail.main { + FlowSearchClient::modify_business_obj_search(&flow_inst_detail.rel_business_obj_id, &flow_inst_detail.tag, funs, ctx).await?; + } Ok(()) } @@ -588,7 +593,7 @@ impl FlowInstServ { ) .await; // 若当前数据项存在未结束的审批流,则清空其中的transitions - let unfinished_approve_flow_insts = Self::find_details( + let unfinished_approve_flow_inst_ids = Self::find_details( &FlowInstFilterReq { rel_business_obj_ids: Some(flow_insts.iter().map(|flow_inst| flow_inst.rel_business_obj_id.clone()).collect_vec()), main: Some(false), @@ -600,13 +605,13 @@ impl FlowInstServ { ) .await? .into_iter() - .map(|inst| inst.id.clone()) + .map(|inst| flow_insts.iter().find(|flow_inst| flow_inst.rel_business_obj_id == inst.rel_business_obj_id).unwrap().id.clone()) .collect_vec(); - let _ = state_and_next_transitions.iter_mut().map(|item| { - if unfinished_approve_flow_insts.contains(&item.flow_inst_id) { + for item in state_and_next_transitions.iter_mut() { + if unfinished_approve_flow_inst_ids.contains(&item.flow_inst_id) { item.next_flow_transitions.clear(); } - }); + } Ok(state_and_next_transitions) } @@ -869,6 +874,10 @@ impl FlowInstServ { funs.db().update_one(flow_inst, ctx).await?; + if next_flow_state.sys_state == FlowSysStateKind::Finish && !flow_inst_detail.main { + FlowSearchClient::modify_business_obj_search(&flow_inst_detail.rel_business_obj_id, &flow_inst_detail.tag, funs, ctx).await?; + } + Self::when_leave_state(flow_inst_detail, &prev_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; Self::when_enter_state(flow_inst_detail, &next_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; diff --git a/backend/middlewares/flow/src/serv/flow_model_serv.rs b/backend/middlewares/flow/src/serv/flow_model_serv.rs index aa3c1d7c..c4a15100 100644 --- a/backend/middlewares/flow/src/serv/flow_model_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_model_serv.rs @@ -590,7 +590,7 @@ impl RbumItemCrudOperation Date: Tue, 10 Dec 2024 20:51:29 +0800 Subject: [PATCH 06/11] flow: fix bug (approve flow instance guard_by_assigned failed) --- .../flow/src/serv/flow_inst_serv.rs | 107 +++++++++++++++++- 1 file changed, 106 insertions(+), 1 deletion(-) diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 41cf398c..8e157014 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -44,7 +44,7 @@ use crate::{ }, flow_model_dto::FlowModelFilterReq, flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, - flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowSysStateKind}, + flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowSysStateKind}, flow_transition_dto::FlowTransitionDetailResp, flow_var_dto::FillType, }, @@ -1472,8 +1472,54 @@ impl FlowInstServ { transitions.iter().map(|transition| guard_custom_conf.guard_by_spec_account_ids.push(transition.op_ctx.owner.clone())).collect::>() }); } + if form_conf.guard_by_assigned { + let _ = Self::get_main_inst_vars(flow_inst_detail, funs, ctx).await? + .get("assigned_to") + .unwrap_or(&json!("")) + .as_str() + .unwrap_or_default() + .split(',') + .collect_vec() + .into_iter() + .map(|str| guard_custom_conf.guard_by_spec_account_ids.push(str.to_string())) + .collect::>(); + } modify_req.guard_conf = Some(guard_custom_conf); Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; + // 当操作人为空时的逻辑 + let curr_guard_conf = Self::get(&flow_inst_detail.id, funs, ctx).await?.artifacts.unwrap_or_default().guard_conf; + if curr_guard_conf.guard_by_spec_account_ids.is_empty() + && curr_guard_conf.guard_by_spec_org_ids.is_empty() + && curr_guard_conf.guard_by_spec_role_ids.is_empty() + && form_conf.auto_transfer_when_empty_kind.is_some() { + match form_conf.auto_transfer_when_empty_kind.unwrap_or_default() { + FlowStatusAutoStrategyKind::Autoskip => { + if let Some(next_transition) = FlowInstServ::find_next_transitions(flow_inst_detail, &FlowInstFindNextTransitionsReq { vars: None }, funs, ctx).await?.pop() { + Self::transfer( + flow_inst_detail, + &FlowInstTransferReq { + flow_transition_id: next_transition.next_flow_transition_id, + message: None, + vars: None, + }, + false, + FlowExternalCallbackOp::Auto, + loop_check_helper::InstancesTransition::default(), + ctx, + funs, + ) + .await?; + } + }, + FlowStatusAutoStrategyKind::SpecifyAgent => { + modify_req.guard_conf = Some(form_conf.auto_transfer_when_empty_guard_custom_conf.clone().unwrap_or_default()); + Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; + }, + FlowStatusAutoStrategyKind::TransferState => { + // @TODO 当前版本不支持 + }, + } + } } FlowStateKind::Approval => { let mut modify_req = FlowInstArtifactsModifyReq { ..Default::default() }; @@ -1487,8 +1533,54 @@ impl FlowInstServ { transitions.iter().map(|transition| guard_custom_conf.guard_by_spec_account_ids.push(transition.op_ctx.owner.clone())).collect::>() }); } + if approval_conf.guard_by_assigned { + let _ = Self::get_main_inst_vars(flow_inst_detail, funs, ctx).await? + .get("assigned_to") + .unwrap_or(&json!("")) + .as_str() + .unwrap_or_default() + .split(',') + .collect_vec() + .into_iter() + .map(|str| guard_custom_conf.guard_by_spec_account_ids.push(str.to_string())) + .collect::>(); + } modify_req.guard_conf = Some(guard_custom_conf); Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; + // 当操作人为空时的逻辑 + let curr_guard_conf = Self::get(&flow_inst_detail.id, funs, ctx).await?.artifacts.unwrap_or_default().guard_conf; + if curr_guard_conf.guard_by_spec_account_ids.is_empty() + && curr_guard_conf.guard_by_spec_org_ids.is_empty() + && curr_guard_conf.guard_by_spec_role_ids.is_empty() + && approval_conf.auto_transfer_when_empty_kind.is_some() { + match approval_conf.auto_transfer_when_empty_kind.unwrap_or_default() { + FlowStatusAutoStrategyKind::Autoskip => { + if let Some(next_transition) = FlowInstServ::find_next_transitions(flow_inst_detail, &FlowInstFindNextTransitionsReq { vars: None }, funs, ctx).await?.pop() { + Self::transfer( + flow_inst_detail, + &FlowInstTransferReq { + flow_transition_id: next_transition.next_flow_transition_id, + message: None, + vars: None, + }, + false, + FlowExternalCallbackOp::Auto, + loop_check_helper::InstancesTransition::default(), + ctx, + funs, + ) + .await?; + } + }, + FlowStatusAutoStrategyKind::SpecifyAgent => { + modify_req.guard_conf = Some(approval_conf.auto_transfer_when_empty_guard_custom_conf.clone().unwrap_or_default()); + Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; + }, + FlowStatusAutoStrategyKind::TransferState => { + // @TODO 当前版本不支持 + }, + } + } } FlowStateKind::Branch => {} FlowStateKind::Finish => match rel_transition.as_str() { @@ -2255,4 +2347,17 @@ impl FlowInstServ { } Ok(order_fragments) } + + // 获取主实例的参数列表 + async fn get_main_inst_vars(flow_inst: &FlowInstDetailResp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { + let main_inst_id = Self::get_inst_ids_by_rel_business_obj_id(vec![flow_inst.rel_business_obj_id.clone()], Some(true), funs, ctx).await? + .pop() + .ok_or_else(|| funs.err().not_found("flow_inst", "get_guard_by_assigned", "illegal response", "404-flow-inst-not-found"))?; + let main_inst = Self::get(&main_inst_id, funs, ctx).await?; + Ok(main_inst + .current_vars + .clone() + .unwrap_or_default() + ) + } } From fc37a3615df5938d86e6b41b6a9c8f4f2840f257 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Wed, 11 Dec 2024 17:02:22 +0800 Subject: [PATCH 07/11] flow: instance add field code --- .../middlewares/flow/src/domain/flow_inst.rs | 4 + .../middlewares/flow/src/dto/flow_inst_dto.rs | 10 +- .../flow/src/serv/flow_inst_serv.rs | 144 +++++++----------- 3 files changed, 67 insertions(+), 91 deletions(-) diff --git a/backend/middlewares/flow/src/domain/flow_inst.rs b/backend/middlewares/flow/src/domain/flow_inst.rs index a88c2a9c..23dafffd 100644 --- a/backend/middlewares/flow/src/domain/flow_inst.rs +++ b/backend/middlewares/flow/src/domain/flow_inst.rs @@ -14,6 +14,10 @@ pub struct Model { #[index] pub rel_flow_version_id: String, + /// Instance code / 实例编码 + #[index(unique)] + pub code: Option, + /// Business object Id / 关联的业务对象Id #[index] pub rel_business_obj_id: String, diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index 5afc3432..eb5dad58 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -74,6 +74,8 @@ pub struct FlowInstAbortReq { #[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] pub struct FlowInstSummaryResp { pub id: String, + + pub code: String, /// Associated [flow_model](super::flow_model_version_dto::FlowModelVersionDetailResp) id /// /// 关联的[工作流模板](super::flow_model_version_dto::FlowModelVersionDetailResp) id @@ -112,6 +114,9 @@ pub struct FlowInstSummaryResp { #[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] pub struct FlowInstDetailResp { pub id: String, + + pub code: String, + /// Associated [flow_model](super::flow_model_dto::FlowModelDetailResp) id /// /// 关联的[工作流模板](super::flow_model_dto::FlowModelDetailResp) id @@ -481,6 +486,7 @@ pub struct FlowInstFilterReq { #[derive(sea_orm::FromQueryResult)] pub struct FlowInstSummaryResult { pub id: String, + pub code: String, pub rel_flow_version_id: String, pub rel_flow_model_id: String, pub rel_flow_model_name: String, @@ -529,7 +535,7 @@ pub struct FlowInstSearchReq { // Search conditions pub query: FlowInstFilterReq, // Advanced search - pub query_kind: Option, + pub query_kind: Option>, // Sort // When the record set is very large, it will seriously affect the performance, it is not recommended to use. pub sort: Option>, @@ -538,8 +544,6 @@ pub struct FlowInstSearchReq { #[derive(Serialize, Deserialize, Debug, poem_openapi::Enum, Eq, Hash, PartialEq, Clone)] pub enum FlowInstQueryKind { - /// 全部 - All, /// 待录入 Form, /// 待审批 diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 8e157014..393d7ef5 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -18,7 +18,7 @@ use itertools::Itertools; use serde_json::json; use tardis::{ basic::{dto::TardisContext, result::TardisResult}, - chrono::{DateTime, Utc}, + chrono::{DateTime, Datelike, Utc}, db::sea_orm::{ self, sea_query::{Alias, Cond, Expr, Query, SelectStatement}, @@ -100,7 +100,7 @@ impl FlowInstServ { return Err(funs.err().internal_error("flow_inst_serv", "start", "The same instance exist", "500-flow-inst-exist")); } let main = start_req.transition_id.is_none(); - let flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { + let mut flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { id: Set(inst_id.clone()), tag: Set(Some(flow_model.tag.clone())), rel_flow_version_id: Set(flow_model.current_version_id.clone()), @@ -115,6 +115,9 @@ impl FlowInstServ { main: Set(main), ..Default::default() }; + if !main { + flow_inst.code = Set(Some(Self::gen_inst_code(&funs).await?)); + } funs.db().insert_one(flow_inst, ctx).await?; let inst = Self::get(&inst_id, &funs, ctx).await?; Self::when_enter_state(&inst, ¤t_state_id, &flow_model.id, &funs, ctx).await?; @@ -330,6 +333,7 @@ impl FlowInstServ { #[derive(sea_orm::FromQueryResult)] pub struct FlowInstDetailResult { pub id: String, + pub code: String, pub tag: String, pub rel_flow_version_id: String, pub rel_flow_model_name: String, @@ -370,6 +374,7 @@ impl FlowInstServ { query .columns([ (flow_inst::Entity, flow_inst::Column::Id), + (flow_inst::Entity, flow_inst::Column::Code), (flow_inst::Entity, flow_inst::Column::Tag), (flow_inst::Entity, flow_inst::Column::RelFlowVersionId), (flow_inst::Entity, flow_inst::Column::RelBusinessObjId), @@ -455,6 +460,7 @@ impl FlowInstServ { let artifacts = inst.artifacts.clone().map(|artifacts| TardisFuns::json.json_to_obj::(artifacts).unwrap_or_default()); FlowInstDetailResp { id: inst.id, + code: inst.code, rel_flow_version_id: inst.rel_flow_version_id, rel_flow_model_name: inst.rel_flow_model_name, tag: inst.tag, @@ -529,6 +535,7 @@ impl FlowInstServ { .into_iter() .map(|inst| FlowInstSummaryResp { id: inst.id, + code: inst.code, rel_flow_version_id: inst.rel_flow_version_id, rel_flow_model_id: inst.rel_flow_model_id, rel_flow_model_name: inst.rel_flow_model_name, @@ -2053,7 +2060,7 @@ impl FlowInstServ { Self::package_query(table_alias_name, search_req.query.clone(), &mut sql_vals, &mut where_fragments, funs, ctx)?; Self::package_query_kind( table_alias_name, - search_req.query_kind.clone().unwrap_or(FlowInstQueryKind::All), + search_req.query_kind.clone().unwrap_or(vec![FlowInstQueryKind::Create, FlowInstQueryKind::Form, FlowInstQueryKind::Approval]), &mut sql_vals, &mut where_fragments, funs, @@ -2067,6 +2074,7 @@ impl FlowInstServ { format!( r#"SELECT flow_inst.id, + flow_inst.code, flow_inst.rel_flow_version_id, flow_inst.rel_business_obj_id, flow_inst.current_state_id, @@ -2113,6 +2121,7 @@ impl FlowInstServ { } Ok(FlowInstSummaryResp { id: item.try_get("", "id")?, + code: item.try_get("", "code")?, rel_flow_version_id: item.try_get("", "rel_flow_version_id")?, rel_flow_model_id: item.try_get("", "rel_flow_model_id")?, rel_flow_model_name: item.try_get("", "rel_flow_model_name")?, @@ -2140,109 +2149,54 @@ impl FlowInstServ { fn package_query_kind( table_alias_name: &str, - query_kind: FlowInstQueryKind, + query_kinds: Vec, sql_vals: &mut Vec, where_fragments: &mut Vec, _funs: &TardisFunsInst, ctx: &TardisContext, ) -> TardisResult<()> { - match query_kind { - FlowInstQueryKind::Create => { - sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); - where_fragments.push(format!("{}.create_ctx ->> 'owner' = ${}", table_alias_name, sql_vals.len())); - } - FlowInstQueryKind::Form => { - let mut child_or_where_fragments = vec![]; - sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); + let mut or_where_fragments = vec![]; + if query_kinds.contains(&FlowInstQueryKind::Create) { + sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); + or_where_fragments.push(format!("({}.create_ctx ->> 'owner' = ${})", table_alias_name, sql_vals.len())); + } + if query_kinds.contains(&FlowInstQueryKind::Form) { + let mut child_or_where_fragments = vec![]; + sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); + child_or_where_fragments.push(format!( + "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_account_ids') AS elem WHERE elem IN (${}))", + table_alias_name, + sql_vals.len() + )); + if !ctx.roles.is_empty() { + sql_vals.push(sea_orm::Value::from(ctx.roles.clone().join(", ").to_string())); child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_account_ids') AS elem WHERE elem IN (${}))", + "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_role_ids') AS elem WHERE elem IN (${}))", table_alias_name, sql_vals.len() )); - if !ctx.roles.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.roles.clone().join(", ").to_string())); - child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_role_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - if !ctx.groups.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.groups.clone().join(", ").to_string())); - child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_org_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - where_fragments.push(format!("current_state.state_kind = 'form' AND ({})", child_or_where_fragments.join(" OR "))); } - FlowInstQueryKind::Approval => { - let mut child_or_where_fragments = vec![]; - sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); + if !ctx.groups.is_empty() { + sql_vals.push(sea_orm::Value::from(ctx.groups.clone().join(", ").to_string())); child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_account_ids') AS elem WHERE elem IN (${}))", + "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_org_ids') AS elem WHERE elem IN (${}))", table_alias_name, sql_vals.len() )); - if !ctx.roles.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.roles.clone().join(", ").to_string())); - child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_role_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - if !ctx.groups.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.groups.clone().join(", ").to_string())); - child_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_org_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - where_fragments.push(format!("current_state.state_kind = 'approval' AND ({})", child_or_where_fragments.join(" OR "))); } - FlowInstQueryKind::All => { - let mut or_where_fragments = vec![]; - sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); - or_where_fragments.push(format!("({}.create_ctx ->> 'owner' = ${})", table_alias_name, sql_vals.len())); - - let mut form_or_where_fragments = vec![]; - sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); - form_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_account_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - if !ctx.roles.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.roles.clone().join(", ").to_string())); - form_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_role_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - if !ctx.groups.is_empty() { - sql_vals.push(sea_orm::Value::from(ctx.groups.clone().join(", ").to_string())); - form_or_where_fragments.push(format!( - "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_org_ids') AS elem WHERE elem IN (${}))", - table_alias_name, - sql_vals.len() - )); - } - or_where_fragments.push(format!("(current_state.state_kind = 'form' AND ({}))", form_or_where_fragments.join(" OR "))); - - let mut approval_or_where_fragments = vec![]; + or_where_fragments.push(format!("(current_state.state_kind = 'form' AND ({}))", child_or_where_fragments.join(" OR "))); + } + if query_kinds.contains(&FlowInstQueryKind::Approval) { + let mut child_or_where_fragments = vec![]; sql_vals.push(sea_orm::Value::from(ctx.owner.clone())); - approval_or_where_fragments.push(format!( + child_or_where_fragments.push(format!( "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_account_ids') AS elem WHERE elem IN (${}))", table_alias_name, sql_vals.len() )); if !ctx.roles.is_empty() { sql_vals.push(sea_orm::Value::from(ctx.roles.clone().join(", ").to_string())); - approval_or_where_fragments.push(format!( + child_or_where_fragments.push(format!( "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_role_ids') AS elem WHERE elem IN (${}))", table_alias_name, sql_vals.len() @@ -2250,16 +2204,15 @@ impl FlowInstServ { } if !ctx.groups.is_empty() { sql_vals.push(sea_orm::Value::from(ctx.groups.clone().join(", ").to_string())); - approval_or_where_fragments.push(format!( + child_or_where_fragments.push(format!( "EXISTS (SELECT 1 FROM jsonb_array_elements_text({}.artifacts->'guard_conf'->'guard_by_spec_org_ids') AS elem WHERE elem IN (${}))", table_alias_name, sql_vals.len() )); } - or_where_fragments.push(format!("(current_state.state_kind = 'approval' AND ({}))", approval_or_where_fragments.join(" OR "))); - where_fragments.push(format!("( {} )", or_where_fragments.join(" OR "))); - } + or_where_fragments.push(format!("(current_state.state_kind = 'approval' AND ({}))", child_or_where_fragments.join(" OR "))); } + where_fragments.push(format!("( {} )", or_where_fragments.join(" OR "))); Ok(()) } @@ -2360,4 +2313,19 @@ impl FlowInstServ { .unwrap_or_default() ) } + + async fn gen_inst_code(funs: &TardisFunsInst) -> TardisResult { + let count = funs + .db() + .count( + Query::select() + .columns([flow_inst::Column::Code]) + .from(flow_inst::Entity) + .and_where(Expr::col(flow_inst::Column::CreateTime).gt(Utc::now().date_naive())) + .and_where(Expr::col(flow_inst::Column::Main).eq(false)), + ) + .await?; + let current_date = Utc::now(); + Ok(format!("SP{}{}{}{:0>5}", current_date.year(), current_date.month(), current_date.day(), count + 1).to_string()) + } } From 61f575f10344b3a651c63c3a5d1402a5152f8c04 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Wed, 11 Dec 2024 21:49:37 +0800 Subject: [PATCH 08/11] flow: complete code (check approval cond) --- .../flow/src/api/cc/flow_cc_inst_api.rs | 4 +- .../middlewares/flow/src/dto/flow_inst_dto.rs | 8 +- .../flow/src/dto/flow_state_dto.rs | 2 +- .../flow/src/serv/clients/search_client.rs | 16 -- .../flow/src/serv/flow_external_serv.rs | 27 ++- .../flow/src/serv/flow_inst_serv.rs | 208 +++++++++++------- 6 files changed, 163 insertions(+), 102 deletions(-) diff --git a/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs b/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs index 9fe910a8..1a253ffa 100644 --- a/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs +++ b/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs @@ -28,8 +28,10 @@ impl FlowCcInstApi { /// 启动实例(返回实例ID) #[oai(path = "/", method = "post")] async fn start(&self, add_req: Json, ctx: TardisContextExtractor, _request: &Request) -> TardisApiResult { - let funs = flow_constants::get_tardis_inst(); + let mut funs = flow_constants::get_tardis_inst(); + funs.begin().await?; let result = FlowInstServ::start(&add_req.0, None, &funs, &ctx.0).await?; + funs.commit().await?; ctx.0.execute_task().await?; TardisResp::ok(result) } diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index eb5dad58..81380d2a 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -74,7 +74,7 @@ pub struct FlowInstAbortReq { #[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] pub struct FlowInstSummaryResp { pub id: String, - + pub code: String, /// Associated [flow_model](super::flow_model_version_dto::FlowModelVersionDetailResp) id /// @@ -218,8 +218,9 @@ pub struct FLowInstStateApprovalConf { #[derive(Serialize, Deserialize, PartialEq, Clone, Debug, Default, poem_openapi::Object, sea_orm::FromJsonQueryResult)] pub struct FlowInstArtifacts { pub guard_conf: FlowGuardConf, // 当前操作人权限 - pub prohibit_guard_by_spec_account_ids: Option>, // 禁止操作的指定用户ID + pub prohibit_guard_by_spec_account_ids: Option>, // 禁止操作的指定用户ID pub approval_result: HashMap>>, // 当前审批结果 + pub approval_total: HashMap, // 审批总数 pub form_state_map: HashMap>, // 录入节点映射 key为节点ID,对应的value为节点中的录入的参数 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID pub prev_non_auto_account_id: Option, // 上一个节点操作人ID @@ -230,7 +231,7 @@ pub struct FlowInstArtifacts { pub struct FlowInstArtifactsModifyReq { pub guard_conf: Option, // 当前操作人权限 pub add_prohibit_guard_conf_account_id: Option, // 增加禁止操作人ID - pub delete_prohibit_guard_conf_account_id: Option, // 删除禁止操作人ID + pub delete_prohibit_guard_conf_account_id: Option, // 删除禁止操作人ID pub add_guard_conf_account_id: Option, // 增加操作人ID pub delete_guard_conf_account_id: Option, // 删除操作人ID pub add_approval_result: Option<(String, FlowApprovalResultKind)>, // 增加审批结果 @@ -239,6 +240,7 @@ pub struct FlowInstArtifactsModifyReq { pub clear_approval_result: Option, // 清除节点审批信息 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID pub prev_non_auto_account_id: Option, // 上一个节点操作人ID + pub curr_approval_total: Option, // 当前审批总数 } /// 审批结果类型 diff --git a/backend/middlewares/flow/src/dto/flow_state_dto.rs b/backend/middlewares/flow/src/dto/flow_state_dto.rs index e83e14e6..c77a4b35 100644 --- a/backend/middlewares/flow/src/dto/flow_state_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_state_dto.rs @@ -171,7 +171,7 @@ pub struct FlowStateCountersignConf { /// 类型 pub kind: FlowStateCountersignKind, /// 多数人通过比例 - pub most_percent: Option, + pub most_percent: Option, /// 审批人权限配置 pub guard_custom_conf: Option, /// 指定人通过即通过 diff --git a/backend/middlewares/flow/src/serv/clients/search_client.rs b/backend/middlewares/flow/src/serv/clients/search_client.rs index c63d428f..8216d16f 100644 --- a/backend/middlewares/flow/src/serv/clients/search_client.rs +++ b/backend/middlewares/flow/src/serv/clients/search_client.rs @@ -35,22 +35,6 @@ const SEARCH_TAG: &str = "flow_model"; pub struct FlowSearchClient; impl FlowSearchClient { - pub async fn async_modify_business_obj_search(inst_id: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { - let ctx_clone = ctx.clone(); - let inst_detail = FlowInstServ::get(inst_id, funs, ctx).await?; - ctx.add_async_task(Box::new(|| { - Box::pin(async move { - let task_handle = tokio::spawn(async move { - let funs = flow_constants::get_tardis_inst(); - let _ = Self::modify_business_obj_search(&inst_detail.rel_business_obj_id, &inst_detail.tag, &funs, &ctx_clone).await; - }); - task_handle.await.unwrap(); - Ok(()) - }) - })) - .await - } - pub async fn modify_business_obj_search(rel_business_obj_id: &str, tag: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { let tag_search_map = HashMap::from([ ("CTS", "idp_test"), diff --git a/backend/middlewares/flow/src/serv/flow_external_serv.rs b/backend/middlewares/flow/src/serv/flow_external_serv.rs index 1ef8711b..ee77fd4a 100644 --- a/backend/middlewares/flow/src/serv/flow_external_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_external_serv.rs @@ -1,9 +1,7 @@ use bios_sdk_invoke::{clients::spi_kv_client::SpiKvClient, invoke_constants::TARDIS_CONTEXT}; use itertools::Itertools; use tardis::{ - basic::{dto::TardisContext, result::TardisResult}, - log::debug, - tokio, TardisFuns, TardisFunsInst, + basic::{dto::TardisContext, result::TardisResult}, log::debug, tokio, web::web_resp::TardisResp, TardisFuns, TardisFunsInst }; use crate::{ @@ -14,8 +12,7 @@ use crate::{ }, flow_state_dto::FlowSysStateKind, flow_transition_dto::{FlowTransitionActionByVarChangeInfoChangedKind, FlowTransitionDetailResp, TagRelKind}, - }, - flow_constants, + }, flow_config::FlowConfig, flow_constants }; pub struct FlowExternalServ; @@ -297,6 +294,26 @@ impl FlowExternalServ { } } + pub async fn do_find_embed_subrole_id(role_id: &str, ctx: &TardisContext, funs: &TardisFunsInst) -> TardisResult { + let iam_url = &funs.conf::().iam_url; + + let header = Self::headers(None, funs, ctx).await?; + let resp = funs + .web_client() + .get::>(&format!("{iam_url}/cc/role/get_embed_subrole_id?id={role_id}"), header) + .await? + .body + .ok_or_else(|| funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response"))?; + if resp.code != *"200" { + return Err(funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response")); + } + if let Some(data) = resp.data { + Ok(data) + } else { + Err(funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response")) + } + } + async fn get_external_url(tag: &str, ctx: &TardisContext, funs: &TardisFunsInst) -> TardisResult { let external_url = SpiKvClient::get_item(format!("{}:config:{}", flow_constants::DOMAIN_CODE, tag), None, funs, ctx) .await? diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index 393d7ef5..aa0ff3fc 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -27,7 +27,7 @@ use tardis::{ futures_util::future::join_all, log::{debug, error}, serde_json::Value, - web::web_resp::TardisPage, + web::{poem_openapi::types::Type, web_resp::TardisPage}, TardisFuns, TardisFunsInst, }; @@ -42,9 +42,8 @@ use crate::{ FlowInstSearchReq, FlowInstSearchSortReq, FlowInstStartReq, FlowInstSummaryResp, FlowInstSummaryResult, FlowInstTransferReq, FlowInstTransferResp, FlowInstTransitionInfo, FlowOperationContext, }, - flow_model_dto::FlowModelFilterReq, flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, - flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowSysStateKind}, + flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateCountersignKind, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowStatusMultiApprovalKind, FlowSysStateKind}, flow_transition_dto::FlowTransitionDetailResp, flow_var_dto::FillType, }, @@ -65,21 +64,19 @@ use super::{ pub struct FlowInstServ; impl FlowInstServ { - pub async fn start(start_req: &FlowInstStartReq, current_state_name: Option, _funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { - let mut funs = flow_constants::get_tardis_inst(); - funs.begin().await?; + pub async fn start(start_req: &FlowInstStartReq, current_state_name: Option, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { // get model by own_paths let flow_model = if let Some(transition_id) = &start_req.transition_id { - FlowModelServ::get_model_id_by_own_paths_and_transition_id(&start_req.tag, transition_id, &funs, ctx).await? + FlowModelServ::get_model_id_by_own_paths_and_transition_id(&start_req.tag, transition_id, funs, ctx).await? } else { - FlowModelServ::get_model_id_by_own_paths_and_rel_template_id(&start_req.tag, None, &funs, ctx).await? + FlowModelServ::get_model_id_by_own_paths_and_rel_template_id(&start_req.tag, None, funs, ctx).await? }; let inst_id = TardisFuns::field.nanoid(); let current_state_id = if let Some(current_state_name) = ¤t_state_name { if current_state_name.is_empty() { flow_model.init_state_id.clone() } else { - FlowStateServ::match_state_id_by_name(&flow_model.current_version_id, current_state_name, &funs, ctx).await? + FlowStateServ::match_state_id_by_name(&flow_model.current_version_id, current_state_name, funs, ctx).await? } } else { flow_model.init_state_id.clone() @@ -91,7 +88,7 @@ impl FlowInstServ { finish: Some(false), ..Default::default() }, - &funs, + funs, ctx, ) .await? @@ -99,6 +96,11 @@ impl FlowInstServ { { return Err(funs.err().internal_error("flow_inst_serv", "start", "The same instance exist", "500-flow-inst-exist")); } + let create_vars = if let Some(create_vars) = start_req.create_vars.clone() { + create_vars + } else { + Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await? + }; let main = start_req.transition_id.is_none(); let mut flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { id: Set(inst_id.clone()), @@ -108,33 +110,35 @@ impl FlowInstServ { current_state_id: Set(current_state_id.clone()), - create_vars: Set(start_req.create_vars.as_ref().map(|vars| TardisFuns::json.obj_to_json(vars).unwrap())), + create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), create_ctx: Set(FlowOperationContext::from_ctx(ctx)), - own_paths: Set(ctx.own_paths.to_string()), + own_paths: Set(ctx.own_paths.clone()), main: Set(main), ..Default::default() }; if !main { - flow_inst.code = Set(Some(Self::gen_inst_code(&funs).await?)); + flow_inst.code = Set(Some(Self::gen_inst_code(funs).await?)); + } else { + flow_inst.code = Set(Some("".to_string())); } funs.db().insert_one(flow_inst, ctx).await?; - let inst = Self::get(&inst_id, &funs, ctx).await?; - Self::when_enter_state(&inst, ¤t_state_id, &flow_model.id, &funs, ctx).await?; if !main { - FlowSearchClient::modify_business_obj_search(&start_req.rel_business_obj_id, &flow_model.tag, &funs, ctx).await?; + Self::modify_inst_artifacts(&inst_id, &FlowInstArtifactsModifyReq { + form_state_map: Some(create_vars), + ..Default::default() + }, funs, ctx).await?; + FlowSearchClient::modify_business_obj_search(&start_req.rel_business_obj_id, &flow_model.tag, funs, ctx).await?; } + let inst = Self::get(&inst_id, funs, ctx).await?; + Self::when_enter_state(&inst, ¤t_state_id, &flow_model.id, funs, ctx).await?; Self::do_request_webhook( None, flow_model.transitions().iter().filter(|model_transition| model_transition.to_flow_state_id == flow_model.init_state_id).collect_vec().pop(), ) .await?; - funs.commit().await?; - let mut funs = flow_constants::get_tardis_inst(); - funs.begin().await?; // 自动流转 - Self::auto_transfer(&inst, loop_check_helper::InstancesTransition::default(), &funs, ctx).await?; - funs.commit().await?; + Self::auto_transfer(&inst, loop_check_helper::InstancesTransition::default(), funs, ctx).await?; Ok(inst_id) } @@ -194,6 +198,7 @@ impl FlowInstServ { query .columns([ (flow_inst::Entity, flow_inst::Column::Id), + (flow_inst::Entity, flow_inst::Column::Code), (flow_inst::Entity, flow_inst::Column::RelFlowVersionId), (flow_inst::Entity, flow_inst::Column::RelBusinessObjId), (flow_inst::Entity, flow_inst::Column::CreateVars), @@ -1226,52 +1231,23 @@ impl FlowInstServ { Ok(()) } - async fn get_new_vars(flow_inst_detail: &FlowInstDetailResp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { - let flow_model_version = FlowModelVersionServ::get_item( - &flow_inst_detail.rel_flow_version_id, - &FlowModelVersionFilterReq { - basic: RbumBasicFilterReq { - with_sub_own_paths: true, - own_paths: Some("".to_string()), - ..Default::default() - }, - ..Default::default() - }, - funs, - ctx, - ) - .await?; - let flow_model = FlowModelServ::get_item( - &flow_model_version.rel_model_id, - &FlowModelFilterReq { - basic: RbumBasicFilterReq { - with_sub_own_paths: true, - own_paths: Some("".to_string()), - ..Default::default() - }, - ..Default::default() - }, - funs, - ctx, - ) - .await?; - Ok( - FlowExternalServ::do_query_field(&flow_model.tag, vec![flow_inst_detail.rel_business_obj_id.clone()], &flow_inst_detail.own_paths, ctx, funs) - .await? - .objs - .into_iter() - .next() - .unwrap_or_default(), - ) + async fn get_new_vars(tag: &str, rel_business_obj_ids: Vec, own_paths: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { + let new_vars = FlowExternalServ::do_query_field(tag, rel_business_obj_ids, own_paths, ctx, funs) + .await? + .objs + .into_iter() + .next() + .unwrap_or_default(); + Ok(TardisFuns::json.json_to_obj(new_vars).unwrap_or_default()) } pub async fn find_var_by_inst_id(flow_inst: &FlowInstDetailResp, key: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { let mut current_vars = flow_inst.current_vars.clone(); if current_vars.is_none() || !current_vars.clone().unwrap_or_default().contains_key(key) { - let new_vars = Self::get_new_vars(flow_inst, funs, ctx).await?; + let new_vars = Self::get_new_vars(&flow_inst.tag, vec![flow_inst.rel_business_obj_id.clone()], &flow_inst.own_paths, funs, ctx).await?; Self::modify_current_vars( flow_inst, - &TardisFuns::json.json_to_obj::>(new_vars).unwrap_or_default(), + &new_vars, loop_check_helper::InstancesTransition::default(), ctx, ) @@ -1553,6 +1529,12 @@ impl FlowInstServ { .collect::>(); } modify_req.guard_conf = Some(guard_custom_conf); + // 若会签,则需要统计审批人数 + if approval_conf.multi_approval_kind == FlowStatusMultiApprovalKind::Countersign { + // @TODO 此处缺少获取人数的逻辑 + let total = 10; + modify_req.curr_approval_total = Some(total); + } Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; // 当操作人为空时的逻辑 let curr_guard_conf = Self::get(&flow_inst_detail.id, funs, ctx).await?.artifacts.unwrap_or_default().guard_conf; @@ -1736,6 +1718,10 @@ impl FlowInstServ { let current_account_ids = current_state_result.entry(add_approval_result.to_string()).or_default(); current_account_ids.push(add_approval_account_id.clone()); } + if let Some(curr_approval_total) = modify_artifacts.curr_approval_total { + let approval_total = inst_artifacts.approval_total.entry(inst.current_state_id.clone()).or_default(); + *approval_total = curr_approval_total; + } if let Some(form_state_vars) = modify_artifacts.form_state_map.clone() { inst_artifacts.form_state_map.insert(inst.current_state_id.clone(), form_state_vars.clone()); } @@ -1903,21 +1889,23 @@ impl FlowInstServ { ctx, ) .await?; - if let Some(next_transition) = FlowInstServ::find_next_transitions(inst, &FlowInstFindNextTransitionsReq { vars: None }, funs, ctx).await?.pop() { - Self::transfer( - inst, - &FlowInstTransferReq { - flow_transition_id: next_transition.next_flow_transition_id, - message: None, - vars: None, - }, - false, - FlowExternalCallbackOp::Default, - loop_check_helper::InstancesTransition::default(), - ctx, - funs, - ) - .await?; + if Self::check_approval_cond(inst, FlowApprovalResultKind::Pass, funs, ctx).await? { + if let Some(next_transition) = FlowInstServ::find_next_transitions(inst, &FlowInstFindNextTransitionsReq { vars: None }, funs, ctx).await?.pop() { + Self::transfer( + inst, + &FlowInstTransferReq { + flow_transition_id: next_transition.next_flow_transition_id, + message: None, + vars: None, + }, + false, + FlowExternalCallbackOp::Default, + loop_check_helper::InstancesTransition::default(), + ctx, + funs, + ) + .await?; + } } } // 拒绝 @@ -1932,12 +1920,79 @@ impl FlowInstServ { ctx, ) .await?; - Self::abort(&inst.id, &FlowInstAbortReq { message: "".to_string() }, funs, ctx).await?; + if Self::check_approval_cond(inst, FlowApprovalResultKind::Overrule, funs, ctx).await? { + Self::abort(&inst.id, &FlowInstAbortReq { message: "".to_string() }, funs, ctx).await?; + } } } Ok(()) } + // 判断审批条件是否满足 + async fn check_approval_cond(inst: &FlowInstDetailResp , kind: FlowApprovalResultKind, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { + let current_state_kind_conf = FlowStateServ::get_item( + &inst.current_state_id, + &FlowStateFilterReq { + basic: RbumBasicFilterReq { + with_sub_own_paths: true, + own_paths: Some("".to_string()), + ..Default::default() + }, + ..Default::default() + }, + funs, + ctx, + ).await?.kind_conf().unwrap_or_default().approval; + let artifacts = inst.artifacts.clone().unwrap_or_default(); + let approval_total = artifacts.approval_total.get(&inst.current_state_id).cloned().unwrap_or_default(); + let approval_result = artifacts + .approval_result + .get(&inst.current_state_id) + .cloned() + .unwrap_or_default(); + if let Some(current_state_kind_conf) = current_state_kind_conf { + let countersign_conf = current_state_kind_conf.countersign_conf; + // 指定人通过,则通过 + if kind == FlowApprovalResultKind::Pass + && countersign_conf.specified_pass_guard.unwrap_or(false) + && countersign_conf.specified_pass_guard_conf.is_some() + && countersign_conf.specified_pass_guard_conf.unwrap().check(ctx) { + return Ok(true); + } + // 指定人拒绝,则拒绝 + if kind == FlowApprovalResultKind::Overrule + && countersign_conf.specified_overrule_guard.unwrap_or(false) + && countersign_conf.specified_overrule_guard_conf.is_some() + && countersign_conf.specified_overrule_guard_conf.unwrap().check(ctx) { + return Ok(true); + } + match countersign_conf.kind { + FlowStateCountersignKind::All => { + if kind == FlowApprovalResultKind::Overrule // 要求全数通过则出现一个拒绝,即拒绝 + || ( + kind == FlowApprovalResultKind::Pass + && approval_result.get(&FlowApprovalResultKind::Pass.to_string()).cloned().unwrap_or_default().len() + 1 >= approval_total + && approval_result.get(&FlowApprovalResultKind::Overrule.to_string()).cloned().unwrap_or_default().len().is_empty() // 要求全数通过则通过人数达到审核人数同时没有一个拒绝 + ) { + return Ok(true); + } + }, + FlowStateCountersignKind::Most => { + if countersign_conf.most_percent.is_none() { + return Ok(false); + } + let pass_total = approval_total * countersign_conf.most_percent.unwrap_or_default() / 100; // 需满足通过的人员数量 + let overrule_total = approval_total - pass_total;// 需满足拒绝的人员数量 + if (kind == FlowApprovalResultKind::Pass && approval_result.get(&FlowApprovalResultKind::Pass.to_string()).cloned().unwrap_or_default().len() + 1 >= pass_total) // 要求大多数通过则通过人数达到通过的比例 + || (kind == FlowApprovalResultKind::Overrule && approval_result.get(&FlowApprovalResultKind::Overrule.to_string()).cloned().unwrap_or_default().len() + 1 >= overrule_total) {// 要求大多数通过则拒绝人数达到拒绝的比例 + return Ok(true); + } + }, + } + } + Ok(false) + } + async fn transfer_spec_state(flow_inst_detail: &FlowInstDetailResp, target_state_id: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { let flow_model_version = FlowModelVersionServ::get_item( &flow_inst_detail.rel_flow_version_id, @@ -2314,6 +2369,7 @@ impl FlowInstServ { ) } + // 生成实例编码 async fn gen_inst_code(funs: &TardisFunsInst) -> TardisResult { let count = funs .db() From ba6f4adc18c8b1682a3fa44508f947804a6aa088 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Fri, 13 Dec 2024 19:06:09 +0800 Subject: [PATCH 09/11] flow: fix bug (inst start failed) --- .../middlewares/flow/src/dto/flow_inst_dto.rs | 16 +- .../flow/src/dto/flow_model_dto.rs | 2 +- .../flow/src/dto/flow_transition_dto.rs | 20 +- .../flow/src/serv/clients/flow_log_client.rs | 78 ++++- .../flow/src/serv/flow_inst_serv.rs | 322 ++++++++++++++---- .../flow/src/serv/flow_model_serv.rs | 23 +- .../flow/src/serv/flow_state_serv.rs | 21 +- 7 files changed, 382 insertions(+), 100 deletions(-) diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index 81380d2a..81734512 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -10,9 +10,7 @@ use tardis::{ }; use super::{ - flow_state_dto::{FlowGuardConf, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStateVar, FlowSysStateKind}, - flow_transition_dto::FlowTransitionDoubleCheckInfo, - flow_var_dto::FlowVarInfo, + flow_model_dto::FlowModelRelTransitionExt, flow_state_dto::{FlowGuardConf, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStateVar, FlowSysStateKind}, flow_transition_dto::FlowTransitionDoubleCheckInfo, flow_var_dto::FlowVarInfo }; #[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] @@ -104,6 +102,8 @@ pub struct FlowInstSummaryResp { pub finish_abort: bool, /// 输出信息 pub output_message: Option, + /// 触发的动作 + pub rel_transition: Option, pub own_paths: String, @@ -185,10 +185,12 @@ pub struct FlowInstDetailResp { pub output_message: Option, /// 动作列表 pub transitions: Option>, - + /// 数据对象 pub artifacts: Option, - + /// 评论 pub comments: Option>, + /// 触发的动作 + pub rel_transition: Option, pub own_paths: String, } @@ -220,7 +222,7 @@ pub struct FlowInstArtifacts { pub guard_conf: FlowGuardConf, // 当前操作人权限 pub prohibit_guard_by_spec_account_ids: Option>, // 禁止操作的指定用户ID pub approval_result: HashMap>>, // 当前审批结果 - pub approval_total: HashMap, // 审批总数 + pub approval_total: Option>, // 审批总数 pub form_state_map: HashMap>, // 录入节点映射 key为节点ID,对应的value为节点中的录入的参数 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID pub prev_non_auto_account_id: Option, // 上一个节点操作人ID @@ -505,6 +507,8 @@ pub struct FlowInstSummaryResult { pub finish_abort: Option, pub output_message: Option, + pub rel_transition: Option, + pub own_paths: String, pub tag: String, diff --git a/backend/middlewares/flow/src/dto/flow_model_dto.rs b/backend/middlewares/flow/src/dto/flow_model_dto.rs index e15a5a78..7748f030 100644 --- a/backend/middlewares/flow/src/dto/flow_model_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_model_dto.rs @@ -178,7 +178,7 @@ pub struct FlowModelSummaryResp { pub rel_transition: Option, } -#[derive(Serialize, Deserialize, Debug, Default, Clone, poem_openapi::Object, sea_orm::FromQueryResult)] +#[derive(Serialize, Deserialize, Clone, PartialEq, Default, Debug, poem_openapi::Object, sea_orm::FromJsonQueryResult)] pub struct FlowModelRelTransitionExt { pub id: String, pub name: String, diff --git a/backend/middlewares/flow/src/dto/flow_transition_dto.rs b/backend/middlewares/flow/src/dto/flow_transition_dto.rs index e7d9c6b2..20562eb1 100644 --- a/backend/middlewares/flow/src/dto/flow_transition_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_transition_dto.rs @@ -613,12 +613,30 @@ pub enum FlowTransitionFrontActionInfoRelevanceRelation { } impl FlowTransitionFrontActionInfoRelevanceRelation { - pub fn check_conform(&self, left_value: String, right_value: String) -> bool { + pub fn check_conform(&self, mut left_value: String, right_value: String) -> bool { use itertools::Itertools; if left_value.is_empty() || left_value == "null" || right_value == "null" { return false; } + // 单项判断(例如等于,不等于,大于,小于),如果参数是单元素数组,则取出数据,否则说明格式错误直接返回false + if *self == FlowTransitionFrontActionInfoRelevanceRelation::Eq + || *self == FlowTransitionFrontActionInfoRelevanceRelation::Ne + || *self == FlowTransitionFrontActionInfoRelevanceRelation::Gt + || *self == FlowTransitionFrontActionInfoRelevanceRelation::Ge + || *self == FlowTransitionFrontActionInfoRelevanceRelation::Lt + || *self == FlowTransitionFrontActionInfoRelevanceRelation::Le + { + let left_values = TardisFuns::json.str_to_obj::>(&left_value).unwrap_or_default(); + if left_values.len() == 1 { + left_value = left_values + .first().cloned() + .unwrap_or_default().as_str() + .unwrap_or("").to_string(); + } else { + return false; + } + } match self { FlowTransitionFrontActionInfoRelevanceRelation::Eq => left_value == right_value, FlowTransitionFrontActionInfoRelevanceRelation::Ne => left_value != right_value, diff --git a/backend/middlewares/flow/src/serv/clients/flow_log_client.rs b/backend/middlewares/flow/src/serv/clients/flow_log_client.rs index 10867772..a2917b4f 100644 --- a/backend/middlewares/flow/src/serv/clients/flow_log_client.rs +++ b/backend/middlewares/flow/src/serv/clients/flow_log_client.rs @@ -1,3 +1,5 @@ +use std::collections::HashMap; + use bios_sdk_invoke::clients::{ iam_client::IamClient, spi_log_client::{LogItemAddV2Req, SpiLogClient}, @@ -16,19 +18,69 @@ pub struct FlowLogClient; #[derive(Serialize, Default, Debug, Clone)] pub struct LogParamContent { - pub subject: String, - pub name: String, - pub sub_kind: String, + pub subject: Option, + pub name: Option, + pub sub_kind: Option, + pub sub_id: Option, + pub old_content: Option, + pub new_content: Option, + pub detail: Option, + pub operand: Option, + pub operand_id: Option, + pub operand_kind: Option, + pub operand_name: Option, + pub output_message: Option, + pub output_result: Option, +} + +#[derive(Serialize, Default, Debug, Clone)] +pub struct LogParamExt { + pub scene_kind: Option>, + pub sys_op: Option, + pub project_id: Option, + pub new_log: Option, + pub include_detail: Option, + pub delete: Option, } pub enum LogParamTag { DynamicLog, + ApprovalFlow, } impl From for String { fn from(val: LogParamTag) -> Self { match val { LogParamTag::DynamicLog => "dynamic_log".to_string(), + LogParamTag::ApprovalFlow => "approval_flow".to_string(), + } + } +} + +pub enum LogParamOp { + // 发起 + Start, + // 审批 + Approval, + // 审批流转 + ApprovalTransfer, + // 录入 + Form, + // 录入流转 + FormTransfer, + // 结束 + Finish, +} + +impl From for String { + fn from(val: LogParamOp) -> Self { + match val { + LogParamOp::Start => "FLOW_START".to_string(), + LogParamOp::Approval => "FLOW_APPROVAL".to_string(), + LogParamOp::ApprovalTransfer => "FLOW_APPROVAL_TRANSFER".to_string(), + LogParamOp::Form => "FLOW_FORM".to_string(), + LogParamOp::FormTransfer => "FLOW_FORM_TRANSFER".to_string(), + LogParamOp::Finish => "FLOW_FINISH".to_string(), } } } @@ -98,7 +150,7 @@ impl FlowLogClient { let req = LogItemAddV2Req { tag: tag.to_string(), - content: TardisFuns::json.obj_to_json(&content).expect("req_msg not a valid json value"), + content: TardisFuns::json.obj_to_json(&content).expect("content not a valid json value"), kind, ext, key, @@ -110,10 +162,26 @@ impl FlowLogClient { own_paths, msg: None, owner_name, - push: push, + push, disable: None, }; SpiLogClient::addv2(req, funs, ctx).await?; Ok(()) } + + pub fn get_flow_kind_text(tag: &str) -> String { + let flow_tag_map = HashMap::from([ + ("PROJ", "项目"), + ("MS", "里程碑"), + ("ITER", "迭代"), + ("TICKET", "工单"), + ("REQ", "需求"), + ("TASK", "任务"), + ("ISSUE", "缺陷"), + ("CTS", "转测单"), + ("TP", "测试计划"), + ("TS", "测试阶段"), + ]); + flow_tag_map.get(tag).map_or("".to_string(), |val| val.to_string()) + } } diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index aa0ff3fc..d0b214e4 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -7,11 +7,10 @@ use async_recursion::async_recursion; use bios_basic::{ dto::BasicQueryCondInfo, rbum::{ - dto::rbum_filer_dto::RbumBasicFilterReq, - serv::{ + dto::rbum_filer_dto::RbumBasicFilterReq, helper::rbum_scope_helper, rbum_enumeration::RbumScopeLevelKind, serv::{ rbum_crud_serv::{ID_FIELD, NAME_FIELD, REL_DOMAIN_ID_FIELD, REL_KIND_ID_FIELD}, rbum_item_serv::{RbumItemCrudOperation, RBUM_ITEM_TABLE}, - }, + } }, }; use itertools::Itertools; @@ -32,20 +31,15 @@ use tardis::{ }; use crate::{ - domain::flow_inst, + domain::{flow_inst, flow_model_version}, dto::{ - flow_external_dto::{FlowExternalCallbackOp, FlowExternalParams}, - flow_inst_dto::{ + flow_external_dto::{FlowExternalCallbackOp, FlowExternalParams}, flow_inst_dto::{ FLowInstStateApprovalConf, FLowInstStateConf, FLowInstStateFormConf, FlowApprovalResultKind, FlowInstAbortReq, FlowInstArtifacts, FlowInstArtifactsModifyReq, FlowInstBatchBindReq, FlowInstBatchBindResp, FlowInstCommentInfo, FlowInstCommentReq, FlowInstDetailResp, FlowInstFilterReq, FlowInstFindNextTransitionResp, FlowInstFindNextTransitionsReq, FlowInstFindStateAndTransitionsReq, FlowInstFindStateAndTransitionsResp, FlowInstOperateReq, FlowInstQueryKind, FlowInstSearchPageReq, FlowInstSearchReq, FlowInstSearchSortReq, FlowInstStartReq, FlowInstSummaryResp, FlowInstSummaryResult, FlowInstTransferReq, FlowInstTransferResp, FlowInstTransitionInfo, FlowOperationContext, - }, - flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, - flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateCountersignKind, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowStatusMultiApprovalKind, FlowSysStateKind}, - flow_transition_dto::FlowTransitionDetailResp, - flow_var_dto::FillType, + }, flow_model_dto::{FlowModelDetailResp, FlowModelRelTransitionExt}, flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateCountersignKind, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowStatusMultiApprovalKind, FlowSysStateKind}, flow_transition_dto::FlowTransitionDetailResp, flow_var_dto::FillType }, flow_constants, helper::loop_check_helper, @@ -53,7 +47,7 @@ use crate::{ }; use super::{ - clients::search_client::FlowSearchClient, + clients::{flow_log_client::{FlowLogClient, LogParamContent, LogParamExt, LogParamOp, LogParamTag}, search_client::FlowSearchClient}, flow_event_serv::FlowEventServ, flow_external_serv::FlowExternalServ, flow_model_version_serv::FlowModelVersionServ, @@ -65,12 +59,15 @@ pub struct FlowInstServ; impl FlowInstServ { pub async fn start(start_req: &FlowInstStartReq, current_state_name: Option, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { - // get model by own_paths - let flow_model = if let Some(transition_id) = &start_req.transition_id { - FlowModelServ::get_model_id_by_own_paths_and_transition_id(&start_req.tag, transition_id, funs, ctx).await? + if start_req.transition_id.is_none() { + Self::start_main_flow(start_req, current_state_name, funs, ctx).await } else { - FlowModelServ::get_model_id_by_own_paths_and_rel_template_id(&start_req.tag, None, funs, ctx).await? - }; + Self::start_secondary_flow(start_req, funs, ctx).await + } + } + async fn start_main_flow(start_req: &FlowInstStartReq, current_state_name: Option, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { + // get model by own_paths + let flow_model = FlowModelServ::get_model_id_by_own_paths_and_rel_template_id(&start_req.tag, None, funs, ctx).await?; let inst_id = TardisFuns::field.nanoid(); let current_state_id = if let Some(current_state_name) = ¤t_state_name { if current_state_name.is_empty() { @@ -81,6 +78,38 @@ impl FlowInstServ { } else { flow_model.init_state_id.clone() }; + let create_vars = Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await?; + let flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { + id: Set(inst_id.clone()), + code: Set(Some("".to_string())), + tag: Set(Some(flow_model.tag.clone())), + rel_flow_version_id: Set(flow_model.current_version_id.clone()), + rel_business_obj_id: Set(start_req.rel_business_obj_id.clone()), + + current_state_id: Set(current_state_id.clone()), + + create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + create_ctx: Set(FlowOperationContext::from_ctx(ctx)), + + own_paths: Set(ctx.own_paths.clone()), + main: Set(true), + ..Default::default() + }; + funs.db().insert_one(flow_inst, ctx).await?; + + Self::do_request_webhook( + None, + flow_model.transitions().iter().filter(|model_transition| model_transition.to_flow_state_id == flow_model.init_state_id).collect_vec().pop(), + ) + .await?; + + Ok(inst_id) + } + + async fn start_secondary_flow(start_req: &FlowInstStartReq, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult { + // get model by own_paths + let flow_model = FlowModelServ::get_model_id_by_own_paths_and_transition_id(&start_req.tag, &start_req.transition_id.clone().unwrap_or_default(), funs, ctx).await?; + let inst_id = TardisFuns::field.nanoid(); if !Self::find_details( &FlowInstFilterReq { rel_business_obj_ids: Some(vec![start_req.rel_business_obj_id.to_string()]), @@ -96,42 +125,33 @@ impl FlowInstServ { { return Err(funs.err().internal_error("flow_inst_serv", "start", "The same instance exist", "500-flow-inst-exist")); } - let create_vars = if let Some(create_vars) = start_req.create_vars.clone() { - create_vars - } else { - Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await? - }; - let main = start_req.transition_id.is_none(); - let mut flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { + let create_vars = Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await?; + let flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { id: Set(inst_id.clone()), + code: Set(Some(Self::gen_inst_code(funs).await?)), tag: Set(Some(flow_model.tag.clone())), rel_flow_version_id: Set(flow_model.current_version_id.clone()), - rel_business_obj_id: Set(start_req.rel_business_obj_id.to_string()), + rel_business_obj_id: Set(start_req.rel_business_obj_id.clone()), - current_state_id: Set(current_state_id.clone()), + current_state_id: Set(flow_model.init_state_id.clone()), create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), create_ctx: Set(FlowOperationContext::from_ctx(ctx)), own_paths: Set(ctx.own_paths.clone()), - main: Set(main), + main: Set(false), ..Default::default() }; - if !main { - flow_inst.code = Set(Some(Self::gen_inst_code(funs).await?)); - } else { - flow_inst.code = Set(Some("".to_string())); - } funs.db().insert_one(flow_inst, ctx).await?; - if !main { - Self::modify_inst_artifacts(&inst_id, &FlowInstArtifactsModifyReq { - form_state_map: Some(create_vars), - ..Default::default() - }, funs, ctx).await?; - FlowSearchClient::modify_business_obj_search(&start_req.rel_business_obj_id, &flow_model.tag, funs, ctx).await?; - } + Self::modify_inst_artifacts(&inst_id, &FlowInstArtifactsModifyReq { + form_state_map: start_req.create_vars.clone(), + ..Default::default() + }, funs, ctx).await?; + FlowSearchClient::modify_business_obj_search(&start_req.rel_business_obj_id, &flow_model.tag, funs, ctx).await?; let inst = Self::get(&inst_id, funs, ctx).await?; - Self::when_enter_state(&inst, ¤t_state_id, &flow_model.id, funs, ctx).await?; + Self::add_start_log(start_req, &inst, &create_vars, &flow_model, ctx).await?; + + Self::when_enter_state(&inst, &flow_model.init_state_id, &flow_model.id, funs, ctx).await?; Self::do_request_webhook( None, flow_model.transitions().iter().filter(|model_transition| model_transition.to_flow_state_id == flow_model.init_state_id).collect_vec().pop(), @@ -195,6 +215,7 @@ impl FlowInstServ { async fn package_ext_query(query: &mut SelectStatement, filter: &FlowInstFilterReq, _: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { let flow_model_version_table = Alias::new("flow_model_version"); + let rel_model_table = Alias::new("rbum_rel"); query .columns([ (flow_inst::Entity, flow_inst::Column::Id), @@ -217,6 +238,10 @@ impl FlowInstServ { Alias::new("rel_flow_model_id"), ) .expr_as(Expr::col((RBUM_ITEM_TABLE.clone(), NAME_FIELD.clone())).if_null(""), Alias::new("rel_flow_model_name")) + .expr_as( + Expr::col((rel_model_table.clone(), Alias::new("ext"))).if_null(""), + Alias::new("rel_transition"), + ) .from(flow_inst::Entity) .left_join( flow_model_version_table.clone(), @@ -225,6 +250,12 @@ impl FlowInstServ { .left_join( RBUM_ITEM_TABLE.clone(), Expr::col((RBUM_ITEM_TABLE.clone(), ID_FIELD.clone())).equals((flow_inst::Entity, flow_inst::Column::RelFlowVersionId)), + ) + .left_join( + rel_model_table.clone(), + Cond::all() + .add(Expr::col((rel_model_table.clone(), Alias::new("from_rbum_id"))).equals((flow_model_version_table.clone(), flow_model_version::Column::RelModelId))) + .add(Expr::col((rel_model_table.clone(), Alias::new("tag"))).eq("FlowModelTransition".to_string())), ); if let Some(ids) = &filter.ids { query.and_where(Expr::col((flow_inst::Entity, flow_inst::Column::Id)).is_in(ids)); @@ -367,14 +398,18 @@ impl FlowInstServ { pub artifacts: Option, pub comments: Option, + pub rel_transition: Option, + pub own_paths: String, pub rel_business_obj_id: String, } let rel_state_table = Alias::new("rel_state"); let flow_state_table = Alias::new("flow_state"); + let flow_model_version_table = Alias::new("flow_model_version"); let rel_model_version_table = Alias::new("rel_model_version"); - let rbum_rel_table = Alias::new("rbum_rel"); + let rel_state_ext_table = Alias::new("rel_state_ext"); + let rel_model_table = Alias::new("rel_model"); let mut query = Query::select(); query .columns([ @@ -412,11 +447,15 @@ impl FlowInstServ { Expr::col((flow_state_table.clone(), Alias::new("kind_conf"))).if_null(json!({})), Alias::new("current_state_kind_conf"), ) - .expr_as(Expr::col((rbum_rel_table.clone(), Alias::new("ext"))).if_null(""), Alias::new("current_state_ext")) + .expr_as(Expr::col((rel_state_ext_table.clone(), Alias::new("ext"))).if_null(""), Alias::new("current_state_ext")) .expr_as( Expr::col((rel_model_version_table.clone(), NAME_FIELD.clone())).if_null(""), Alias::new("rel_flow_model_name"), ) + .expr_as( + Expr::col((rel_model_table.clone(), Alias::new("ext"))).if_null(""), + Alias::new("rel_transition"), + ) .from(flow_inst::Entity) .join_as( JoinType::LeftJoin, @@ -444,12 +483,27 @@ impl FlowInstServ { ) .join_as( JoinType::LeftJoin, - rbum_rel_table.clone(), - rbum_rel_table.clone(), + Alias::new("rbum_rel"), + rel_state_ext_table.clone(), + Cond::all() + .add(Expr::col((rel_state_ext_table.clone(), Alias::new("to_rbum_item_id"))).equals((flow_inst::Entity, flow_inst::Column::CurrentStateId))) + .add(Expr::col((rel_state_ext_table.clone(), Alias::new("from_rbum_id"))).equals((flow_inst::Entity, flow_inst::Column::RelFlowVersionId))) + .add(Expr::col((rel_state_ext_table.clone(), Alias::new("tag"))).eq("FlowModelState".to_string())), + ) + .join_as( + JoinType::LeftJoin, + flow_model_version_table.clone(), + flow_model_version_table.clone(), + Cond::all() + .add(Expr::col((flow_model_version_table.clone(), ID_FIELD.clone())).equals((flow_inst::Entity, flow_inst::Column::RelFlowVersionId))), + ) + .join_as( + JoinType::LeftJoin, + Alias::new("rbum_rel"), + rel_model_table.clone(), Cond::all() - .add(Expr::col((rbum_rel_table.clone(), Alias::new("to_rbum_item_id"))).equals((flow_inst::Entity, flow_inst::Column::CurrentStateId))) - .add(Expr::col((rbum_rel_table.clone(), Alias::new("from_rbum_id"))).equals((flow_inst::Entity, flow_inst::Column::RelFlowVersionId))) - .add(Expr::col((rbum_rel_table.clone(), Alias::new("tag"))).eq("FlowModelState".to_string())), + .add(Expr::col((rel_model_table.clone(), Alias::new("from_rbum_id"))).equals((flow_model_version_table.clone(), flow_model_version::Column::RelModelId))) + .add(Expr::col((rel_model_table.clone(), Alias::new("tag"))).eq("FlowModelTransition".to_string())), ) .and_where(Expr::col((flow_inst::Entity, flow_inst::Column::Id)).is_in(flow_inst_ids)) .and_where(Expr::col((flow_inst::Entity, flow_inst::Column::OwnPaths)).like(format!("{}%", ctx.own_paths))); @@ -481,6 +535,7 @@ impl FlowInstServ { transitions: inst.transitions.map(|transitions| TardisFuns::json.json_to_obj(transitions).unwrap()), artifacts: inst.artifacts.map(|artifacts| TardisFuns::json.json_to_obj(artifacts).unwrap()), comments: inst.comments.map(|comments| TardisFuns::json.json_to_obj(comments).unwrap()), + rel_transition: inst.rel_transition.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), current_state_id: inst.current_state_id.clone(), current_state_name: inst.current_state_name, current_state_color: inst.current_state_color, @@ -553,6 +608,7 @@ impl FlowInstServ { own_paths: inst.own_paths, current_state_id: inst.current_state_id, rel_business_obj_id: inst.rel_business_obj_id, + rel_transition: inst.rel_transition.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), tag: inst.tag, }) .collect_vec(), @@ -706,23 +762,26 @@ impl FlowInstServ { let result = Self::do_transfer(flow_inst_detail, transfer_req, skip_filter, callback_kind, funs, ctx).await; Self::auto_transfer(flow_inst_detail, modified_instance_transations_cp.clone(), funs, ctx).await?; - let flow_inst_id_cp = flow_inst_detail.id.clone(); - let flow_transition_id = transfer_req.flow_transition_id.clone(); - let ctx_cp = ctx.clone(); - tardis::tokio::spawn(async move { - let mut funs = flow_constants::get_tardis_inst(); - let flow_inst_cp = Self::get(&flow_inst_id_cp, &funs, &ctx_cp).await.unwrap(); - funs.begin().await.unwrap(); - match FlowEventServ::do_post_change(&flow_inst_cp, &flow_transition_id, modified_instance_transations_cp.clone(), &ctx_cp, &funs).await { - Ok(_) => {} - Err(e) => error!("Flow Instance {} do_post_change error:{:?}", flow_inst_id_cp, e), - } - match FlowEventServ::do_front_change(&flow_inst_cp, modified_instance_transations_cp.clone(), &ctx_cp, &funs).await { - Ok(_) => {} - Err(e) => error!("Flow Instance {} do_front_change error:{:?}", flow_inst_id_cp, e), - } - funs.commit().await.unwrap(); - }); + + if flow_inst_detail.main { + let flow_inst_id_cp = flow_inst_detail.id.clone(); + let flow_transition_id = transfer_req.flow_transition_id.clone(); + let ctx_cp = ctx.clone(); + tardis::tokio::spawn(async move { + let mut funs = flow_constants::get_tardis_inst(); + let flow_inst_cp = Self::get(&flow_inst_id_cp, &funs, &ctx_cp).await.unwrap(); + funs.begin().await.unwrap(); + match FlowEventServ::do_post_change(&flow_inst_cp, &flow_transition_id, modified_instance_transations_cp.clone(), &ctx_cp, &funs).await { + Ok(_) => {} + Err(e) => error!("Flow Instance {} do_post_change error:{:?}", flow_inst_id_cp, e), + } + match FlowEventServ::do_front_change(&flow_inst_cp, modified_instance_transations_cp.clone(), &ctx_cp, &funs).await { + Ok(_) => {} + Err(e) => error!("Flow Instance {} do_front_change error:{:?}", flow_inst_id_cp, e), + } + funs.commit().await.unwrap(); + }); + } result } @@ -1719,8 +1778,10 @@ impl FlowInstServ { current_account_ids.push(add_approval_account_id.clone()); } if let Some(curr_approval_total) = modify_artifacts.curr_approval_total { - let approval_total = inst_artifacts.approval_total.entry(inst.current_state_id.clone()).or_default(); - *approval_total = curr_approval_total; + if let Some(approval_total) = inst_artifacts.approval_total.as_mut() { + let old_approval_total = approval_total.entry(inst.current_state_id.clone()).or_default(); + *old_approval_total = curr_approval_total; + } } if let Some(form_state_vars) = modify_artifacts.form_state_map.clone() { inst_artifacts.form_state_map.insert(inst.current_state_id.clone(), form_state_vars.clone()); @@ -1782,7 +1843,13 @@ impl FlowInstServ { FlowStateKind::Approval => kind_conf.approval.as_ref().map(|approval| { let mut operators = HashMap::new(); let artifacts = artifacts.clone().unwrap_or_default(); - if artifacts.guard_conf.check(ctx) && !artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) { + if artifacts.guard_conf.check(ctx) + && !artifacts.prohibit_guard_by_spec_account_ids.clone().unwrap_or_default().contains(&ctx.owner) + && !( + artifacts.approval_result.get(state_id).cloned().unwrap_or_default().get(FlowApprovalResultKind::Pass.to_string().as_str()).cloned().unwrap_or_default().contains(&ctx.owner) + || artifacts.approval_result.get(state_id).cloned().unwrap_or_default().get(FlowApprovalResultKind::Overrule.to_string().as_str()).cloned().unwrap_or_default().contains(&ctx.owner) + ) + { operators.insert(FlowStateOperatorKind::Pass, approval.pass_btn_name.clone()); operators.insert(FlowStateOperatorKind::Overrule, approval.overrule_btn_name.clone()); operators.insert(FlowStateOperatorKind::Back, approval.back_btn_name.clone()); @@ -1944,7 +2011,7 @@ impl FlowInstServ { ctx, ).await?.kind_conf().unwrap_or_default().approval; let artifacts = inst.artifacts.clone().unwrap_or_default(); - let approval_total = artifacts.approval_total.get(&inst.current_state_id).cloned().unwrap_or_default(); + let approval_total = artifacts.approval_total.unwrap_or_default().get(&inst.current_state_id).cloned().unwrap_or_default(); let approval_result = artifacts .approval_result .get(&inst.current_state_id) @@ -2142,13 +2209,15 @@ impl FlowInstServ { flow_inst.own_paths, flow_inst.tag, model_version.name as rel_flow_model_name, - flow_model_version.rel_model_id as rel_flow_model_id + flow_model_version.rel_model_id as rel_flow_model_id, + rbum_rel.ext as rel_transition {} FROM flow_inst LEFT JOIN flow_state AS current_state ON flow_inst.current_state_id = current_state.id LEFT JOIN rbum_item AS model_version ON flow_inst.rel_flow_version_id = model_version.id LEFT JOIN flow_model_version ON flow_inst.rel_flow_version_id = flow_model_version.id + LEFT JOIN rbum_rel ON flow_model_version.rel_model_id = rbum_rel.from_rbum_id AND rbum_rel.tag = 'FlowModelTransition' WHERE {} {} @@ -2181,6 +2250,7 @@ impl FlowInstServ { rel_flow_model_id: item.try_get("", "rel_flow_model_id")?, rel_flow_model_name: item.try_get("", "rel_flow_model_name")?, rel_business_obj_id: item.try_get("", "rel_business_obj_id")?, + rel_transition: item.try_get::>("", "rel_transition")?.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), current_state_id: item.try_get("", "current_state_id")?, create_ctx: item.try_get("", "create_ctx")?, create_time: item.try_get("", "create_time")?, @@ -2384,4 +2454,118 @@ impl FlowInstServ { let current_date = Utc::now(); Ok(format!("SP{}{}{}{:0>5}", current_date.year(), current_date.month(), current_date.day(), count + 1).to_string()) } + + // 添加审批流发起日志 + async fn add_start_log(start_req: &FlowInstStartReq, flow_inst_detail: &FlowInstDetailResp, create_vars: &HashMap, flow_model: &FlowModelDetailResp, ctx: &TardisContext) -> TardisResult<()> { + let rel_transition = flow_model.rel_transition().unwrap_or_default(); + let operand = match rel_transition.id.as_str() { + "__EDIT__" => { + "编辑审批".to_string() + }, + "__DELETE__" => { + "删除审批".to_string() + } + _ => { + format!("{}({})", rel_transition.name, rel_transition.from_flow_state_name).to_string() + }, + }; + let mut log_ext = LogParamExt { + scene_kind: Some(vec!["approval_flow".to_string()]), + sys_op: Some(ctx.owner.clone()), + new_log: Some(true), + project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + ..Default::default() + }; + let mut log_content = LogParamContent { + subject: Some(FlowLogClient::get_flow_kind_text(&start_req.tag)), + name: Some(create_vars.get("name").map_or("".to_string(), |val| val.as_str().unwrap_or("").to_string())), + sub_id: Some(start_req.rel_business_obj_id.clone()), + sub_kind: None, // @TODO 缺少类型 + operand: Some(operand), + operand_id: Some(flow_inst_detail.id.clone()), + operand_name: Some(flow_inst_detail.code.clone()), + operand_kind: None, + ..Default::default() + }; + if start_req.create_vars.is_none() { + log_ext.include_detail = Some(false); + } else { + log_content.old_content = Some(create_vars.get("content").map_or("".to_string(), |val| val.as_str().unwrap_or("").to_string())); + log_content.new_content = start_req.create_vars.clone().unwrap_or_default().get("content").map(|content| content.as_str().unwrap_or("").to_string()); + log_content.detail = None; // @TODO 暂未实现 + log_ext.include_detail = Some(true); + } + FlowLogClient::add_ctx_task( + LogParamTag::ApprovalFlow, + Some(flow_inst_detail.id.clone()), + log_content, + Some(TardisFuns::json.obj_to_json(&log_ext).expect("ext not a valid json value")), + Some("dynamic_log_approval_flow".to_string()), + Some(LogParamOp::Start.into()), + rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + ctx, + false, + ) + .await?; + Ok(()) + } + + // 添加审批流操作日志 + // async fn add_operate_log(operate_req: &FlowInstOperateReq, flow_inst_detail: &FlowInstDetailResp, flow_model: &FlowModelDetailResp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { + // let current_state = FlowStateServ::get_item( + // &flow_inst_detail.current_state_id, + // &FlowStateFilterReq { + // basic: RbumBasicFilterReq { + // with_sub_own_paths: true, + // own_paths: Some("".to_string()), + // ..Default::default() + // }, + // ..Default::default() + // }, + // funs, + // ctx, + // ).await?; + // let subject_text = match { + + // }; + // let mut log_ext = LogParamExt { + // scene_kind: Some(vec!["approval_flow".to_string()]), + // sys_op: Some(ctx.owner.clone()), + // new_log: Some(true), + // project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + // ..Default::default() + // }; + // let mut log_content = LogParamContent { + // subject: Some(FlowLogClient::get_flow_kind_text(&start_req.tag)), + // name: Some(create_vars.get("name").map_or("".to_string(), |val| TardisFuns::json.json_to_string(val.clone()).unwrap_or_default())), + // sub_id: Some(start_req.rel_business_obj_id.clone()), + // sub_kind: None, // @TODO 缺少类型 + // operand: Some(operand), + // operand_id: Some(flow_inst_detail.id.clone()), + // operand_name: Some(flow_inst_detail.code.clone()), + // operand_kind: None, + // ..Default::default() + // }; + // if operate_req.vars.is_none() { + // log_ext.include_detail = Some(false); + // } else { + // log_content.old_content = Some(create_vars.get("content").map_or("".to_string(), |val| TardisFuns::json.json_to_string(val.clone()).unwrap_or_default())); + // log_content.new_content = operate_req.vars.clone().unwrap_or_default().get("content").map(|content| TardisFuns::json.json_to_string(content.clone()).unwrap_or_default()); + // log_content.detail = None; // @TODO 暂未实现 + // log_ext.include_detail = Some(true); + // } + // FlowLogClient::add_ctx_task( + // LogParamTag::ApprovalFlow, + // Some(flow_inst_detail.id.clone()), + // log_content, + // Some(TardisFuns::json.obj_to_json(&log_ext).expect("ext not a valid json value")), + // Some("dynamic_log_approval_flow".to_string()), + // Some(LogParamOp::Start.into()), + // rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + // ctx, + // false, + // ) + // .await?; + // Ok(()) + // } } diff --git a/backend/middlewares/flow/src/serv/flow_model_serv.rs b/backend/middlewares/flow/src/serv/flow_model_serv.rs index c4a15100..131d459b 100644 --- a/backend/middlewares/flow/src/serv/flow_model_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_model_serv.rs @@ -239,9 +239,10 @@ impl RbumItemCrudOperation Date: Sat, 14 Dec 2024 14:57:40 +0800 Subject: [PATCH 10/11] flow: improve log --- .../flow/src/api/cs/flow_cs_config_api.rs | 76 -------- .../middlewares/flow/src/dto/flow_inst_dto.rs | 10 +- .../flow/src/serv/clients/flow_log_client.rs | 18 +- .../flow/src/serv/clients/search_client.rs | 37 ++-- .../flow/src/serv/flow_external_serv.rs | 20 --- .../invoke/src/clients/spi_search_client.rs | 10 +- .../sdks/invoke/src/dto/search_item_dto.rs | 170 ++++++++++++++++++ 7 files changed, 224 insertions(+), 117 deletions(-) diff --git a/backend/middlewares/flow/src/api/cs/flow_cs_config_api.rs b/backend/middlewares/flow/src/api/cs/flow_cs_config_api.rs index 845c322f..3f86be9d 100644 --- a/backend/middlewares/flow/src/api/cs/flow_cs_config_api.rs +++ b/backend/middlewares/flow/src/api/cs/flow_cs_config_api.rs @@ -47,80 +47,4 @@ impl FlowCsConfigApi { ctx.0.execute_task().await?; TardisResp::ok(result) } - - /// Get Config / 获取配置 - #[oai(path = "/update_instance_state", method = "get")] - async fn update_instance_state(&self, ctx: TardisContextExtractor) -> TardisApiResult { - tokio::spawn(async move { - let global_ctx = TardisContext::default(); - let funs = flow_constants::get_tardis_inst(); - let tag_search_map = HashMap::from([ - ("CTS", "idp_test"), - ("ISSUE", "idp_test"), - ("ITER", "idp_project"), - ("MS", "idp_project"), - ("PROJ", "idp_project"), - ("REQ", "idp_project"), - ("TASK", "idp_project"), - ("TICKET", "ticket"), - ("TP", "idp_test"), - ("TS", "idp_test"), - ]); - let states = FlowStateServ::find_id_name_items( - &FlowStateFilterReq { - basic: RbumBasicFilterReq { - ignore_scope: true, - with_sub_own_paths: true, - ..Default::default() - }, - ..Default::default() - }, - None, - None, - &funs, - &global_ctx, - ) - .await - .unwrap(); - let mut page = 1; - loop { - let insts = FlowInstServ::paginate(None, None, None, None, None, None, Some(true), page, 200, &funs, &global_ctx).await.unwrap().records; - if insts.is_empty() { - break; - } - for inst in insts { - let state_name = states.get(&inst.current_state_id).cloned().unwrap_or_default(); - if let Some(table) = tag_search_map.get(&inst.tag.as_str()) { - SpiSearchClient::modify_item_and_name( - table, - &inst.rel_business_obj_id, - &SearchItemModifyReq { - kind: None, - title: None, - name: None, - content: None, - owner: None, - own_paths: None, - create_time: None, - update_time: None, - ext: Some(json!({ - "status": state_name, - })), - ext_override: None, - visit_keys: None, - kv_disable: None, - }, - &funs, - &global_ctx, - ) - .await - .unwrap_or_default(); - } - } - page += 1; - } - }); - ctx.0.execute_task().await?; - TardisResp::ok(Void {}) - } } diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index 81734512..78b7d3e8 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -222,8 +222,9 @@ pub struct FlowInstArtifacts { pub guard_conf: FlowGuardConf, // 当前操作人权限 pub prohibit_guard_by_spec_account_ids: Option>, // 禁止操作的指定用户ID pub approval_result: HashMap>>, // 当前审批结果 - pub approval_total: Option>, // 审批总数 + pub approval_total: Option>, // 审批总数 pub form_state_map: HashMap>, // 录入节点映射 key为节点ID,对应的value为节点中的录入的参数 + pub curr_vars: Option>, // 当前参数列表 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID pub prev_non_auto_account_id: Option, // 上一个节点操作人ID } @@ -242,7 +243,8 @@ pub struct FlowInstArtifactsModifyReq { pub clear_approval_result: Option, // 清除节点审批信息 pub prev_non_auto_state_id: Option, // 上一个非自动节点ID pub prev_non_auto_account_id: Option, // 上一个节点操作人ID - pub curr_approval_total: Option, // 当前审批总数 + pub curr_approval_total: Option, // 当前审批总数 + pub curr_vars: Option>, // 当前参数列表 } /// 审批结果类型 @@ -457,8 +459,10 @@ pub struct FlowInstModifyCurrentVarsReq { #[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] pub struct FlowInstOperateReq { pub operate: FlowStateOperatorKind, - /// 参数列表 + /// 修改参数列表 pub vars: Option>, + /// 全量参数列表 + pub all_vars: Option>, /// 输出信息 pub output_message: Option, /// 操作人 diff --git a/backend/middlewares/flow/src/serv/clients/flow_log_client.rs b/backend/middlewares/flow/src/serv/clients/flow_log_client.rs index a2917b4f..79548d47 100644 --- a/backend/middlewares/flow/src/serv/clients/flow_log_client.rs +++ b/backend/middlewares/flow/src/serv/clients/flow_log_client.rs @@ -29,8 +29,8 @@ pub struct LogParamContent { pub operand_id: Option, pub operand_kind: Option, pub operand_name: Option, - pub output_message: Option, - pub output_result: Option, + pub flow_message: Option, + pub flow_result: Option, } #[derive(Serialize, Default, Debug, Clone)] @@ -184,4 +184,18 @@ impl FlowLogClient { ]); flow_tag_map.get(tag).map_or("".to_string(), |val| val.to_string()) } + + pub fn get_junp_kind(tag: &str) -> String { + let flow_tag_map = HashMap::from([ + ("MS", "idp_feed_ms"), + ("ITER", "idp_feed_iter "), + ("REQ", "idp_feed_req"), + ("TASK", "idp_feed_task"), + ("ISSUE", "idp_test_issue"), + ("CTS", "idp_test_cts"), + ("TP", "idp_test_plan"), + ("TS", "idp_test_stage"), + ]); + flow_tag_map.get(tag).map_or("".to_string(), |val| val.to_string()) + } } diff --git a/backend/middlewares/flow/src/serv/clients/search_client.rs b/backend/middlewares/flow/src/serv/clients/search_client.rs index 8216d16f..f86fc7fa 100644 --- a/backend/middlewares/flow/src/serv/clients/search_client.rs +++ b/backend/middlewares/flow/src/serv/clients/search_client.rs @@ -6,13 +6,12 @@ use bios_sdk_invoke::{ event_client::{get_topic, EventCenterClient, SPI_RPC_TOPIC}, spi_search_client::SpiSearchClient, }, - dto::search_item_dto::{SearchItemAddReq, SearchItemModifyReq, SearchItemVisitKeysReq}, + dto::search_item_dto::{SearchItemAddReq, SearchItemModifyReq, SearchItemSearchReq, SearchItemSearchResp, SearchItemVisitKeysReq}, }; use itertools::Itertools; use serde_json::json; use tardis::{ - basic::{dto::TardisContext, field::TrimString, result::TardisResult}, - tokio, TardisFuns, TardisFunsInst, + basic::{dto::TardisContext, field::TrimString, result::TardisResult}, tokio, web::web_resp::TardisPage, TardisFuns, TardisFunsInst }; use crate::{ @@ -36,18 +35,7 @@ pub struct FlowSearchClient; impl FlowSearchClient { pub async fn modify_business_obj_search(rel_business_obj_id: &str, tag: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { - let tag_search_map = HashMap::from([ - ("CTS", "idp_test"), - ("ISSUE", "idp_test"), - ("ITER", "idp_project"), - ("MS", "idp_project"), - ("PROJ", "idp_project"), - ("REQ", "idp_project"), - ("TASK", "idp_project"), - ("TICKET", "ticket"), - ("TP", "idp_test"), - ("TS", "idp_test"), - ]); + let tag_search_map = Self::get_tag_search_map(); let rel_version_ids = FlowInstServ::find_details( &FlowInstFilterReq { rel_business_obj_ids: Some(vec![rel_business_obj_id.to_string()]), @@ -264,4 +252,23 @@ impl FlowSearchClient { } Ok(()) } + + pub async fn search(search_req: &SearchItemSearchReq, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult>> { + SpiSearchClient::search(search_req, funs, ctx).await + } + + pub fn get_tag_search_map() -> HashMap { + HashMap::from([ + ("CTS".to_string(), "idp_test".to_string()), + ("ISSUE".to_string(), "idp_test".to_string()), + ("ITER".to_string(), "idp_project".to_string()), + ("MS".to_string(), "idp_project".to_string()), + ("PROJ".to_string(), "idp_project".to_string()), + ("REQ".to_string(), "idp_project".to_string()), + ("TASK".to_string(), "idp_project".to_string()), + ("TICKET".to_string(), "ticket".to_string()), + ("TP".to_string(), "idp_test".to_string()), + ("TS".to_string(), "idp_test".to_string()), + ]) + } } diff --git a/backend/middlewares/flow/src/serv/flow_external_serv.rs b/backend/middlewares/flow/src/serv/flow_external_serv.rs index ee77fd4a..470a4c0d 100644 --- a/backend/middlewares/flow/src/serv/flow_external_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_external_serv.rs @@ -294,26 +294,6 @@ impl FlowExternalServ { } } - pub async fn do_find_embed_subrole_id(role_id: &str, ctx: &TardisContext, funs: &TardisFunsInst) -> TardisResult { - let iam_url = &funs.conf::().iam_url; - - let header = Self::headers(None, funs, ctx).await?; - let resp = funs - .web_client() - .get::>(&format!("{iam_url}/cc/role/get_embed_subrole_id?id={role_id}"), header) - .await? - .body - .ok_or_else(|| funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response"))?; - if resp.code != *"200" { - return Err(funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response")); - } - if let Some(data) = resp.data { - Ok(data) - } else { - Err(funs.err().internal_error("flow_external", "do_find_embed_subrole_id", "illegal response", "500-external-illegal-response")) - } - } - async fn get_external_url(tag: &str, ctx: &TardisContext, funs: &TardisFunsInst) -> TardisResult { let external_url = SpiKvClient::get_item(format!("{}:config:{}", flow_constants::DOMAIN_CODE, tag), None, funs, ctx) .await? diff --git a/frontend/sdks/invoke/src/clients/spi_search_client.rs b/frontend/sdks/invoke/src/clients/spi_search_client.rs index 627080aa..c2ccd841 100644 --- a/frontend/sdks/invoke/src/clients/spi_search_client.rs +++ b/frontend/sdks/invoke/src/clients/spi_search_client.rs @@ -1,8 +1,9 @@ use tardis::basic::dto::TardisContext; use tardis::basic::result::TardisResult; +use tardis::web::web_resp::{TardisPage, TardisResp}; use tardis::TardisFunsInst; -use crate::dto::search_item_dto::{SearchEventItemDeleteReq, SearchEventItemModifyReq, SearchItemAddReq, SearchItemModifyReq}; +use crate::dto::search_item_dto::{SearchEventItemDeleteReq, SearchEventItemModifyReq, SearchItemAddReq, SearchItemModifyReq, SearchItemSearchReq, SearchItemSearchResp}; use crate::invoke_enumeration::InvokeModuleKind; use super::base_spi_client::BaseSpiClient; @@ -74,6 +75,13 @@ impl SpiSearchClient { SpiKvClient::delete_item(&format!("__k_n__:{tag}:{key}"), funs, ctx).await?; Ok(()) } + + pub async fn search(search_req: &SearchItemSearchReq, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult>> { + let search_url = BaseSpiClient::module_url(InvokeModuleKind::Search, funs).await?; + let headers = BaseSpiClient::headers(None, funs, ctx).await?; + let resp = funs.web_client().put::>>(format!("{search_url}/ci/item/search"), search_req, headers.clone()).await?; + BaseSpiClient::package_resp(resp) + } } #[cfg(feature = "event")] impl EventCenterClient { diff --git a/frontend/sdks/invoke/src/dto/search_item_dto.rs b/frontend/sdks/invoke/src/dto/search_item_dto.rs index 9b13a1cd..06153dff 100644 --- a/frontend/sdks/invoke/src/dto/search_item_dto.rs +++ b/frontend/sdks/invoke/src/dto/search_item_dto.rs @@ -73,3 +73,173 @@ pub struct SearchEventItemDeleteReq { pub tag: String, pub key: String, } + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug)] +pub struct SearchItemSearchReq { + #[oai(validator(pattern = r"^[a-z0-9-_]+$"))] + pub tag: String, + // Search context for record permission filtering + pub ctx: SearchItemSearchCtxReq, + // Search conditions + pub query: SearchItemQueryReq, + // Advanced search + pub adv_query: Option>, + // Sort + // When the record set is very large, it will seriously affect the performance, it is not recommended to use. + pub sort: Option>, + pub page: SearchItemSearchPageReq, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug, Default, Clone)] +pub struct SearchItemSearchCtxReq { + pub accounts: Option>, + pub apps: Option>, + pub tenants: Option>, + pub roles: Option>, + pub groups: Option>, + pub cond_by_or: Option, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug, Default, Clone)] +pub struct SearchItemQueryReq { + pub in_q_content: Option, + // Fuzzy search content + pub q: Option, + // Fuzzy search scope + pub q_scope: Option, + pub kinds: Option>, + // Match keys, supports prefix match + pub keys: Option>, + #[oai(validator(min_length = "2"))] + // Match owners, supports prefix match + pub owners: Option>, + // Match own_path, supports prefix match + pub own_paths: Option>, + pub rlike_own_paths: Option>, + pub create_time_start: Option>, + pub create_time_end: Option>, + pub update_time_start: Option>, + pub update_time_end: Option>, + // Extended filtering conditions + pub ext: Option>, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug, Clone)] +pub struct SearchItemSearchPageReq { + pub number: u32, + pub size: u16, + // Get the total number of matching records. + // When the record set is very large, it will seriously affect the performance. It is not recommended to open it. + pub fetch_total: bool, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug, Clone)] +pub struct SearchItemSearchSortReq { + #[oai(validator(min_length = "2"))] + pub field: String, + pub order: SearchItemSearchSortKind, +} + +#[derive(poem_openapi::Enum, Serialize, Deserialize, Debug, Clone)] +pub enum SearchItemSearchSortKind { + #[oai(rename = "asc")] + Asc, + #[oai(rename = "desc")] + Desc, +} + +#[derive(poem_openapi::Enum, Serialize, Deserialize, Debug, Clone)] +pub enum SearchItemSearchQScopeKind { + #[oai(rename = "title")] + Title, + #[oai(rename = "content")] + Content, + #[oai(rename = "title_content")] + TitleContent, +} + +/// Basic query condition object +/// +/// 基础的查询条件对象 +#[derive(Serialize, Deserialize, Debug, Clone, poem_openapi::Object)] +pub struct BasicQueryCondInfo { + /// Query field + #[oai(validator(min_length = "1"))] + pub field: String, + /// Query operator + pub op: BasicQueryOpKind, + /// Query value + pub value: Value, +} + +/// Basic query operator +/// +/// 基础查询操作符 +#[derive(Clone, Debug, PartialEq, Eq, Deserialize, Serialize, poem_openapi::Enum)] +pub enum BasicQueryOpKind { + #[oai(rename = "=")] + Eq, + #[oai(rename = "!=")] + Ne, + #[oai(rename = ">")] + Gt, + #[oai(rename = ">=")] + Ge, + #[oai(rename = "<")] + Lt, + #[oai(rename = "<=")] + Le, + #[oai(rename = "like")] + Like, + #[oai(rename = "not_like")] + NotLike, + #[oai(rename = "l_like")] + LLike, + #[oai(rename = "not_l_like")] + NotLLike, + #[oai(rename = "r_like")] + RLike, + #[oai(rename = "not_r_like")] + NotRLike, + #[oai(rename = "in")] + In, + #[oai(rename = "not_in")] + NotIn, + #[oai(rename = "is_null")] + IsNull, + #[oai(rename = "is_not_null")] + IsNotNull, + #[oai(rename = "is_null_or_empty")] + IsNullOrEmpty, + #[oai(rename = "length")] + Len, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug, Default, Clone)] +pub struct AdvSearchItemQueryReq { + pub group_by_or: Option, + pub ext_by_or: Option, + // Extended filtering conditions + pub ext: Option>, +} + +#[derive(poem_openapi::Object, Serialize, Deserialize, Debug)] +pub struct SearchItemSearchResp { + #[oai(validator(min_length = "2"))] + pub kind: String, + #[oai(validator(min_length = "2"))] + pub key: String, + #[oai(validator(min_length = "2"))] + pub title: String, + #[oai(validator(min_length = "2"))] + pub content: String, + #[oai(validator(min_length = "2"))] + pub owner: String, + #[oai(validator(min_length = "2"))] + pub own_paths: String, + pub create_time: DateTime, + pub update_time: DateTime, + pub ext: Value, + pub rank_title: f32, + pub rank_content: f32, +} \ No newline at end of file From 871c36b3ddd4d8b5e85394885b1021fbe68405e9 Mon Sep 17 00:00:00 2001 From: ZzIsGod1019 <1498852723@qq.com> Date: Sun, 15 Dec 2024 18:15:05 +0800 Subject: [PATCH 11/11] flow: fix bug (auto skip state failed) --- .../flow/src/api/cc/flow_cc_inst_api.rs | 12 +- .../flow/src/api/ci/flow_ci_inst_api.rs | 12 +- .../middlewares/flow/src/dto/flow_inst_dto.rs | 8 +- .../flow/src/dto/flow_state_dto.rs | 2 +- .../flow/src/serv/clients/search_client.rs | 39 +- .../flow/src/serv/flow_inst_serv.rs | 483 ++++++++++++------ .../flow/src/serv/flow_model_serv.rs | 36 +- 7 files changed, 398 insertions(+), 194 deletions(-) diff --git a/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs b/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs index 1a253ffa..27b09f22 100644 --- a/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs +++ b/backend/middlewares/flow/src/api/cc/flow_cc_inst_api.rs @@ -213,10 +213,12 @@ impl FlowCcInstApi { ctx: TardisContextExtractor, _request: &Request, ) -> TardisApiResult { - let funs = flow_constants::get_tardis_inst(); + let mut funs = flow_constants::get_tardis_inst(); let vars = HashMap::from([("current_assigned".to_string(), Value::String(modify_req.0.current_assigned))]); let inst = FlowInstServ::get(&flow_inst_id.0, &funs, &ctx.0).await?; - FlowInstServ::modify_current_vars(&inst, &vars, loop_check_helper::InstancesTransition::default(), &ctx.0).await?; + funs.begin().await?; + FlowInstServ::modify_current_vars(&inst, &vars, loop_check_helper::InstancesTransition::default(), &funs, &ctx.0).await?; + funs.commit().await?; ctx.0.execute_task().await?; TardisResp::ok(Void {}) } @@ -232,9 +234,11 @@ impl FlowCcInstApi { ctx: TardisContextExtractor, _request: &Request, ) -> TardisApiResult { - let funs = flow_constants::get_tardis_inst(); + let mut funs = flow_constants::get_tardis_inst(); let inst = FlowInstServ::get(&flow_inst_id.0, &funs, &ctx.0).await?; - FlowInstServ::modify_current_vars(&inst, &modify_req.0.vars, loop_check_helper::InstancesTransition::default(), &ctx.0).await?; + funs.begin().await?; + FlowInstServ::modify_current_vars(&inst, &modify_req.0.vars, loop_check_helper::InstancesTransition::default(), &funs, &ctx.0).await?; + funs.commit().await?; ctx.0.execute_task().await?; TardisResp::ok(Void {}) } diff --git a/backend/middlewares/flow/src/api/ci/flow_ci_inst_api.rs b/backend/middlewares/flow/src/api/ci/flow_ci_inst_api.rs index b7e6ee80..f68b4afa 100644 --- a/backend/middlewares/flow/src/api/ci/flow_ci_inst_api.rs +++ b/backend/middlewares/flow/src/api/ci/flow_ci_inst_api.rs @@ -182,11 +182,13 @@ impl FlowCiInstApi { mut ctx: TardisContextExtractor, request: &Request, ) -> TardisApiResult { - let funs = flow_constants::get_tardis_inst(); + let mut funs = flow_constants::get_tardis_inst(); check_without_owner_and_unsafe_fill_ctx(request, &funs, &mut ctx.0)?; let vars = HashMap::from([("assigned_to".to_string(), Value::String(modify_req.0.current_assigned))]); let inst = FlowInstServ::get(&flow_inst_id.0, &funs, &ctx.0).await?; - FlowInstServ::modify_current_vars(&inst, &vars, loop_check_helper::InstancesTransition::default(), &ctx.0).await?; + funs.begin().await?; + FlowInstServ::modify_current_vars(&inst, &vars, loop_check_helper::InstancesTransition::default(), &funs, &ctx.0).await?; + funs.commit().await?; ctx.0.execute_task().await?; TardisResp::ok(Void {}) } @@ -202,10 +204,12 @@ impl FlowCiInstApi { mut ctx: TardisContextExtractor, request: &Request, ) -> TardisApiResult { - let funs = flow_constants::get_tardis_inst(); + let mut funs = flow_constants::get_tardis_inst(); check_without_owner_and_unsafe_fill_ctx(request, &funs, &mut ctx.0)?; let inst = FlowInstServ::get(&flow_inst_id.0, &funs, &ctx.0).await?; - FlowInstServ::modify_current_vars(&inst, &modify_req.0.vars, loop_check_helper::InstancesTransition::default(), &ctx.0).await?; + funs.begin().await?; + FlowInstServ::modify_current_vars(&inst, &modify_req.0.vars, loop_check_helper::InstancesTransition::default(), &funs, &ctx.0).await?; + funs.commit().await?; ctx.0.execute_task().await?; TardisResp::ok(Void {}) } diff --git a/backend/middlewares/flow/src/dto/flow_inst_dto.rs b/backend/middlewares/flow/src/dto/flow_inst_dto.rs index 78b7d3e8..1d1d25b6 100644 --- a/backend/middlewares/flow/src/dto/flow_inst_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_inst_dto.rs @@ -111,7 +111,7 @@ pub struct FlowInstSummaryResp { } /// 工作流详细信息 -#[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] +#[derive(Serialize, Deserialize, Debug, Clone, poem_openapi::Object)] pub struct FlowInstDetailResp { pub id: String, @@ -196,7 +196,7 @@ pub struct FlowInstDetailResp { } // 状态配置 -#[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] +#[derive(Serialize, Deserialize, Debug, Clone, poem_openapi::Object)] pub struct FLowInstStateConf { pub operators: HashMap, pub form_conf: Option, @@ -204,13 +204,13 @@ pub struct FLowInstStateConf { } // 状态录入配置 -#[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] +#[derive(Serialize, Deserialize, Debug, Clone, poem_openapi::Object)] pub struct FLowInstStateFormConf { pub form_vars_collect_conf: HashMap, } // 状态审批配置 -#[derive(Serialize, Deserialize, Debug, poem_openapi::Object)] +#[derive(Serialize, Deserialize, Debug, Clone, poem_openapi::Object)] pub struct FLowInstStateApprovalConf { pub approval_vars_collect_conf: Option>, pub form_vars_collect: HashMap, diff --git a/backend/middlewares/flow/src/dto/flow_state_dto.rs b/backend/middlewares/flow/src/dto/flow_state_dto.rs index c77a4b35..db8d839d 100644 --- a/backend/middlewares/flow/src/dto/flow_state_dto.rs +++ b/backend/middlewares/flow/src/dto/flow_state_dto.rs @@ -438,7 +438,7 @@ pub struct FLowStateIdAndName { } /// 可操作类型 -#[derive(Display, Serialize, Deserialize, Hash, Eq, PartialEq, Debug, poem_openapi::Enum, Clone)] +#[derive(Display, Clone, Serialize, Deserialize, Hash, Eq, PartialEq, Debug, poem_openapi::Enum)] pub enum FlowStateOperatorKind { /// 转办 Referral, diff --git a/backend/middlewares/flow/src/serv/clients/search_client.rs b/backend/middlewares/flow/src/serv/clients/search_client.rs index f86fc7fa..3e6b892c 100644 --- a/backend/middlewares/flow/src/serv/clients/search_client.rs +++ b/backend/middlewares/flow/src/serv/clients/search_client.rs @@ -6,7 +6,7 @@ use bios_sdk_invoke::{ event_client::{get_topic, EventCenterClient, SPI_RPC_TOPIC}, spi_search_client::SpiSearchClient, }, - dto::search_item_dto::{SearchItemAddReq, SearchItemModifyReq, SearchItemSearchReq, SearchItemSearchResp, SearchItemVisitKeysReq}, + dto::search_item_dto::{SearchItemAddReq, SearchItemModifyReq, SearchItemQueryReq, SearchItemSearchCtxReq, SearchItemSearchPageReq, SearchItemSearchReq, SearchItemSearchResp, SearchItemVisitKeysReq}, }; use itertools::Itertools; use serde_json::json; @@ -18,7 +18,7 @@ use crate::{ dto::{ flow_inst_dto::FlowInstFilterReq, flow_model_dto::{FlowModelDetailResp, FlowModelFilterReq, FlowModelRelTransitionExt}, - flow_model_version_dto::FlowModelVersionFilterReq, + flow_model_version_dto::FlowModelVersionFilterReq, flow_state_dto::FlowGuardConf, }, flow_constants, serv::{ @@ -257,6 +257,41 @@ impl FlowSearchClient { SpiSearchClient::search(search_req, funs, ctx).await } + pub async fn search_guard_account_num(guard_conf: &FlowGuardConf, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { + if guard_conf.guard_by_spec_account_ids.is_empty() + && guard_conf.guard_by_spec_org_ids.is_empty() + && guard_conf.guard_by_spec_role_ids.is_empty() + { + return Ok(Some(0)); + } + let mut search_ctx_req = SearchItemSearchCtxReq { + apps: Some(vec![rbum_scope_helper::get_path_item(RbumScopeLevelKind::L2.to_int(), &ctx.own_paths).unwrap_or_default()]), + ..Default::default() + }; + let mut query = SearchItemQueryReq::default(); + if !guard_conf.guard_by_spec_account_ids.is_empty() { + query.keys = Some(guard_conf.guard_by_spec_account_ids.clone().into_iter().map(|account_id| account_id.into()).collect_vec()); + } + if !guard_conf.guard_by_spec_org_ids.is_empty() { + search_ctx_req.groups = Some(guard_conf.guard_by_spec_org_ids.clone()); + } + if !guard_conf.guard_by_spec_role_ids.is_empty() { + search_ctx_req.roles = Some(guard_conf.guard_by_spec_role_ids.clone()); + } + Ok(SpiSearchClient::search(&SearchItemSearchReq { + tag: "iam_account".to_string(), + ctx: search_ctx_req, + query: SearchItemQueryReq { ..Default::default() }, + adv_query: None, + sort: None, + page:SearchItemSearchPageReq { + number: 1, + size: 1, + fetch_total: true, + }, + }, funs, ctx).await?.map(|result| result.total_size)) + } + pub fn get_tag_search_map() -> HashMap { HashMap::from([ ("CTS".to_string(), "idp_test".to_string()), diff --git a/backend/middlewares/flow/src/serv/flow_inst_serv.rs b/backend/middlewares/flow/src/serv/flow_inst_serv.rs index d0b214e4..5e53c92a 100644 --- a/backend/middlewares/flow/src/serv/flow_inst_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_inst_serv.rs @@ -13,6 +13,7 @@ use bios_basic::{ } }, }; +use bios_sdk_invoke::dto::search_item_dto::{SearchItemQueryReq, SearchItemSearchCtxReq, SearchItemSearchPageReq, SearchItemSearchReq}; use itertools::Itertools; use serde_json::json; use tardis::{ @@ -39,7 +40,7 @@ use crate::{ FlowInstFindNextTransitionsReq, FlowInstFindStateAndTransitionsReq, FlowInstFindStateAndTransitionsResp, FlowInstOperateReq, FlowInstQueryKind, FlowInstSearchPageReq, FlowInstSearchReq, FlowInstSearchSortReq, FlowInstStartReq, FlowInstSummaryResp, FlowInstSummaryResult, FlowInstTransferReq, FlowInstTransferResp, FlowInstTransitionInfo, FlowOperationContext, - }, flow_model_dto::{FlowModelDetailResp, FlowModelRelTransitionExt}, flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateCountersignKind, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowStatusMultiApprovalKind, FlowSysStateKind}, flow_transition_dto::FlowTransitionDetailResp, flow_var_dto::FillType + }, flow_model_dto::{FlowModelDetailResp, FlowModelFilterReq, FlowModelRelTransitionExt}, flow_model_version_dto::{FlowModelVersionDetailResp, FlowModelVersionFilterReq}, flow_state_dto::{FLowStateKindConf, FlowStateAggResp, FlowStateCountersignKind, FlowStateFilterReq, FlowStateKind, FlowStateOperatorKind, FlowStateRelModelExt, FlowStatusAutoStrategyKind, FlowStatusMultiApprovalKind, FlowSysStateKind}, flow_transition_dto::FlowTransitionDetailResp, flow_var_dto::FillType }, flow_constants, helper::loop_check_helper, @@ -78,7 +79,7 @@ impl FlowInstServ { } else { flow_model.init_state_id.clone() }; - let create_vars = Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await?; + let create_vars = Self::get_new_vars(&flow_model.tag, start_req.rel_business_obj_id.to_string(), funs, ctx).await?; let flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { id: Set(inst_id.clone()), code: Set(Some("".to_string())), @@ -89,6 +90,7 @@ impl FlowInstServ { current_state_id: Set(current_state_id.clone()), create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + current_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), create_ctx: Set(FlowOperationContext::from_ctx(ctx)), own_paths: Set(ctx.own_paths.clone()), @@ -125,7 +127,7 @@ impl FlowInstServ { { return Err(funs.err().internal_error("flow_inst_serv", "start", "The same instance exist", "500-flow-inst-exist")); } - let create_vars = Self::get_new_vars(&flow_model.tag, vec![start_req.rel_business_obj_id.to_string()], &ctx.own_paths, funs, ctx).await?; + let create_vars = Self::get_new_vars(&flow_model.tag, start_req.rel_business_obj_id.to_string(), funs, ctx).await?; let flow_inst: flow_inst::ActiveModel = flow_inst::ActiveModel { id: Set(inst_id.clone()), code: Set(Some(Self::gen_inst_code(funs).await?)), @@ -136,6 +138,8 @@ impl FlowInstServ { current_state_id: Set(flow_model.init_state_id.clone()), create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + current_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + create_ctx: Set(FlowOperationContext::from_ctx(ctx)), own_paths: Set(ctx.own_paths.clone()), @@ -144,7 +148,7 @@ impl FlowInstServ { }; funs.db().insert_one(flow_inst, ctx).await?; Self::modify_inst_artifacts(&inst_id, &FlowInstArtifactsModifyReq { - form_state_map: start_req.create_vars.clone(), + curr_vars: start_req.create_vars.clone(), ..Default::default() }, funs, ctx).await?; FlowSearchClient::modify_business_obj_search(&start_req.rel_business_obj_id, &flow_model.tag, funs, ctx).await?; @@ -183,6 +187,8 @@ impl FlowInstServ { FlowModelServ::get_model_id_by_own_paths_and_rel_template_id(&batch_bind_req.tag, None, funs, ctx).await? }; + let create_vars = Self::get_new_vars(&flow_model.tag, rel_business_obj.rel_business_obj_id.clone().unwrap_or_default(), funs, ctx).await?; + let current_state_id = FlowStateServ::match_state_id_by_name(&flow_model.current_version_id, &rel_business_obj.current_state_name.clone().unwrap_or_default(), funs, ctx).await?; let mut inst_id = Self::get_inst_ids_by_rel_business_obj_id(vec![rel_business_obj.rel_business_obj_id.clone().unwrap_or_default()], Some(true), funs, ctx).await?.pop(); if inst_id.is_none() { @@ -194,6 +200,9 @@ impl FlowInstServ { current_state_id: Set(current_state_id), + create_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + current_vars: Set(Some(TardisFuns::json.obj_to_json(&create_vars).unwrap_or(json!("")))), + create_ctx: Set(FlowOperationContext::from_ctx(¤t_ctx)), own_paths: Set(rel_business_obj.own_paths.clone().unwrap_or_default()), @@ -350,6 +359,7 @@ impl FlowInstServ { funs.db().update_one(flow_inst, ctx).await?; let flow_inst_detail = Self::get(flow_inst_id, funs, ctx).await?; + Self::add_finish_log(&flow_inst_detail, funs, ctx).await?; if !flow_inst_detail.main { FlowSearchClient::modify_business_obj_search(&flow_inst_detail.rel_business_obj_id, &flow_inst_detail.tag, funs, ctx).await?; } @@ -509,51 +519,62 @@ impl FlowInstServ { .and_where(Expr::col((flow_inst::Entity, flow_inst::Column::OwnPaths)).like(format!("{}%", ctx.own_paths))); let flow_insts = funs.db().find_dtos::(&query).await?; - Ok(flow_insts - .into_iter() - .map(|inst| { - let current_state_kind_conf = inst - .current_state_kind_conf - .clone() - .map(|current_state_kind_conf| TardisFuns::json.json_to_obj::(current_state_kind_conf).unwrap_or_default()); - let artifacts = inst.artifacts.clone().map(|artifacts| TardisFuns::json.json_to_obj::(artifacts).unwrap_or_default()); - FlowInstDetailResp { - id: inst.id, - code: inst.code, - rel_flow_version_id: inst.rel_flow_version_id, - rel_flow_model_name: inst.rel_flow_model_name, - tag: inst.tag, - main: inst.main, - create_vars: inst.create_vars.map(|create_vars| TardisFuns::json.json_to_obj(create_vars).unwrap()), - create_ctx: inst.create_ctx, - create_time: inst.create_time, - finish_ctx: inst.finish_ctx, - finish_time: inst.finish_time, - finish_abort: inst.finish_abort, - output_message: inst.output_message, - own_paths: inst.own_paths, - transitions: inst.transitions.map(|transitions| TardisFuns::json.json_to_obj(transitions).unwrap()), - artifacts: inst.artifacts.map(|artifacts| TardisFuns::json.json_to_obj(artifacts).unwrap()), - comments: inst.comments.map(|comments| TardisFuns::json.json_to_obj(comments).unwrap()), - rel_transition: inst.rel_transition.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), - current_state_id: inst.current_state_id.clone(), - current_state_name: inst.current_state_name, - current_state_color: inst.current_state_color, - current_state_sys_kind: inst.current_state_sys_kind, - current_state_kind: inst.current_state_kind.clone(), - current_state_ext: inst.current_state_ext.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), - current_state_conf: Self::get_state_conf( - &inst.current_state_id, - &inst.current_state_kind.unwrap_or_default(), - current_state_kind_conf, - artifacts, - ctx, - ), - current_vars: inst.current_vars.map(|current_vars| TardisFuns::json.json_to_obj(current_vars).unwrap()), - rel_business_obj_id: inst.rel_business_obj_id, - } - }) - .collect_vec()) + let result = flow_insts + .into_iter() + .map(|inst| { + let current_state_kind_conf = inst + .current_state_kind_conf + .clone() + .map(|current_state_kind_conf| TardisFuns::json.json_to_obj::(current_state_kind_conf).unwrap_or_default()); + let artifacts = inst.artifacts.clone().map(|artifacts| TardisFuns::json.json_to_obj::(artifacts).unwrap_or_default()); + FlowInstDetailResp { + id: inst.id, + code: inst.code, + rel_flow_version_id: inst.rel_flow_version_id, + rel_flow_model_name: inst.rel_flow_model_name, + tag: inst.tag, + main: inst.main, + create_vars: inst.create_vars.map(|create_vars| TardisFuns::json.json_to_obj(create_vars).unwrap()), + create_ctx: inst.create_ctx, + create_time: inst.create_time, + finish_ctx: inst.finish_ctx, + finish_time: inst.finish_time, + finish_abort: inst.finish_abort, + output_message: inst.output_message, + own_paths: inst.own_paths, + transitions: inst.transitions.map(|transitions| TardisFuns::json.json_to_obj(transitions).unwrap()), + artifacts: artifacts.clone(), + comments: inst.comments.map(|comments| TardisFuns::json.json_to_obj(comments).unwrap()), + rel_transition: inst.rel_transition.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), + current_state_id: inst.current_state_id.clone(), + current_state_name: inst.current_state_name, + current_state_color: inst.current_state_color, + current_state_sys_kind: inst.current_state_sys_kind, + current_state_kind: inst.current_state_kind.clone(), + current_state_ext: inst.current_state_ext.map(|ext| TardisFuns::json.str_to_obj::(&ext).unwrap_or_default()), + current_state_conf: Self::get_state_conf( + &inst.current_state_id, + &inst.current_state_kind.unwrap_or_default(), + current_state_kind_conf, + artifacts, + ctx, + ), + current_vars: inst.current_vars.map(|current_vars| TardisFuns::json.json_to_obj(current_vars).unwrap()), + rel_business_obj_id: inst.rel_business_obj_id, + } + }) + .collect_vec(); + Ok(result.into_iter().map(|mut inst_detail| { + let mut artifacts = inst_detail.artifacts.clone(); + if let Some(artifacts) = artifacts.as_mut() { + let mut curr_vars= artifacts.curr_vars.clone().unwrap_or_default(); + curr_vars.extend(Self::get_modify_vars(&inst_detail)); + + artifacts.curr_vars = Some(curr_vars); + } + inst_detail.artifacts = artifacts; + inst_detail + }).collect_vec()) } pub async fn paginate( @@ -945,12 +966,16 @@ impl FlowInstServ { funs.db().update_one(flow_inst, ctx).await?; + let curr_inst = Self::get(&flow_inst_detail.id, funs, ctx).await?; + if !curr_inst.main { + Self::add_finish_log(&curr_inst, funs, ctx).await?; + } if next_flow_state.sys_state == FlowSysStateKind::Finish && !flow_inst_detail.main { FlowSearchClient::modify_business_obj_search(&flow_inst_detail.rel_business_obj_id, &flow_inst_detail.tag, funs, ctx).await?; } - Self::when_leave_state(flow_inst_detail, &prev_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; - Self::when_enter_state(flow_inst_detail, &next_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; + Self::when_leave_state(&curr_inst, &prev_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; + Self::when_enter_state(&curr_inst, &next_flow_state.id, &flow_model_version.rel_model_id, funs, ctx).await?; Self::do_request_webhook( from_transition_id.and_then(|id| version_transition.iter().find(|model_transition| model_transition.id == id)), @@ -959,11 +984,11 @@ impl FlowInstServ { .await?; // notify change state - if flow_inst_detail.main { + if curr_inst.main { FlowExternalServ::do_notify_changes( - &flow_inst_detail.tag, - &flow_inst_detail.id, - &flow_inst_detail.rel_business_obj_id, + &curr_inst.tag, + &curr_inst.id, + &curr_inst.rel_business_obj_id, next_flow_state.name.clone(), next_flow_state.sys_state.clone(), prev_flow_state.name.clone(), @@ -977,7 +1002,7 @@ impl FlowInstServ { .await?; } - Self::gen_transfer_resp(flow_inst_detail, &prev_flow_state.id, ctx, funs).await + Self::gen_transfer_resp(&curr_inst, &prev_flow_state.id, ctx, funs).await } async fn gen_transfer_resp(flow_inst_detail: &FlowInstDetailResp, prev_flow_state_id: &str, ctx: &TardisContext, funs: &TardisFunsInst) -> TardisResult { @@ -1266,10 +1291,9 @@ impl FlowInstServ { flow_inst_detail: &FlowInstDetailResp, current_vars: &HashMap, modified_instance_transations: loop_check_helper::InstancesTransition, + funs: &TardisFunsInst, ctx: &TardisContext, ) -> TardisResult<()> { - let mut funs = flow_constants::get_tardis_inst(); - funs.begin().await?; let mut new_vars: HashMap = HashMap::new(); if let Some(old_current_vars) = &flow_inst_detail.current_vars { new_vars.extend(old_current_vars.clone()); @@ -1281,33 +1305,44 @@ impl FlowInstServ { ..Default::default() }; funs.db().update_one(flow_inst, ctx).await.unwrap(); - funs.commit().await?; - - let funs = flow_constants::get_tardis_inst(); - - FlowEventServ::do_front_change(flow_inst_detail, modified_instance_transations, ctx, &funs).await?; + let curr_inst = Self::get(&flow_inst_detail.id, funs, ctx).await?; + FlowEventServ::do_front_change(&curr_inst, modified_instance_transations, ctx, funs).await?; Ok(()) } - async fn get_new_vars(tag: &str, rel_business_obj_ids: Vec, own_paths: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { - let new_vars = FlowExternalServ::do_query_field(tag, rel_business_obj_ids, own_paths, ctx, funs) - .await? - .objs - .into_iter() - .next() - .unwrap_or_default(); + async fn get_new_vars(tag: &str, rel_business_obj_id: String, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { + let tag_search_map = FlowSearchClient::get_tag_search_map(); + let new_vars = if let Some(mut search_resp) = FlowSearchClient::search(&SearchItemSearchReq { + tag: tag_search_map.get(tag).ok_or_else(|| funs.err().not_found("flow_inst", "get_new_vars", "illegal response", "404-flow-tag-not-found"))?.clone(), + ctx: SearchItemSearchCtxReq::default(), + query: SearchItemQueryReq { + keys: Some(vec![rel_business_obj_id.into()]), + ..Default::default() + }, + adv_query: None, + sort: None, + page: SearchItemSearchPageReq { + number: 1, + size: 1, + fetch_total: false, + }}, funs, ctx).await? { + search_resp.records.pop().map(|item| item.ext) + } else { + None + }.unwrap_or_default(); Ok(TardisFuns::json.json_to_obj(new_vars).unwrap_or_default()) } pub async fn find_var_by_inst_id(flow_inst: &FlowInstDetailResp, key: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult> { let mut current_vars = flow_inst.current_vars.clone(); if current_vars.is_none() || !current_vars.clone().unwrap_or_default().contains_key(key) { - let new_vars = Self::get_new_vars(&flow_inst.tag, vec![flow_inst.rel_business_obj_id.clone()], &flow_inst.own_paths, funs, ctx).await?; + let new_vars = Self::get_new_vars(&flow_inst.tag, flow_inst.rel_business_obj_id.clone(), funs, ctx).await?; Self::modify_current_vars( flow_inst, &new_vars, loop_check_helper::InstancesTransition::default(), + funs, ctx, ) .await?; @@ -1353,6 +1388,7 @@ impl FlowInstServ { update_statement.table(flow_inst::Entity); update_statement.value(flow_inst::Column::RelFlowVersionId, modify_version_id); update_statement.and_where(Expr::col((flow_inst::Entity, flow_inst::Column::Tag)).eq(tag)); + update_statement.and_where(Expr::col((flow_inst::Entity, flow_inst::Column::Main)).eq(true)); update_statement.and_where(Expr::col((flow_inst::Entity, flow_inst::Column::OwnPaths)).eq(ctx.own_paths.as_str())); funs.db().execute(&update_statement).await?; @@ -1363,6 +1399,7 @@ impl FlowInstServ { pub async fn unsafe_modify_state(tag: &str, modify_model_states: Vec, state_id: &str, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { let insts = Self::find_details( &FlowInstFilterReq { + main: Some(true), tag: Some(tag.to_string()), ..Default::default() }, @@ -1523,16 +1560,18 @@ impl FlowInstServ { .split(',') .collect_vec() .into_iter() - .map(|str| guard_custom_conf.guard_by_spec_account_ids.push(str.to_string())) + .map(|str| { + if !str.is_empty() { + guard_custom_conf.guard_by_spec_account_ids.push(str.to_string()); + } + }) .collect::>(); } modify_req.guard_conf = Some(guard_custom_conf); Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; // 当操作人为空时的逻辑 let curr_guard_conf = Self::get(&flow_inst_detail.id, funs, ctx).await?.artifacts.unwrap_or_default().guard_conf; - if curr_guard_conf.guard_by_spec_account_ids.is_empty() - && curr_guard_conf.guard_by_spec_org_ids.is_empty() - && curr_guard_conf.guard_by_spec_role_ids.is_empty() + if FlowSearchClient::search_guard_account_num(&curr_guard_conf, funs, ctx).await?.unwrap_or_default().is_empty() && form_conf.auto_transfer_when_empty_kind.is_some() { match form_conf.auto_transfer_when_empty_kind.unwrap_or_default() { FlowStatusAutoStrategyKind::Autoskip => { @@ -1558,7 +1597,7 @@ impl FlowInstServ { Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; }, FlowStatusAutoStrategyKind::TransferState => { - // @TODO 当前版本不支持 + // 当前版本不支持 }, } } @@ -1584,22 +1623,23 @@ impl FlowInstServ { .split(',') .collect_vec() .into_iter() - .map(|str| guard_custom_conf.guard_by_spec_account_ids.push(str.to_string())) + .map(|str| {if !str.is_empty() { + guard_custom_conf.guard_by_spec_account_ids.push(str.to_string()); + }} + ) .collect::>(); } modify_req.guard_conf = Some(guard_custom_conf); // 若会签,则需要统计审批人数 if approval_conf.multi_approval_kind == FlowStatusMultiApprovalKind::Countersign { - // @TODO 此处缺少获取人数的逻辑 - let total = 10; - modify_req.curr_approval_total = Some(total); + let countersign_total = FlowSearchClient::search_guard_account_num(&approval_conf.countersign_conf.guard_custom_conf.clone().unwrap_or_default(), funs, ctx).await?.unwrap_or_default(); + modify_req.curr_approval_total = Some(countersign_total as usize); } Self::modify_inst_artifacts(&flow_inst_detail.id, &modify_req, funs, ctx).await?; + let curr_inst = Self::get(&flow_inst_detail.id, funs, ctx).await?; // 当操作人为空时的逻辑 - let curr_guard_conf = Self::get(&flow_inst_detail.id, funs, ctx).await?.artifacts.unwrap_or_default().guard_conf; - if curr_guard_conf.guard_by_spec_account_ids.is_empty() - && curr_guard_conf.guard_by_spec_org_ids.is_empty() - && curr_guard_conf.guard_by_spec_role_ids.is_empty() + let curr_guard_conf = curr_inst.artifacts.unwrap_or_default().guard_conf; + if FlowSearchClient::search_guard_account_num(&curr_guard_conf, funs, ctx).await?.unwrap_or_default().is_empty() && approval_conf.auto_transfer_when_empty_kind.is_some() { match approval_conf.auto_transfer_when_empty_kind.unwrap_or_default() { FlowStatusAutoStrategyKind::Autoskip => { @@ -1633,23 +1673,7 @@ impl FlowInstServ { FlowStateKind::Branch => {} FlowStateKind::Finish => match rel_transition.as_str() { "__EDIT__" => { - let transition = FlowTransitionServ::find_transitions(&flow_inst_detail.rel_flow_version_id, None, funs, ctx).await?; - let mut state_ids = vec![]; - let mut vars_collect = HashMap::new(); - let artifacts = flow_inst_detail.artifacts.clone().unwrap_or_default(); - for tran in flow_inst_detail.transitions.clone().unwrap_or_default() { - if let Some(tran) = transition.iter().find(|version_tran| tran.id == version_tran.id) { - let current_state_id = tran.to_flow_state_id.clone(); - if !state_ids.contains(¤t_state_id) { - state_ids.push(current_state_id.clone()); - if let Some(form_state_vars) = artifacts.form_state_map.get(¤t_state_id) { - for (key, value) in form_state_vars { - *vars_collect.entry(key.clone()).or_insert(json!({})) = value.clone(); - } - } - } - } - } + let vars_collect = Self::get_modify_vars(flow_inst_detail); let params = vars_collect .into_iter() .map(|(key, value)| FlowExternalParams { @@ -1778,10 +1802,10 @@ impl FlowInstServ { current_account_ids.push(add_approval_account_id.clone()); } if let Some(curr_approval_total) = modify_artifacts.curr_approval_total { - if let Some(approval_total) = inst_artifacts.approval_total.as_mut() { - let old_approval_total = approval_total.entry(inst.current_state_id.clone()).or_default(); - *old_approval_total = curr_approval_total; - } + let mut approval_total = inst_artifacts.approval_total.clone().unwrap_or_default(); + let approval_state_total = approval_total.entry(inst.current_state_id.clone()).or_default(); + *approval_state_total = curr_approval_total; + inst_artifacts.approval_total = Some(approval_total); } if let Some(form_state_vars) = modify_artifacts.form_state_map.clone() { inst_artifacts.form_state_map.insert(inst.current_state_id.clone(), form_state_vars.clone()); @@ -1798,6 +1822,9 @@ impl FlowInstServ { if let Some(prev_non_auto_account_id) = &modify_artifacts.prev_non_auto_account_id { inst_artifacts.prev_non_auto_account_id = Some(prev_non_auto_account_id.clone()); } + if let Some(curr_vars) = &modify_artifacts.curr_vars { + inst_artifacts.curr_vars = Some(curr_vars.clone()); + } let flow_inst = flow_inst::ActiveModel { id: Set(inst.id.clone()), artifacts: Set(Some(inst_artifacts)), @@ -1881,6 +1908,24 @@ impl FlowInstServ { } pub async fn operate(inst: &FlowInstDetailResp, operate_req: &FlowInstOperateReq, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { + let current_state = FlowStateServ::get_item( + &inst.current_state_id, + &FlowStateFilterReq { + basic: RbumBasicFilterReq { + with_sub_own_paths: true, + own_paths: Some("".to_string()), + ..Default::default() + }, + ..Default::default() + }, + funs, + ctx, + ).await?; + Self::add_operate_log(operate_req, inst, if current_state.state_kind == FlowStateKind::Approval {LogParamOp::Approval} else {LogParamOp::Form}, funs, ctx).await?; + let mut modify_artifacts = FlowInstArtifactsModifyReq::default(); + if let Some(all_vars) = &operate_req.all_vars { + modify_artifacts.curr_vars = Some(all_vars.clone()); + } match operate_req.operate { // 转办 FlowStateOperatorKind::Referral => { @@ -1958,6 +2003,7 @@ impl FlowInstServ { .await?; if Self::check_approval_cond(inst, FlowApprovalResultKind::Pass, funs, ctx).await? { if let Some(next_transition) = FlowInstServ::find_next_transitions(inst, &FlowInstFindNextTransitionsReq { vars: None }, funs, ctx).await?.pop() { + Self::add_operate_log(operate_req, inst, if current_state.state_kind == FlowStateKind::Approval {LogParamOp::ApprovalTransfer} else {LogParamOp::FormTransfer}, funs, ctx).await?; Self::transfer( inst, &FlowInstTransferReq { @@ -1988,6 +2034,7 @@ impl FlowInstServ { ) .await?; if Self::check_approval_cond(inst, FlowApprovalResultKind::Overrule, funs, ctx).await? { + Self::add_operate_log(operate_req, inst, if current_state.state_kind == FlowStateKind::Approval {LogParamOp::ApprovalTransfer} else {LogParamOp::FormTransfer}, funs, ctx).await?; Self::abort(&inst.id, &FlowInstAbortReq { message: "".to_string() }, funs, ctx).await?; } } @@ -2018,6 +2065,14 @@ impl FlowInstServ { .cloned() .unwrap_or_default(); if let Some(current_state_kind_conf) = current_state_kind_conf { + // 或签直接通过 + if current_state_kind_conf.multi_approval_kind == FlowStatusMultiApprovalKind::Orsign { + return Ok(true); + } + // 会签但是人数为空,直接通过 + if current_state_kind_conf.multi_approval_kind == FlowStatusMultiApprovalKind::Countersign && approval_total == 0 { + return Ok(true); + } let countersign_conf = current_state_kind_conf.countersign_conf; // 指定人通过,则通过 if kind == FlowApprovalResultKind::Pass @@ -2136,6 +2191,9 @@ impl FlowInstServ { }; funs.db().update_one(flow_inst, ctx).await?; + + let from_state_id = flow_inst_detail.current_state_id.clone(); + let curr_inst = Self::get(&flow_inst_detail.id, funs, ctx).await?; // 删除目标节点的旧记录 Self::modify_inst_artifacts( &flow_inst_detail.id, @@ -2149,8 +2207,8 @@ impl FlowInstServ { ) .await?; - Self::when_leave_state(flow_inst_detail, &flow_inst_detail.current_state_id, &flow_model_version.rel_model_id, funs, ctx).await?; - Self::when_enter_state(flow_inst_detail, target_state_id, &flow_model_version.rel_model_id, funs, ctx).await?; + Self::when_leave_state(&curr_inst, &from_state_id, &flow_model_version.rel_model_id, funs, ctx).await?; + Self::when_enter_state(&curr_inst, target_state_id, &flow_model_version.rel_model_id, funs, ctx).await?; Ok(()) } @@ -2471,20 +2529,17 @@ impl FlowInstServ { }; let mut log_ext = LogParamExt { scene_kind: Some(vec!["approval_flow".to_string()]), - sys_op: Some(ctx.owner.clone()), new_log: Some(true), - project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L2.to_int(), &ctx.own_paths), ..Default::default() }; let mut log_content = LogParamContent { subject: Some(FlowLogClient::get_flow_kind_text(&start_req.tag)), name: Some(create_vars.get("name").map_or("".to_string(), |val| val.as_str().unwrap_or("").to_string())), sub_id: Some(start_req.rel_business_obj_id.clone()), - sub_kind: None, // @TODO 缺少类型 + sub_kind: Some(FlowLogClient::get_junp_kind(&start_req.tag)), operand: Some(operand), - operand_id: Some(flow_inst_detail.id.clone()), operand_name: Some(flow_inst_detail.code.clone()), - operand_kind: None, ..Default::default() }; if start_req.create_vars.is_none() { @@ -2511,61 +2566,151 @@ impl FlowInstServ { } // 添加审批流操作日志 - // async fn add_operate_log(operate_req: &FlowInstOperateReq, flow_inst_detail: &FlowInstDetailResp, flow_model: &FlowModelDetailResp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { - // let current_state = FlowStateServ::get_item( - // &flow_inst_detail.current_state_id, - // &FlowStateFilterReq { - // basic: RbumBasicFilterReq { - // with_sub_own_paths: true, - // own_paths: Some("".to_string()), - // ..Default::default() - // }, - // ..Default::default() - // }, - // funs, - // ctx, - // ).await?; - // let subject_text = match { - - // }; - // let mut log_ext = LogParamExt { - // scene_kind: Some(vec!["approval_flow".to_string()]), - // sys_op: Some(ctx.owner.clone()), - // new_log: Some(true), - // project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), - // ..Default::default() - // }; - // let mut log_content = LogParamContent { - // subject: Some(FlowLogClient::get_flow_kind_text(&start_req.tag)), - // name: Some(create_vars.get("name").map_or("".to_string(), |val| TardisFuns::json.json_to_string(val.clone()).unwrap_or_default())), - // sub_id: Some(start_req.rel_business_obj_id.clone()), - // sub_kind: None, // @TODO 缺少类型 - // operand: Some(operand), - // operand_id: Some(flow_inst_detail.id.clone()), - // operand_name: Some(flow_inst_detail.code.clone()), - // operand_kind: None, - // ..Default::default() - // }; - // if operate_req.vars.is_none() { - // log_ext.include_detail = Some(false); - // } else { - // log_content.old_content = Some(create_vars.get("content").map_or("".to_string(), |val| TardisFuns::json.json_to_string(val.clone()).unwrap_or_default())); - // log_content.new_content = operate_req.vars.clone().unwrap_or_default().get("content").map(|content| TardisFuns::json.json_to_string(content.clone()).unwrap_or_default()); - // log_content.detail = None; // @TODO 暂未实现 - // log_ext.include_detail = Some(true); - // } - // FlowLogClient::add_ctx_task( - // LogParamTag::ApprovalFlow, - // Some(flow_inst_detail.id.clone()), - // log_content, - // Some(TardisFuns::json.obj_to_json(&log_ext).expect("ext not a valid json value")), - // Some("dynamic_log_approval_flow".to_string()), - // Some(LogParamOp::Start.into()), - // rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), - // ctx, - // false, - // ) - // .await?; - // Ok(()) - // } + async fn add_operate_log(operate_req: &FlowInstOperateReq, flow_inst_detail: &FlowInstDetailResp, op_kind: LogParamOp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { + let current_state = FlowStateServ::get_item( + &flow_inst_detail.current_state_id, + &FlowStateFilterReq { + basic: RbumBasicFilterReq { + with_sub_own_paths: true, + own_paths: Some("".to_string()), + ..Default::default() + }, + ..Default::default() + }, + funs, + ctx, + ).await?; + let subject_text = match current_state.state_kind { + FlowStateKind::Approval => "审批节点".to_string(), + FlowStateKind::Form => "录入节点".to_string(), + _ => "".to_string(), + }; + let mut log_ext = LogParamExt { + scene_kind: Some(vec!["approval_flow".to_string()]), + new_log: Some(true), + project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L2.to_int(), &ctx.own_paths), + ..Default::default() + }; + let mut log_content = LogParamContent { + subject: Some(subject_text), + name: Some(current_state.name), + sub_id: Some(flow_inst_detail.rel_business_obj_id.clone()), + sub_kind: Some(FlowLogClient::get_junp_kind(&flow_inst_detail.tag)), + flow_message: operate_req.output_message.clone(), + flow_result: Some(operate_req.operate.to_string().to_uppercase()), + ..Default::default() + }; + if operate_req.vars.is_none() { + log_ext.include_detail = Some(false); + } else { + log_content.old_content = Some(flow_inst_detail.create_vars.clone().unwrap_or_default().get("content").map_or("".to_string(), |val| TardisFuns::json.json_to_string(val.clone()).unwrap_or_default())); + log_content.new_content = operate_req.vars.clone().unwrap_or_default().get("content").map(|content| TardisFuns::json.json_to_string(content.clone()).unwrap_or_default()); + log_content.detail = None; // @TODO 缺少翻译逻辑 + log_ext.include_detail = Some(true); + } + if operate_req.output_message.is_some() { + log_ext.include_detail = Some(true); + } + FlowLogClient::add_ctx_task( + LogParamTag::ApprovalFlow, + Some(flow_inst_detail.id.clone()), + log_content, + Some(TardisFuns::json.obj_to_json(&log_ext).expect("ext not a valid json value")), + Some("dynamic_log_approval_flow".to_string()), + Some(op_kind.into()), + rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + ctx, + false, + ) + .await?; + Ok(()) + } + + async fn add_finish_log(flow_inst_detail: &FlowInstDetailResp, funs: &TardisFunsInst, ctx: &TardisContext) -> TardisResult<()> { + let flow_model_version = FlowModelVersionServ::get_item( + &flow_inst_detail.rel_flow_version_id, + &FlowModelVersionFilterReq { + basic: RbumBasicFilterReq { + with_sub_own_paths: true, + own_paths: Some("".to_string()), + ..Default::default() + }, + ..Default::default() + }, + funs, + ctx, + ) + .await?; + let flow_model = FlowModelServ::get_item( + &flow_model_version.rel_model_id, + &FlowModelFilterReq { + basic: RbumBasicFilterReq { + with_sub_own_paths: true, + own_paths: Some("".to_string()), + ..Default::default() + }, + ..Default::default() + }, + funs, + ctx, + ) + .await?; + let rel_transition = flow_model.rel_transition().unwrap_or_default(); + let subject_text = match rel_transition.id.as_str() { + "__EDIT__" => { + "编辑审批".to_string() + }, + "__DELETE__" => { + "删除审批".to_string() + } + _ => { + format!("{}({})", rel_transition.name, rel_transition.from_flow_state_name).to_string() + }, + }; + let log_ext = LogParamExt { + scene_kind: Some(vec!["approval_flow".to_string()]), + new_log: Some(true), + project_id: rbum_scope_helper::get_path_item(RbumScopeLevelKind::L2.to_int(), &ctx.own_paths), + ..Default::default() + }; + let log_content = LogParamContent { + subject: Some(subject_text), + name: Some(flow_inst_detail.code.clone()), + ..Default::default() + }; + FlowLogClient::add_ctx_task( + LogParamTag::ApprovalFlow, + Some(flow_inst_detail.id.clone()), + log_content, + Some(TardisFuns::json.obj_to_json(&log_ext).expect("ext not a valid json value")), + Some("dynamic_log_approval_flow".to_string()), + Some(LogParamOp::Finish.into()), + rbum_scope_helper::get_path_item(RbumScopeLevelKind::L1.to_int(), &ctx.own_paths), + ctx, + false, + ) + .await?; + Ok(()) + } + + // 获取需要更新的参数列表 + pub fn get_modify_vars(flow_inst_detail: &FlowInstDetailResp) -> HashMap { + let mut vars_collect = HashMap::new(); + if let Some(artifacts) = &flow_inst_detail.artifacts { + let mut state_ids = vec![]; + for tran in flow_inst_detail.transitions.clone().unwrap_or_default() { + let current_state_id = tran.from_state_id.clone().unwrap_or_default(); + if !state_ids.contains(¤t_state_id) { + state_ids.push(current_state_id.clone()); + if let Some(form_state_vars) = artifacts.form_state_map.get(¤t_state_id) { + for (key, value) in form_state_vars { + *vars_collect.entry(key.clone()).or_insert(json!({})) = value.clone(); + } + } + } + } + }; + + vars_collect + } } diff --git a/backend/middlewares/flow/src/serv/flow_model_serv.rs b/backend/middlewares/flow/src/serv/flow_model_serv.rs index 131d459b..9f9f3ef5 100644 --- a/backend/middlewares/flow/src/serv/flow_model_serv.rs +++ b/backend/middlewares/flow/src/serv/flow_model_serv.rs @@ -97,16 +97,32 @@ impl RbumItemCrudOperation TardisResult<()> { if let Some(rel_transition_ids) = &add_req.rel_transition_ids { - if Self::get_model_id_by_own_paths_and_transition_id( - &add_req.tag.clone().unwrap_or_default(), - &rel_transition_ids.first().cloned().unwrap_or_default(), - funs, - ctx, - ) - .await - .is_ok() - { - return Err(funs.err().not_found(&Self::get_obj_name(), "before_add_item", "The model is not repeatable", "400-flow-model-duplicate")); + for rel_transition_id in rel_transition_ids { + if Self::find_one_detail_item( + &FlowModelFilterReq { + basic: RbumBasicFilterReq { + enabled: Some(true), + ..Default::default() + }, + main: Some(false), + rel_template_id: add_req.rel_template_ids.clone().unwrap_or_default().first().cloned(), + tags: Some(vec![add_req.tag.clone().unwrap_or_default()]), + rel: Some(RbumItemRelFilterReq { + optional: false, + rel_by_from: true, + tag: Some(FlowRelKind::FlowModelTransition.to_string()), + from_rbum_kind: Some(RbumRelFromKind::Item), + rel_item_id: Some(rel_transition_id.clone()), + ..Default::default() + }), + ..Default::default() + }, + funs, + ctx, + ) + .await?.is_some() { + return Err(funs.err().not_found(&Self::get_obj_name(), "before_add_item", "The model is not repeatable", "400-flow-model-duplicate")); + } } }