AWImplClient.java 62 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115
  1. package com.malk.aiwei.service.impl;
  2. import cn.hutool.core.util.ObjectUtil;
  3. import com.alibaba.fastjson.JSON;
  4. import com.malk.aiwei.server.AWServer;
  5. import com.malk.aiwei.service.AWClint;
  6. import com.malk.aiwei.service.AwDingService;
  7. import com.malk.server.aliwork.YDConf;
  8. import com.malk.server.aliwork.YDParam;
  9. import com.malk.server.common.McException;
  10. import com.malk.server.dingtalk.DDConf;
  11. import com.malk.server.teambition.TBConf;
  12. import com.malk.service.aliwork.YDClient;
  13. import com.malk.service.aliwork.YDService;
  14. import com.malk.service.teambition.TBClient;
  15. import com.malk.utils.*;
  16. import lombok.Synchronized;
  17. import lombok.extern.slf4j.Slf4j;
  18. import org.apache.commons.lang3.StringUtils;
  19. import org.springframework.beans.factory.annotation.Autowired;
  20. import org.springframework.stereotype.Service;
  21. import java.util.*;
  22. import java.util.stream.Collectors;
  23. @Service
  24. @Slf4j
  25. public class AWImplClient implements AWClint {
  26. @Autowired
  27. private YDClient ydClient;
  28. @Autowired
  29. private YDService ydService;
  30. @Autowired
  31. private YDConf ydConf;
  32. @Autowired
  33. private AwDingService awDingService;
  34. // 项目主数据表
  35. String _matchFormUuid(String code) {
  36. Map<String, String> formUuid = UtilMap.empty();
  37. if (true || UtilEnv.getActiveProfile().equals(UtilEnv.ENV_PROD)) {
  38. formUuid.put("REVIEW", "FORM-812FD46AF391449A8F206EDB3221B38840UQ"); // 交付物审批记录
  39. formUuid.put("REVIEW_PROCESS", "TPROC--RJC66SC1NEFHXJ0H770K0CF4WN1K21HQ706RL5"); // 交付物审批记录
  40. formUuid.put("PROJECT", "FORM-141E21DF183846028E21727CE43CD1C75CLZ"); // 项目主数据
  41. formUuid.put("APPROVE", "FORM-A25299893F614A6EAA672514D3A76BB0QDBF"); // 交付物审批矩阵
  42. formUuid.put("CHECK", "FORM-1A5D4D7FBF88409B956EBE51F9342A6BKOLP"); // 预检项
  43. formUuid.put("RECORD", "FORM-6E2C0D1197264B8AA23EB3FECAE7344B00BN"); // 预检项记录
  44. formUuid.put("ROLE", "FORM-3C7396A12ADB48A8833EBD90089C93833R21"); // 项目角色
  45. formUuid.put("ROLE_PDT", "FORM-69B6A8151F3346DCA20B0ED54F5380675LJX"); // PDT角色
  46. formUuid.put("CRM_LOG", "FORM-16DD578308E64763AE1539D8176CCCD0GX6F"); // crm推送日志
  47. formUuid.put("TEMPLATE", "FORM-24EFE5C1BCE54E7192E53B33ADFBF1C52NS3"); // TB模板与项目类型映射表
  48. formUuid.put("DOMAIN", "https://yida.awinic.com/"); // 宜搭域名
  49. } else {
  50. // [ fixme: 测试环境添加主数据, 匹配不同架构下企业成员 ]
  51. formUuid.put("REVIEW", "FORM-FBC1A390B4C348089020C763938A6F54RUNY");
  52. formUuid.put("REVIEW_PROCESS", "TPROC--YU966T91PIDH4XDR82OJC8GVOP7Z19PODTXQL6");
  53. formUuid.put("PROJECT", "FORM-84EF78C7DBA047E58A8C8511106F91D5WNVI");
  54. formUuid.put("APPROVE", "FORM-AB7263D2A12F4E01A871656F0D995BC96PR1");
  55. formUuid.put("CHECK", "FORM-E6CB042D7929448888AE5E2B27631E57IVPM");
  56. formUuid.put("RECORD", "FORM-7B63BB056145452F8BC0A2C52492DE00QVBH");
  57. formUuid.put("ROLE", "FORM-5BE21392886E46DF955D1EBC100ADA429NON");
  58. formUuid.put("CRM_LOG", "FORM-7F40E85D4FA8487C8E00C9156B76A953CBMG");
  59. formUuid.put("TEMPLATE", "FORM-C864290977A64FC4A061434DF21D5871JA04");
  60. formUuid.put("DOMAIN", "https://kabom7.aliwork.com/");
  61. }
  62. return formUuid.get(code);
  63. }
  64. /**
  65. * 交付物审批 [ppExt: 宜搭附件传递 downloadUrl 和 name 即可实现在线预览]
  66. */
  67. @Override
  68. public void doApprove(Map data) {
  69. String projectId = UtilMap.getString(data, "projectId");
  70. String workFlowApprove = _getWorkFlowStatus(projectId, AWServer.WORKFLOW_APPROVE);
  71. if (!data.get("tfsId").equals(workFlowApprove)) {
  72. return;
  73. }
  74. log.info("交付物审批, {}", data);
  75. String pCode = UtilMap.getString(data, "projectId");
  76. String creatorId = UtilMap.getString(data, "creatorId");
  77. String taskId = UtilMap.getString(data, "taskId");
  78. Map taskData = _getTaskFieldMap(UtilMap.getString(data, "taskId"), AWServer.TASK_CODE, AWServer.TASK_APPROVE_ATTACHMENT, AWServer.TASK_APPROVE_DESC);
  79. String tCode = UtilMap.getString(taskData, AWServer.TASK_CODE);
  80. // 工作流tb虽然配置了流转逻辑, 冗余稳定性
  81. List<Map> formList = (List<Map>) ydClient.queryData(YDParam.builder()
  82. .formUuid(_matchFormUuid("REVIEW"))
  83. .searchFieldJson(JSON.toJSONString(UtilMap.map("selectField_lqxuswzd, textField_lrncs2fu", tCode, pCode)))
  84. .instanceStatus("RUNNING")
  85. .build(), YDConf.FORM_QUERY.retrieve_search_process).getData();
  86. if (formList.size() > 0 && formList.stream().filter(item -> tCode.equals(((Map) item.get("data")).get("selectField_lqxuswzd"))).findAny().isPresent()) {
  87. return;
  88. }
  89. String result = "";
  90. List<Map> tList = null;
  91. Map rProject = null;
  92. if (StringUtils.isBlank(tCode)) {
  93. result = "无任务编码";
  94. } else {
  95. List<Map> pList = ydService.queryDataList_FormData(_matchFormUuid("PROJECT"), UtilMap.map("textField_lqxtykce", pCode));
  96. pList = pList.stream().filter(item -> pCode.equals(item.get("textField_lqxtykce"))).collect(Collectors.toList());
  97. if (pList.size() == 0) {
  98. result = "未匹配到项目主数据";
  99. } else {
  100. rProject = pList.get(0);
  101. // prd 多模板适配: [任务号 + 项目类型]
  102. String proType = UtilMap.getString(rProject, "textField_ltwcq7s6");
  103. tList = ydService.queryDataList_FormData(_matchFormUuid("APPROVE"), UtilMap.map("selectField_lrncf4hk, radioField_lrnddfq6", tCode, "启用"));
  104. tList = tList.stream().filter(item -> tCode.equals(item.get("selectField_lrncf4hk")) && UtilMap.getList(item, "multiSelectField_ltwjre9s").contains(proType)).collect(Collectors.toList());
  105. if (tList.size() == 0) {
  106. result = "未配置交付物审批";
  107. }
  108. }
  109. /// 发起评审
  110. if (StringUtils.isBlank(result)) {
  111. List<Map> roles = (List<Map>) rProject.get("tableField_lqxtykcf");
  112. Map formData = UtilMap.map("selectField_lqxuswzd, textField_lrncs2fu", tCode, pCode);
  113. formData.put("attachmentField_lqxtebtq", _getDocs(taskData));
  114. formData.put("selectField_lqxuswze", UtilMap.getString(rProject, "textField_lrj7vnxb")); // 项目编号
  115. formData.put("textField_lqxuc9m4", UtilMap.getString(rProject, "textareaField_lrj7vnxl")); // 项目描述
  116. formData.put("employeeField_ltzn872j", UtilMap.getString(rProject, "employeeField_ltzn872j_id")); // 项目经理 0402 控制矩阵角色为空, 流转到PM
  117. formData.put("textField_ltwcq7s6", UtilMap.getString(rProject, "textField_ltwcq7s6")); // 项目类型
  118. formData.put("textareaField_lw0nmvko", UtilMap.getString(taskData, AWServer.TASK_APPROVE_DESC)); // 交付件描述
  119. // 匹配任务编码与项目角色
  120. List<Map<String, String>> compIds = Arrays.asList( // 任务表角色, 交付物评审表: 角色, 审批人
  121. UtilMap.map("tsRole, prRole, prEmp", "multiSelectField_lrokzlo7, multiSelectField_lrokzlo7, employeeField_lqxtebtw"),
  122. UtilMap.map("tsRole, prRole, prEmp", "multiSelectField_lrokzlo8, multiSelectField_lrokzlo8, employeeField_lqxtebtx"),
  123. UtilMap.map("tsRole, prRole, prEmp", "multiSelectField_lrokzlo9, multiSelectField_lrokzlo9, employeeField_lqxtebty"));
  124. Map rCondition = tList.get(0);
  125. for (Map compId : compIds) {
  126. // ppExt: 宜搭多选若无值则不返回字段 [prd 角色多选, 会签逻辑]
  127. if (!rCondition.containsKey(compId.get("tsRole"))) {
  128. continue;
  129. }
  130. List<String> namesCondition = (List<String>) rCondition.get(compId.get("tsRole"));
  131. List<String> namesApproveRole = new ArrayList<>();
  132. List<String> namesApprover = new ArrayList<>();
  133. for (String name : namesCondition) {
  134. Optional optional = roles.stream().filter(item -> UtilString.equal(name, item.get("selectField_lqxu6bgf"))).findAny();
  135. if (optional.isPresent()) {
  136. Map detail = (Map) optional.get();
  137. if (ObjectUtil.isNotNull(detail.get("selectField_lqxu6bgf"))) {
  138. namesApproveRole.add(String.valueOf(detail.get("selectField_lqxu6bgf")));
  139. }
  140. if (ObjectUtil.isNotNull(detail.get("employeeField_lqxtykch_id"))) {
  141. namesApprover.addAll((List) detail.get("employeeField_lqxtykch_id"));
  142. }
  143. }
  144. }
  145. formData.put(compId.get("prRole"), namesApproveRole);
  146. formData.put(compId.get("prEmp"), namesApprover);
  147. }
  148. // 组装数据
  149. Map<String, String> extra = (Map) tbClient.idMapQuery(creatorId, "dingTalk-user", ddConf.getCorpId()).get(0).get("extra");
  150. UtilMap.putAll(formData, UtilMap.map("textField_lr3dlwsa, textField_lr3er4qb", taskId, creatorId));
  151. formData.put("employeeField_lui5fu7z", Arrays.asList(extra.get("userId"))); // 0402 指定发起人, 用于结构化数据子流程发起人
  152. String instanceId = (String) ydClient.operateData(YDParam.builder()
  153. .formUuid(_matchFormUuid("REVIEW"))
  154. .processCode(_matchFormUuid("REVIEW_PROCESS"))
  155. .formDataJson(JSON.toJSONString(formData))
  156. .userId(extra.get("userId"))
  157. .build(), YDConf.FORM_OPERATION.start);
  158. result = _matchFormUuid("DOMAIN") + ydConf.getAppType() + "/processDetail?procInsId=" + instanceId;
  159. }
  160. }
  161. log.info("交付物结果, {}", result);
  162. Map body = TBConf.assembleCustomFieldName(AWServer.TASK_APPROVE_LINK, result);
  163. tbClient.updateTaskCustomField(taskId, tbConf.getOperatorId(), body);
  164. }
  165. /**
  166. * 交付物审批回调
  167. */
  168. @Override
  169. public void approved(Map data) {
  170. log.info("交付物审批回调, {}", data);
  171. String result = String.valueOf(data.get("approve"));
  172. String workFlowId = _getWorkFlowStatus(String.valueOf(data.get("projectId")), result);
  173. tbClient.updateTaskFlowStatus(String.valueOf(data.get("taskId")), tbConf.getOperatorId(), workFlowId, String.valueOf(data.get("approve")));
  174. }
  175. @Autowired
  176. private TBClient tbClient;
  177. @Autowired
  178. private DDConf ddConf;
  179. // 获取任务详情 fixme 提取到 tbservice
  180. private Map<String, String> _getTaskFieldMap(Map task, List<Map> projectCustomField, String... fieldNames) {
  181. return __getTaskMap(task, projectCustomField, fieldNames);
  182. }
  183. // 组装任务数据
  184. private Map __getTaskMap(Map task, List<Map> projectCustomField, String... fieldNames) {
  185. Map result = UtilMap.map("task", task);
  186. List<Map> taskCustomField = (List<Map>) task.get("customfields");
  187. for (String filed : fieldNames) {
  188. Optional optional = projectCustomField.stream().filter(item -> filed.equals(item.get("name"))).findAny();
  189. if (optional.isPresent()) {
  190. Map map = (Map) optional.get();
  191. result.put(filed, TBConf.getTaskFieldValue_First(taskCustomField, String.valueOf(map.get("id"))));
  192. result.put(filed + "_id", map.get("id")); // ppExt: 返回字段ID
  193. }
  194. }
  195. return result;
  196. }
  197. // 获取任务详情 fixme 提取到 tbservice
  198. private Map _getTaskFieldMap(String taskId, String... fieldNames) {
  199. // 查询任务详情
  200. Map rTask = tbClient.queryTaskDetail(taskId, null, null).get(0);
  201. // 查询项目任务ID
  202. List<Map> customField = tbClient.queryProjectCustomField(String.valueOf(rTask.get("projectId")), null);
  203. return __getTaskMap(rTask, customField, fieldNames);
  204. }
  205. // 匹配工作流名称, 获取ID
  206. @Deprecated
  207. String _getWorkFlowStatus(String projectId, String workFlowStatusName) {
  208. List<Map> customFlowStatus = tbClient.queryProjectCustomFlowStatus(projectId, UtilMap.map("q", workFlowStatusName));
  209. McException.assertAccessException(customFlowStatus.isEmpty(), "工作流名称未匹配");
  210. return String.valueOf(customFlowStatus.get(0).get("id"));
  211. }
  212. // 前置过滤 fixme 提取方法
  213. List<String> _getWorkFlowStatusList(String projectId, String... workFlowStatusNames) {
  214. List<String> workFlowStatusList = new ArrayList<>();
  215. for (String name : workFlowStatusNames) {
  216. List<Map> customFlowStatus = tbClient.queryProjectCustomFlowStatus(projectId, UtilMap.map("q", name));
  217. workFlowStatusList.addAll(customFlowStatus.stream().map(item -> UtilMap.getString(item, "id")).collect(Collectors.toList()));
  218. }
  219. return workFlowStatusList;
  220. }
  221. // 后置过滤 fixme 提取方法
  222. List<String> _getWorkFlowStatusList(String projectId, List<String> workFlowStatusNames) {
  223. //List<String> names = Arrays.stream(workFlowStatusNames).collect(Collectors.toList());
  224. List<Map> customFlowStatus = tbClient.queryProjectCustomFlowStatus(projectId, null);
  225. List<Map> workFlowStatusList = customFlowStatus.stream().filter(item -> workFlowStatusNames.contains(item.get("name"))).collect(Collectors.toList());
  226. return workFlowStatusList.stream().map(item -> UtilMap.getString(item, "id")).collect(Collectors.toList());
  227. }
  228. /**
  229. * 获取知识库附件传递到宜搭
  230. * - ppExt -
  231. * 1. 宜搭附件传递 downloadUrl 和 name 即可实现在线预览
  232. * 2. 知识库绑定没有文件后缀, 宜搭识别目前仅能点击下载跳转预览, 添加统一docx后缀, 文档会自行区分
  233. */
  234. List<Map> _getDocs(Map taskData) {
  235. Map rTask = UtilMap.getMap(taskData, "task");
  236. List<Map> customfields = UtilMap.getList(rTask, "customfields");
  237. String deliverable = String.valueOf(taskData.get(AWServer.TASK_APPROVE_ATTACHMENT + "_id")); // 交付物任务字段ID
  238. List<Map> attachments = TBConf.getTaskFieldValue(customfields, deliverable).stream().map(item -> {
  239. Map link = (Map) JSON.parse(String.valueOf(item.get("metaString")));
  240. return UtilMap.map("downloadUrl, name", link.get("url"), link.get("title") + ".docx");
  241. }).collect(Collectors.toList());
  242. return attachments;
  243. }
  244. /**
  245. * 检查项check
  246. *
  247. * @param isTask 服务于任务自定义详情卡片, 否则为宜搭发起: 抛出错误信息
  248. */
  249. @Override
  250. public Map doCheck(String taskId, boolean isTask) {
  251. log.info("检查项check, {}", taskId);
  252. Map taskData = _getTaskFieldMap(taskId, AWServer.TASK_CODE, AWServer.TASK_CHECK_LINK, AWServer.TASK_ROLE, AWServer.TASK_STAGE, AWServer.TASK_CHECK_STATUS, AWServer.TASK_PRODUCT, AWServer.TASK_PRODUCT_VERSION);
  253. Map rTask = UtilMap.getMap(taskData, "task");
  254. String pCode = UtilMap.getString(rTask, "projectId");
  255. if (AWServer.PROJECT_IGNORE_ID.contains(pCode)) {
  256. return UtilMap.map("result", "模板/复用项目");
  257. }
  258. String checkLink = UtilMap.getString(taskData, AWServer.TASK_CHECK_LINK);
  259. /// prd 预检项持续维护需要重新匹配, 任务号和主数据判定为非法, 无需重新刷新
  260. if (isTask && (StringUtils.isNotBlank(checkLink) && !"未配置预检项".equals(checkLink))) {
  261. return UtilMap.map("result", "链接已加载");
  262. }
  263. // prd 非未完成任务若是未配置预检项状态, 忽略更新
  264. if (isTask && !_getWorkFlowStatusList(pCode, AWServer.WORKFLOW_INITIAL).contains(rTask.get("tfsId"))) {
  265. return UtilMap.map("result", "任务已执行");
  266. }
  267. String creatorId = UtilMap.getString(rTask, "creatorId");
  268. String tCode = UtilMap.getString(taskData, AWServer.TASK_CODE);
  269. String result = "";
  270. List<Map> tList = null;
  271. Map rProject = null;
  272. if (StringUtils.isBlank(tCode)) {
  273. result = "无任务编码";
  274. } else {
  275. List<Map> pList = ydService.queryDataList_FormData(_matchFormUuid("PROJECT"), UtilMap.map("textField_lqxtykce", pCode));
  276. pList = pList.stream().filter(item -> pCode.equals(item.get("textField_lqxtykce"))).collect(Collectors.toList());
  277. if (pList.size() == 0) {
  278. result = "未匹配到项目主数据";
  279. } else {
  280. rProject = pList.get(0);
  281. String proType = UtilMap.getString(rProject, "textField_ltwcq7s6");
  282. tList = ydService.queryFormData_all(YDParam.builder()
  283. .formUuid(_matchFormUuid("CHECK"))
  284. .searchFieldJson(JSON.toJSONString(UtilMap.map("selectField_lrncf4hk, radioField_lrnddfq6", tCode, "启用")))
  285. .build());
  286. // prd 多模板适配: [任务号 + 项目类型]
  287. tList = tList.stream().filter(item -> tCode.equals(item.get("selectField_lrncf4hk")) && UtilMap.getList(item, "multiSelectField_ltwjre9s").contains(proType)).collect(Collectors.toList());
  288. if (tList.size() == 0) {
  289. result = "未配置预检项";
  290. } else {
  291. result = _matchFormUuid("DOMAIN") + ydConf.getAppType() + "/workbench/" + _matchFormUuid("RECORD") + "?taskId=" + taskId;
  292. }
  293. }
  294. }
  295. log.info("检查项结果, {}", result);
  296. if (isTask) {
  297. if (!checkLink.equals(result)) {
  298. Map body = TBConf.assembleCustomFieldName(AWServer.TASK_CHECK_LINK, result);
  299. tbClient.updateTaskCustomField(taskId, tbConf.getOperatorId(), body);
  300. }
  301. // prd 未配置预检项更新为已检查, 避免完成任务完成触发必填校验 [ppExt 避免重复写入, 重复写入TB会重复记录日志
  302. if (!result.startsWith("http") && !taskData.get(AWServer.TASK_CHECK_STATUS).equals("已检查")) {
  303. Map body = TBConf.assembleCustomFieldName(AWServer.TASK_CHECK_STATUS, "已检查");
  304. tbClient.updateTaskCustomField(taskId, tbConf.getOperatorId(), body);
  305. }
  306. return UtilMap.map("result", result);
  307. }
  308. McException.assertAccessException(!result.startsWith("https://"), result);
  309. Map formData = UtilMap.map("selectField_lqxuswzd, textField_lrndwu00, textField_lr3dlwsa, textField_lr3er4qb", tCode, pCode, taskId, creatorId);
  310. // 项目信息
  311. formData.putAll(UtilMap.map("selectField_lqxuswze, textField_lrndwu07, textField_lqxuc9m4", UtilMap.getString(rProject, "textField_lrj7vnxb"), UtilMap.getString(rProject, "textareaField_lrj7vnxl"), UtilMap.getString(rProject, "textField_lqxtykcd")));
  312. // 任务信息 [TBManager 操作]
  313. Map<String, String> extra = (Map) tbClient.idMapQuery(creatorId, "dingTalk-user", ddConf.getCorpId()).get(0).get("extra");
  314. // ppExt: TB映射钉钉信息仅返回 userId, 人员组件返回前端展示需要 name. 通过页面公式临时自动加载, 可通过TB接口或钉钉用户接口查询
  315. formData.putAll(UtilMap.map("textField_lrndwu09, textField_lrndwu0a, textField_lrndwu0b, employeeField_lrndwu0e, textField_lvbrueqs, textField_lvbrueqt", rTask.get("content"), taskData.get(AWServer.TASK_STAGE), taskData.get(AWServer.TASK_ROLE), Arrays.asList(UtilMap.map("value, name", extra.get("userId"), extra.get("userId"))), taskData.get(AWServer.TASK_PRODUCT), taskData.get(AWServer.TASK_PRODUCT_VERSION)));
  316. // 返回检查项
  317. formData.put("tableField_lqxxgj4s", tList.stream().map(item -> {
  318. item.put("associationFormField_lrrnem5r", YDConf.associationForm(ydConf.getAppType(), _matchFormUuid("CHECK"), UtilMap.getString(item, "instanceId"), UtilMap.getString(item, "textField_lrnd3h0r"), "", false));
  319. return item;
  320. }).collect(Collectors.toList()));
  321. return formData;
  322. }
  323. /**
  324. * 检查项回调
  325. */
  326. @Override
  327. public void checked(Map data) {
  328. log.info("检查项回调, {}", data);
  329. Map body = TBConf.assembleCustomFieldName(AWServer.TASK_CHECK_STATUS, "已检查");
  330. tbClient.updateTaskCustomField(String.valueOf(data.get("taskId")), tbConf.getOperatorId(), body);
  331. String result = _matchFormUuid("DOMAIN") + ydConf.getAppType() + "/formDetail?formInstId=" + data.get("formInstId");
  332. Map body2 = TBConf.assembleCustomFieldName(AWServer.TASK_CHECK_LINK, result);
  333. tbClient.updateTaskCustomField(String.valueOf(data.get("taskId")), tbConf.getOperatorId(), body2);
  334. }
  335. /**
  336. * 同步项目主数据 [兼容新增]
  337. */
  338. @Synchronized
  339. @Override
  340. public Map syncProject(String projectCode) {
  341. // 新增场景下, 提供实例ID回写TB项目编号
  342. Map formData = null;
  343. // 查询项目主数据 [艾为应用]
  344. YDParam ydParam = YDParam.builder()
  345. .appType("APP_QBWQITQBSPJNYTUTNPDK")
  346. .systemToken("8F966HB12J27MQJM6V4IQDYHYTPA2G4GTZGCLN1")
  347. .formUuid("FORM-Q4A664A1V158R81L662N9507PLND288T7O6ELN")
  348. .build();
  349. if (StringUtils.isNotBlank(projectCode)) {
  350. ydParam.setSearchFieldJson(JSON.toJSONString(UtilMap.map("textField_le6o88w0", projectCode)));
  351. }
  352. // prd 客户系统主数据, 是全量定时从mdm更新, 因此不能增量
  353. List<Map> mapBaseList = ydService.queryFormData_all(ydParam);
  354. if (StringUtils.isNotBlank(projectCode)) {
  355. mapBaseList = mapBaseList.stream().filter(item -> projectCode.equals(item.get("textField_le6o88w0"))).collect(Collectors.toList());
  356. }
  357. log.info("项目主数据, {}", mapBaseList.size());
  358. McException.assertAccessException(mapBaseList.isEmpty(), "未匹配项目主数据");
  359. // 查询项目角色
  360. YDParam ydParam1 = YDParam.builder()
  361. .formUuid(_matchFormUuid("ROLE"))
  362. .build();
  363. List<Map> mapRole = (List<Map>) ydClient.queryData(ydParam1, YDConf.FORM_QUERY.retrieve_search_form).getData();
  364. // prd 5.14 C类项目成员同步PDT成员主数据
  365. ydParam1.setFormUuid(_matchFormUuid("ROLE_PDT"));
  366. List<Map> rolePDT = (List<Map>) ydClient.queryData(ydParam1, YDConf.FORM_QUERY.retrieve_search_form).getData();
  367. // 匹配数据组件ID
  368. for (Map baseFormData : mapBaseList) {
  369. // 项目主数据
  370. formData = new HashMap();
  371. formData.put("textField_lqxtykcd", baseFormData.get("textField_lo2c1f0l")); // 项目名称
  372. formData.put("textField_lrj7vnxb", baseFormData.get("textField_le6o88w0")); // 项目号
  373. formData.put("textField_lrj7vnxc", baseFormData.get("textField_le6o88w1")); // 项目简称
  374. formData.put("textareaField_lrj7vnxl", baseFormData.get("textField_le6o88w2")); // 项目描述
  375. formData.put("textField_lrj7vnxf", baseFormData.get("formInstanceId")); // 项目实例ID
  376. formData.put("textField_ltsdsti6", baseFormData.get("textField_libg2ea8")); // PDT (项目分组)
  377. formData.put("textField_ltsdsti7", baseFormData.get("textField_llouhiyf")); // 项目状态
  378. formData.put("textField_ltsdsti8", baseFormData.get("textField_lo2c1f0r")); // mPDT
  379. formData.put("textField_ltsdsti9", baseFormData.get("textField_lki3egzn")); // Charter子类
  380. formData.put("textField_ltwcq7s5", baseFormData.get("textField_lki3egzm")); // Charter
  381. formData.put("textField_ltwcq7s6", baseFormData.get("textField_lnjqpeie")); // 项目类型 (TB项目模板)
  382. List<Map> details = new ArrayList<>();
  383. List<Map> roleMap = mapRole;
  384. Map projectMembers = baseFormData;
  385. // 项目角色详情 [prd 5.14 C类项目成员同步PDT成员主数据]
  386. if ("Charter开发项目".equals(formData.get("textField_ltwcq7s6"))) {
  387. roleMap = rolePDT;
  388. String charter = String.valueOf(formData.get("textField_ltsdsti6"));
  389. YDParam ydParam2 = YDParam.builder()
  390. .appType("APP_LH2TJTIT3EWHCG084TGL")
  391. .systemToken("QYC66KA1F4ZG5RQV8HR9WBHYW03S3CQAJKDQLZ1")
  392. .formUuid("FORM-90E23DE1BBEB43759ED9517F98BA4938P61C")
  393. .searchFieldJson(JSON.toJSONString(UtilMap.map("textField_lr0dvhqh", charter)))
  394. .build();
  395. List<Map> listPDT = (List<Map>) ydClient.queryData(ydParam2, YDConf.FORM_QUERY.retrieve_search_form).getData();
  396. listPDT = listPDT.stream().filter(item -> charter.equals(((Map) item.get("formData")).get("textField_lr0dvhqh"))).collect(Collectors.toList());
  397. if (listPDT.size() == 0) {
  398. continue;
  399. }
  400. projectMembers = (Map) listPDT.get(0).get("formData");
  401. }
  402. for (Map roleData : roleMap) {
  403. Map roleFormData = (Map) roleData.get("formData");
  404. // 创建详情的角色名称和成员组件Map
  405. Object roleMember = projectMembers.get(roleFormData.get("textField_lr7bgi76"));
  406. String roleName = UtilMap.getString(roleFormData, "textField_lqxu439g");
  407. if (roleMember != null && roleName != null) {
  408. // prd 提交代表标识, 用于同步项目角色
  409. details.add(UtilMap.map("employeeField_lqxtykch, selectField_lqxu6bgf, radioField_lu8li5ie", roleMember, roleName, roleFormData.get("radioField_lu8li5ie")));
  410. // prd 添加项目经理, 用于数据权限控制
  411. if (AWServer.PROJECT_PM_NAME.equals(roleName)) {
  412. formData.put("employeeField_ltzn872j", roleMember);
  413. }
  414. }
  415. }
  416. formData.put("tableField_lqxtykcf", details);
  417. // 通过项目号获取项目是否已存在
  418. List<Map> projectMaps = ydService.queryDataList_FormData(_matchFormUuid("PROJECT"), UtilMap.map("textField_lrj7vnxb", baseFormData.get("textField_le6o88w0")));
  419. projectMaps = projectMaps.stream().filter(sub -> baseFormData.get("textField_le6o88w0").equals(sub.get("textField_lrj7vnxb"))).collect(Collectors.toList());
  420. YDParam ydParam2 = YDParam.builder()
  421. .formUuid(_matchFormUuid("PROJECT"))
  422. .formDataJson(JSON.toJSONString(formData))
  423. .updateFormDataJson(JSON.toJSONString(formData))
  424. .useLatestVersion(true)
  425. .build();
  426. if (projectMaps.size() > 0) {
  427. ydParam2.setFormInstanceId(String.valueOf(projectMaps.get(0).get("instanceId")));
  428. ydClient.operateData(ydParam2, YDConf.FORM_OPERATION.update); // 主表为增量更新
  429. // 保留tb项目号, 避免冲重复通过模板参加项目 [供新增用]
  430. formData.put("textField_lqxtykce", UtilMap.getString(projectMaps.get(0), "textField_lqxtykce"));
  431. formData.put("formInstanceId", projectMaps.get(0).get("instanceId"));
  432. } else {
  433. String formInstId = (String) ydClient.operateData(ydParam2, YDConf.FORM_OPERATION.create);
  434. formData.put("formInstanceId", formInstId);
  435. }
  436. // 全量兼容: tb项目成员更新 [兼容存量创建项目, 被删除后执行异常]
  437. String projectId = UtilMap.getString(formData, "textField_lqxtykce");
  438. if (StringUtils.isNotBlank(projectId) && StringUtils.isBlank(projectCode)) {
  439. try {
  440. _syncProjectRole(formData, projectId, false);
  441. } catch (Exception e) {
  442. // 记录错误信息
  443. e.printStackTrace();
  444. }
  445. }
  446. log.info("同步项目主数据, code = {}, id = {}", projectCode, projectCode);
  447. }
  448. return formData;
  449. }
  450. @Autowired
  451. private TBConf tbConf;
  452. /**
  453. * 通过模板创建项目 [ppExt: 过滤宜搭项目档案\TB项目, 避免重复创建, 若判定为更新下执行项目成员更新]
  454. */
  455. @Override
  456. @Synchronized
  457. public void createProject(String projectCode, String templateId) {
  458. log.info("通过模板创建项目, {}, {}", projectCode, templateId);
  459. // 虽然先创建TB, 可以一次性同步项目数据, 但避免执行异常, 因此成功创建TB项目再回写
  460. Map formData = syncProject(projectCode);
  461. String projectId = UtilMap.getString(formData, "textField_lqxtykce");
  462. log.info("项目主数据, {}, {}", projectId, formData);
  463. boolean isCreate = StringUtils.isBlank(projectId);
  464. if (isCreate) {
  465. //awDingService.saveGroup(upMap,formData);
  466. // prd 多模板适配: [templateId 为空, 触发项目类型匹配]
  467. if (StringUtils.isBlank(templateId)) {
  468. List<Map> dataList = ydService.queryDataList_FormData(_matchFormUuid("TEMPLATE"), null);
  469. Optional optional = dataList.stream().filter(item -> item.get("selectField_ltwcsr8w").equals(formData.get("textField_ltwcq7s6"))).findAny();
  470. McException.assertAccessException(!optional.isPresent(), projectCode + ": TB项目模板未维护!");
  471. templateId = UtilMap.getString((Map) optional.get(), "textField_ltwcybz1");
  472. McException.assertAccessException(templateId.length() <= 1, projectCode + ": TB项目模板未匹配!");
  473. }
  474. // 通过模板创建项目, 创建项目并更新项目TB项目ID
  475. Map result = tbClient.projectCreateWithTemplate(projectCode, templateId, tbConf.getOperatorId());
  476. projectId = UtilMap.getString(result, "id");
  477. ydClient.operateData(YDParam.builder()
  478. .formInstanceId(UtilMap.getString(formData, "formInstanceId"))
  479. .updateFormDataJson(JSON.toJSONString(UtilMap.map("textField_lqxtykce", projectId)))
  480. .build(), YDConf.FORM_OPERATION.update);
  481. log.info("TB项目号更新, {}, {}", projectCode, UtilMap.getString(formData, "formInstanceId"), projectId);
  482. }
  483. _syncProjectRole(formData, projectId, isCreate);
  484. }
  485. /**
  486. * 项目主数据增量更新
  487. */
  488. @Override
  489. public void updateProject(String projectCode) {
  490. Map formData = syncProject(projectCode);
  491. log.info("项目主数据增量更新, {}", formData);
  492. String projectId = UtilMap.getString(formData, "textField_lqxtykce");
  493. if (StringUtils.isNotBlank(projectId)) { // 屏蔽未创建项目, 出现新增
  494. _syncProjectRole(formData, projectId, false);
  495. }
  496. }
  497. // 同步TB项目成员/项目分组 todo 群成员
  498. void _syncProjectRole(Map formData, String projectId, boolean isCreate) {
  499. // prd TB项目成员, 项目经理设置为管理者
  500. List<String> pmUserId = new ArrayList<>();
  501. List<String> roleIds = new ArrayList<>();
  502. List<String> representativeIds = new ArrayList<>();
  503. String representative = _getProjectRoleId(projectId, "项目各代表");
  504. String pmRoleId = _getProjectRoleId(projectId, AWServer.PROJECT_PM_ROLE);
  505. String staffRoleId = _getProjectRoleId(projectId, "项目成员");
  506. List<Map> details = (List<Map>) formData.get("tableField_lqxtykcf");
  507. for (Map row : details) {
  508. // 同步项目主数据, 会重新匹配项目成员, 因此无需 employeeField_xxx_id 取值
  509. List<String> userIds = (List<String>) UtilMap.getList(row, "employeeField_lqxtykch");
  510. if (userIds.isEmpty()) {
  511. continue;
  512. }
  513. String roleName = UtilMap.getString(row, "selectField_lqxu6bgf");
  514. if (AWServer.PROJECT_PM_NAME.equals(roleName)) {
  515. pmUserId.addAll(userIds);
  516. }
  517. // prd 项目各代表, 可独立N/A任务, 添加角色控制权限
  518. if ("是".equals(row.get("radioField_lu8li5ie"))) {
  519. representativeIds.addAll(userIds);
  520. }
  521. roleIds.addAll(userIds);
  522. }
  523. // 钉钉人员ID转TB人员ID
  524. roleIds = _convertUserId(roleIds, false);
  525. tbClient.createProjectMember(projectId, roleIds, tbConf.getOperatorId());
  526. if (!representativeIds.isEmpty()) {
  527. tbClient.updateProjectMember(_convertUserId(representativeIds, false), Arrays.asList(staffRoleId, representative), projectId);
  528. }
  529. // prd 不做删除, 离职自动退出, 避免临时交接不可见的问题. 但需要移除项目经理权限
  530. List<Map> roleList = tbClient.queryProjectMember(projectId, UtilMap.map("projectRoleId", pmRoleId));
  531. for (Map role : roleList) {
  532. List<String> tRoleIds = UtilMap.getList(role, "roleIds");
  533. tRoleIds.remove(pmRoleId);
  534. if (tRoleIds.isEmpty()) {
  535. tRoleIds.add(staffRoleId);
  536. }
  537. tbClient.updateProjectMember(Arrays.asList(UtilMap.getString(role, "userId")), tRoleIds, projectId);
  538. }
  539. // 指定项目经理为编辑权限
  540. if (!pmUserId.isEmpty()) {
  541. tbClient.updateProjectMember(_convertUserId(pmUserId, false), Arrays.asList(pmRoleId), projectId);
  542. }
  543. // 同步项目分组 [prd 主数据建立产品与分组对照表,绑定末级分类接口,若未匹配则不添加]
  544. String groupName = UtilMap.getString(formData, "textField_ltsdsti6");
  545. if (StringUtils.isNotBlank(groupName)) {
  546. List<Map> tagList = tbClient.queryTagList(UtilMap.map("q", groupName));
  547. if (tagList.size() > 0) {
  548. tbClient.updateProjectTag(projectId, Arrays.asList(UtilMap.getString(tagList.get(0), "id"), staffRoleId), tbConf.getOperatorId());
  549. }
  550. }
  551. // prd 人员变更, 自动同步更新执行人: 仅触发存在执行人 & 未完成场景 [5.10 新增项目不执行, TB任务号宜搭回写有延迟, 避免异常]
  552. if (!isCreate) {
  553. updateProjectRole(projectId, Arrays.asList("全部", "自动更新"));
  554. }
  555. log.info("TB项目信息, {}, {}", projectId, roleIds.size());
  556. }
  557. // fixme 提取方法, 参考 getProjectCFID 实现
  558. private String _getProjectRoleId(String projectId, String roleName) {
  559. List<Map> roles = tbClient.queryProjectRoles(projectId);
  560. Optional optional = roles.stream().filter(item -> roleName.equals(item.get("name"))).findAny();
  561. McException.assertAccessException(!optional.isPresent(), roleName + ": 项目角色不存在");
  562. return UtilMap.getString((Map) optional.get(), "id");
  563. }
  564. /**
  565. * 分配项目角色 prd 若是一人直接指定, 多人情况下忽略
  566. */
  567. @Synchronized
  568. @Override
  569. public void updateProjectRole(String projectId, List<String> trNode) {
  570. // 项目档案 [tb项目号查询, 避免模糊匹配]
  571. List<Map> pList = ydService.queryDataList_FormData(_matchFormUuid("PROJECT"), UtilMap.map("textField_lqxtykce", projectId));
  572. McException.assertAccessException(pList.isEmpty(), "未匹配到项目主数据");
  573. List<Map> rList = (List<Map>) pList.get(0).get("tableField_lqxtykcf");
  574. log.info("分配项目角色 projectId = {}, 项目数量 = {}, 角色列表 = {}", projectId, pList.size(), rList);
  575. // 任务编码字段ID
  576. List<Map> customField = tbClient.queryProjectCustomField(projectId, null);
  577. // prd 非未完成任务, 不执行人员匹配更新
  578. List<String> workFlowStatusList = _getWorkFlowStatusList(projectId, AWServer.WORKFLOW_INITIAL);
  579. //List<Map> taskList = tbClient.queryProjectTaskList(projectId, UtilMap.map("pageSize", 10), null);
  580. tbClient.queryProjectTaskList(projectId, null, taskList -> {
  581. for (Map task : taskList) {
  582. if (!workFlowStatusList.contains(task.get("tfsId"))) {
  583. continue;
  584. }
  585. // 获取资源名称对应的项目角色
  586. Optional optional = customField.stream().filter(item -> AWServer.TASK_ROLE.equals(item.get("name"))).findAny();
  587. if (!optional.isPresent()) {
  588. continue;
  589. }
  590. List<Map> customfields = (List<Map>) task.get("customfields");
  591. // prd 按照tr节点分配成员, 若是含有全部则忽略
  592. if (!trNode.contains("全部")) {
  593. Optional optional2 = customField.stream().filter(item -> AWServer.TASK_STAGE.equals(item.get("name"))).findAny();
  594. if (!optional2.isPresent()) {
  595. continue;
  596. }
  597. // prd 传递无TR评审节点值, 匹配空值, 作为分配条件
  598. String trName = TBConf.getTaskFieldValue_First(customfields, UtilMap.getString((Map) optional2.get(), "id"));
  599. if (StringUtils.isBlank(trName)) {
  600. if (!trNode.contains(AWServer.TASK_STAGE_BLANK)) {
  601. continue;
  602. }
  603. } else {
  604. if (!trNode.contains(trName)) {
  605. continue;
  606. }
  607. }
  608. log.info("tr节点控制, {}, {}", trName, trNode);
  609. }
  610. // prd 人员变更, 自动同步更新执行人: 仅触发存在执行人 & 未完成场景
  611. if (trNode.contains("自动更新")) {
  612. if (StringUtils.isBlank(UtilMap.getString(task, "executorId"))) {
  613. continue;
  614. }
  615. }
  616. String roleName = TBConf.getTaskFieldValue_First(customfields, UtilMap.getString((Map) optional.get(), "id"));
  617. // 获取角色在项目主数据对应成员
  618. optional = rList.stream().filter(item -> roleName.equals(item.get("selectField_lqxu6bgf"))).findAny();
  619. if (!optional.isPresent()) {
  620. continue;
  621. }
  622. log.info("分配项目角色 taskId = {}, 资源名称 = {}, 角色信息 = {}", UtilMap.getString(task, "id"), roleName, optional.get());
  623. List<String> roleIds = (List<String>) UtilMap.getList((Map) optional.get(), "employeeField_lqxtykch_id");
  624. if (roleIds.size() == 1) {
  625. String tbUserId = _convertUserId(roleIds.get(0), false);
  626. if (StringUtils.isNotBlank(tbUserId)) {
  627. tbClient.updateTaskExecutor(UtilMap.getString(task, "id"), tbConf.getOperatorId(), tbUserId, false, false);
  628. }
  629. }
  630. }
  631. return true;
  632. });
  633. }
  634. /**
  635. * 项目迁移: 删除依赖项
  636. */
  637. @Synchronized
  638. @Override
  639. public void removeDependencies(String projectId, List<String> trNode) {
  640. // 任务编码字段ID
  641. List<Map> customField = tbClient.queryProjectCustomField(projectId, null);
  642. tbClient.queryProjectTaskList(projectId, null, taskList -> {
  643. for (Map task : taskList) {
  644. Optional optional = customField.stream().filter(item -> AWServer.TASK_STAGE.equals(item.get("name"))).findAny();
  645. if (!optional.isPresent()) {
  646. continue;
  647. }
  648. List<Map> customfields = (List<Map>) task.get("customfields");
  649. String trName = TBConf.getTaskFieldValue_First(customfields, UtilMap.getString((Map) optional.get(), "id"));
  650. // prd 传递无TR评审节点值, 匹配空值, 作为删除条件
  651. if (trNode.contains(trName) || (StringUtils.isBlank(trName) && trNode.contains(AWServer.TASK_STAGE_BLANK))) {
  652. String taskId = UtilMap.getString(task, "id");
  653. List<Map> dependencies = tbClient.queryTaskDependency(taskId, null);
  654. log.info("taskID = {}, trNode, {}, 依赖数量: {}", taskId, trName, dependencies.size());
  655. for (Map dependency : dependencies) {
  656. tbClient.removeTaskDependency(UtilMap.getString(dependency, "id"), tbConf.getOperatorId());
  657. }
  658. }
  659. }
  660. return true;
  661. });
  662. }
  663. // TB与宜搭userId转换 fixme 提取 tbService
  664. private String _convertUserId(String userId, boolean isTBID) {
  665. List<Map> tbMap = tbClient.idMapQuery(userId, isTBID);
  666. // 过滤未匹配人员信息 [ppExt: TB人员未匹配, 不执行修改也无报错]
  667. if (tbMap.isEmpty()) {
  668. log.warn("TB与宜搭userId转换为空, userId = {}, tbID = {}", userId, isTBID);
  669. return "";
  670. }
  671. if (!isTBID) {
  672. return UtilMap.getString(tbMap.get(0), "tbId");
  673. } else {
  674. Map extra = UtilMap.getMap(tbMap.get(0), "extra");
  675. return UtilMap.getString(extra, "userId");
  676. }
  677. }
  678. // TB与宜搭userId转换 fixme 提取 tbService
  679. private List<String> _convertUserId(List<String> userIds, boolean isTBID) {
  680. return userIds.stream().distinct()
  681. .map(id -> _convertUserId(id, isTBID))
  682. .filter(id -> StringUtils.isNotBlank(id))
  683. .collect(Collectors.toList());
  684. }
  685. /**
  686. * 同步crm基线
  687. */
  688. @Override
  689. public void syncBaseLineForCRM(String projectId) {
  690. String C_TR5_03_13 = "TR5-03-13";
  691. String C_TR6_01_11 = "TR6-01-11";
  692. // prd 推CRM的逻辑, 完成时间晚于2024/3/29的就更新,否则不更新
  693. Date dlDate = UtilDateTime.parseDateTime("2024-03-29 23:59:59");
  694. // 7. 产品型号 (Name) : 以上计划时间对应的产品型号 / 取产品型号字段
  695. List<Map> customField = tbClient.queryProjectCustomField(projectId, null);
  696. // 基线同步需求
  697. List<Map> baseLines = tbClient.queryGanttBaselineList(projectId, null).stream()
  698. .filter(item -> UtilMap.getString(item, "title").toUpperCase().contains("PDCP")).collect(Collectors.toList());
  699. log.info("PDCP基线, {}, {}", projectId, baseLines.size());
  700. List<Map> baseLineTaskList = new ArrayList();
  701. if (baseLines.size() > 0) {
  702. // prd 若存在多个同名, 取最新基线 [返回值创建时间升序]
  703. tbClient.queryGanttBaselineTask(UtilMap.getString(baseLines.get(baseLines.size() - 1), "id"), null, taskList -> {
  704. baseLineTaskList.addAll(taskList);
  705. return true;
  706. });
  707. }
  708. log.info("基线任务, {}", baseLineTaskList.size());
  709. // 最新同步需求
  710. Map<String, String> proData = getProjectCFID(projectId, Arrays.asList(AWServer.TASK_CODE, AWServer.TASK_PRODUCT));
  711. String tql = "cf:" + proData.get(AWServer.TASK_CODE) + " = " + C_TR5_03_13 + " AND (cf:" + proData.get(AWServer.TASK_PRODUCT) + " != null)" + " ORDER BY startDate DESC";
  712. List<Map> taskList = tbClient.queryProjectTaskList(projectId, UtilMap.map("q", tql), null);
  713. log.info("项目计划: {}, {}", C_TR5_03_13, taskList.size());
  714. List<Map> dataList = new ArrayList<>();
  715. for (Map task : taskList) {
  716. String name = _getTaskFieldMap(task, customField, AWServer.TASK_PRODUCT).get(AWServer.TASK_PRODUCT);
  717. // 多料号兼容
  718. if (dataList.stream().filter(item -> name.equals(item.get("Name"))).findAny().isPresent()) {
  719. continue;
  720. }
  721. Map tData = UtilMap.map("Name", name);
  722. // 2. 预计α推广 (AlphaDate): 最新项目计划(非基线)里的ADCPα任务对应的计划完成时间 / TR5-03-13任务号 取最新版本
  723. String date = UtilMap.getString(task, "dueDate");
  724. if (UtilString.isNotBlankCompatNull(date)) {
  725. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  726. if (tDate.after(dlDate)) {
  727. tData.put("AlphaDate", UtilDateTime.formatDate(tDate));
  728. }
  729. }
  730. // 3. α推广 (AlphaDate2): 【ADCPα】对应的实际完成时间 / TR5-03-13任务号 取最新版本
  731. date = UtilMap.getString(task, "accomplishTime");
  732. if (UtilString.isNotBlankCompatNull(date)) {
  733. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  734. if (tDate.after(dlDate)) {
  735. tData.put("AlphaDate2", UtilDateTime.formatDate(tDate));
  736. }
  737. }
  738. // 1. α推广基线 (AlphaBaseline): PDCP basseline 基线下的ADCPα任务的计划完成时间 / PDCP basseline 基线 TR5-03-13任务号
  739. Optional optional = baseLineTaskList.stream().filter(item -> task.get("id").equals(item.get("id"))).findAny();
  740. if (optional.isPresent()) {
  741. date = UtilMap.getString((Map) optional.get(), "planDueDate");
  742. if (StringUtils.isNotBlank(date)) {
  743. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  744. if (tDate.after(dlDate)) {
  745. tData.put("AlphaBaseline", UtilDateTime.formatDate(tDate));
  746. }
  747. }
  748. }
  749. if (tData.keySet().size() >= 2) {
  750. dataList.add(tData);
  751. _pushCRM(projectId, tData);
  752. }
  753. }
  754. tql = "cf:" + proData.get(AWServer.TASK_CODE) + " = " + C_TR6_01_11 + " AND (cf:" + proData.get(AWServer.TASK_PRODUCT) + " != null)" + " ORDER BY startDate ASC";
  755. taskList = tbClient.queryProjectTaskList(projectId, UtilMap.map("q", tql), null);
  756. log.info("项目计划: {}, {}", C_TR6_01_11, taskList.size());
  757. dataList = new ArrayList<>();
  758. for (Map task : taskList) {
  759. String name = _getTaskFieldMap(task, customField, AWServer.TASK_PRODUCT).get(AWServer.TASK_PRODUCT);
  760. // 多料号兼容
  761. if (dataList.stream().filter(item -> name.equals(item.get("Name"))).findAny().isPresent()) {
  762. continue;
  763. }
  764. Map tData = UtilMap.map("Name", name);
  765. // 5. 预计正式发布 (AgentSampleDate): 最新项目计划(非基线),取最早那个产品版本的计划完成时间 / TR6-01-11任务号 最早版本
  766. String date = UtilMap.getString(task, "dueDate");
  767. if (UtilString.isNotBlankCompatNull(date)) {
  768. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  769. if (tDate.after(dlDate)) {
  770. tData.put("AgentSampleDate", UtilDateTime.formatDate(tDate));
  771. }
  772. }
  773. // 6. 正式发布 (ProductSendDate): 【ADCP】对应的取最早那个产品版本的实际完成时间 / TR6-01-11任务号 最早版本
  774. date = UtilMap.getString(task, "accomplishTime");
  775. if (UtilString.isNotBlankCompatNull(date)) {
  776. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  777. if (tDate.after(dlDate)) {
  778. tData.put("ProductSendDate", UtilDateTime.formatDate(tDate));
  779. }
  780. }
  781. // 4. 正式发布基线 (ReleaseBaseline): 取PDCPbasseline基线里的ADCP的计划完成时间,取最早那个产品版本的计划完成时间 / PDCP basseline 基线 TR6-01-11任务号
  782. Optional optional = baseLineTaskList.stream().filter(item -> task.get("id").equals(item.get("id"))).findAny();
  783. if (optional.isPresent()) {
  784. date = UtilMap.getString((Map) optional.get(), "planDueDate");
  785. if (StringUtils.isNotBlank(date)) {
  786. Date tDate = UtilDateTime.parse(date, UtilDateTime.DATE_MSEL_ISO);
  787. if (tDate.after(dlDate)) {
  788. tData.put("ReleaseBaseline", UtilDateTime.formatDate(tDate));
  789. }
  790. }
  791. }
  792. if (tData.keySet().size() >= 2) {
  793. dataList.add(tData);
  794. _pushCRM(projectId, tData);
  795. }
  796. }
  797. }
  798. // 推送crm: prd 单次推送必须为相同料号
  799. void _pushCRM(String projectId, Map record) {
  800. log.info("推送crm, {}", record);
  801. //Map param = UtilMap.map("username, password, client_id, client_secret, grant_type", "interface@awinic.com.cn.uat", "welcome12", "3MVG959Nd8JMmavT2IGqAtf_hIbxepsElGbOpno6AO8KdQJSpSNqY9bnLRU2exuAEh3qIXb1oTn98S9h0WWZk", "3C9FC7427866D69586964F65A16D288EBBB1544335DD9FF3A03DE21DF14A7C6A", "password");
  802. Map param = UtilMap.map("username, password, client_id, client_secret, grant_type", "interface@awinic.com.cn", "welcome12", "3MVG9d8..z.hDcPKSaWop76C8GRQarYhn0LpBnP_U2S3VJaT8a6l05e2pDkdk5oE4MInQYwyLMpqWM7bLDzqm", "26A90098BDBE8C356817938986472730BAE5D9536DA9096AC7C45A4973867100", "password");
  803. //String rsp = UtilHttp.doPost("https://test.salesforce.com/services/oauth2/token", null, param, UtilMap.empty());
  804. String rsp = UtilHttp.doPost("https://login.salesforce.com/services/oauth2/token", null, param, UtilMap.empty());
  805. Map result = (Map) JSON.parse(rsp);
  806. Map header = UtilMap.map("Authorization", "OAuth " + result.get("access_token"));
  807. rsp = UtilHttp.doPost(UtilMap.getString(result, "instance_url") + "/services/apexrest/TBProductStageTimeRest", header, null, Arrays.asList(record), null);
  808. ydClient.operateData(YDParam.builder()
  809. .formUuid(_matchFormUuid("CRM_LOG"))
  810. .formDataJson(JSON.toJSONString(UtilMap.map("textareaField_lu8kinep, textareaField_lu8kineu, textField_lu8kinew", Arrays.asList(record), rsp, projectId)))
  811. .build(), YDConf.FORM_OPERATION.create);
  812. }
  813. /**
  814. * 全量同步crm基线
  815. */
  816. @Override
  817. public void syncBaseLineForCRM() {
  818. // prd 同步CRM: 项目状态为执行 & TB项目号不为空
  819. YDParam ydParam = YDParam.builder()
  820. .formUuid(_matchFormUuid("PROJECT"))
  821. .searchFieldJson(JSON.toJSONString(UtilMap.map("textField_llouhiyf", "执行")))
  822. .build();
  823. List<Map> proList = ydService.queryFormData_all(ydParam);
  824. for (Map pro : proList) {
  825. String projectId = UtilMap.getString(pro, "textField_lqxtykce");
  826. if (StringUtils.isBlank(projectId)) {
  827. continue;
  828. }
  829. syncBaseLineForCRM(projectId);
  830. }
  831. }
  832. // 获取项目自定义字段ID, 格式 { 名称: id } fixme 提取
  833. Map<String, String> getProjectCFID(String projectId, List<String> names) {
  834. List<Map> customField = tbClient.queryProjectCustomField(projectId, null);
  835. List tList = customField.stream().filter(item -> names.contains(item.get("name"))).collect(Collectors.toList());
  836. return (Map<String, String>) tList.stream().reduce(UtilMap.empty(), (acc, cur) -> {
  837. ((Map) acc).put(((Map) cur).get("name"), ((Map) cur).get("id"));
  838. return acc;
  839. });
  840. }
  841. /**
  842. * 修改任务自定义字段内容
  843. *
  844. * @param projectId 1. 若为空, 触发全量修改; 2. 仅修改非未完成任务
  845. */
  846. @Override
  847. public void batchUpdate(String fieldName, String preName, String modifyName, String projectId) {
  848. if (StringUtils.isBlank(projectId)) {
  849. // prd 同步修改: 项目状态为执行 & TB项目号不为空
  850. YDParam ydParam = YDParam.builder()
  851. .formUuid(_matchFormUuid("PROJECT"))
  852. .searchFieldJson(JSON.toJSONString(UtilMap.map("textField_llouhiyf", "执行")))
  853. .build();
  854. List<Map> proList = ydService.queryFormData_all(ydParam);
  855. for (Map pro : proList) {
  856. String _projectId = UtilMap.getString(pro, "textField_lqxtykce");
  857. if (StringUtils.isBlank(_projectId)) {
  858. continue;
  859. }
  860. _batchUpdate(fieldName, preName, modifyName, _projectId);
  861. }
  862. } else {
  863. _batchUpdate(fieldName, preName, modifyName, projectId);
  864. }
  865. }
  866. /// 修改任务自定义字段
  867. void _batchUpdate(String fieldName, String preName, String modifyName, String projectId) {
  868. // prd 非未完成任务字段ID
  869. List<String> workFlowStatusList = _getWorkFlowStatusList(projectId, AWServer.WORKFLOW_INITIAL);
  870. Map<String, String> proData = getProjectCFID(projectId, Arrays.asList(fieldName));
  871. String tql = "cf:" + proData.get(fieldName) + " = " + preName + " AND tfsId IN (" + String.join(", ", workFlowStatusList) + ")";
  872. List<Map> taskList = tbClient.queryProjectTaskList(projectId, UtilMap.put(UtilMap.empty(), "q", tql), null);
  873. for (Map task : taskList) {
  874. Map body = TBConf.assembleCustomFieldName(fieldName, modifyName);
  875. tbClient.updateTaskCustomField(UtilMap.getString(task, "id"), tbConf.getOperatorId(), body);
  876. }
  877. }
  878. /**
  879. * 同步预检项 [全量同步, 忽略任务号为空记录]
  880. */
  881. @Synchronized
  882. @Override
  883. public void syncCheckList(YDParam srcParam, Map<String, ?> compIds, String taskCompId, String codeCompId, String checkType, String associationCompId) {
  884. String curFormUuid = _matchFormUuid("CHECK");
  885. // 数据来源表
  886. List<Map> srcList = ydService.queryFormData_all(srcParam);
  887. // 当前记录表
  888. List<Map> curList = ydService.queryFormData_all(YDParam.builder()
  889. .formUuid(curFormUuid)
  890. .searchFieldJson(JSON.toJSONString(UtilMap.map("selectField_lrnd3h0s", checkType)))
  891. .build());
  892. log.info("全量同步 ##, srcList = {}, curList = {}", srcList.size(), curList.size());
  893. for (Map item : srcList) {
  894. if (StringUtils.isBlank(UtilMap.getString(item, taskCompId))) {
  895. continue;
  896. }
  897. Map formData = UtilMap.map(associationCompId, YDConf.associationForm(srcParam.getAppType(), srcParam.getFormUuid(), UtilMap.getString(item, "formInstanceId"), UtilMap.getString(item, codeCompId), "", false));
  898. formData.put("radioField_lrnddfq6", "启用");
  899. formData.put("selectField_lrnd3h0s", checkType);
  900. for (String compId : compIds.keySet()) {
  901. formData.put(compId, item.get(compIds.get(compId)));
  902. }
  903. YDConf.FORM_OPERATION operate = YDConf.FORM_OPERATION.create;
  904. YDParam ydParam = YDParam.builder()
  905. .formUuid(curFormUuid)
  906. .formDataJson(JSON.toJSONString(formData))
  907. .updateFormDataJson(JSON.toJSONString(formData))
  908. .useLatestVersion(true)
  909. .build();
  910. // 避免循坏内查询, 也避免了模糊匹配
  911. Optional optional = curList.stream().filter(cur -> cur.get("textField_lrnd3h0r").equals(item.get(codeCompId))).findAny();
  912. if (optional.isPresent()) {
  913. ydParam.setFormInstanceId(UtilMap.getString((Map) optional.get(), "formInstanceId"));
  914. operate = YDConf.FORM_OPERATION.update;
  915. // prd 0402 兼容删除场景, 标记状态为停用, 不做实际删除
  916. ((Map) optional.get()).put("status", true);
  917. }
  918. ydClient.operateData(ydParam, operate);
  919. }
  920. List<Map> dataList = curList.stream().filter(item -> "启用".equals(item.get("radioField_lrnddfq6")) && !item.containsKey("status")).collect(Collectors.toList());
  921. log.info(" 兼容删除场景, 标记状态为停用, {}", dataList.size());
  922. dataList.forEach(item -> {
  923. ydClient.operateData(YDParam.builder()
  924. .formInstanceId(UtilMap.getString(item, "formInstanceId"))
  925. .updateFormDataJson(JSON.toJSONString(UtilMap.map("radioField_lrnddfq6", "停用")))
  926. .build(), YDConf.FORM_OPERATION.update);
  927. });
  928. }
  929. /**
  930. * 同步预检项 [通用]
  931. */
  932. @Synchronized
  933. @Override
  934. public void syncCheckList(int type) {
  935. log.info("## 同步预检项, {}", type);
  936. switch (type) {
  937. case 0:
  938. // 经验库
  939. YDParam ydParam1 = YDParam.builder()
  940. .appType("APP_D4NPOH9J3KCG5VE1RVN6")
  941. .systemToken("6J8668A1B59FA7SAFTAF89PTKKGP2HY3FP3OLV1")
  942. .formUuid("FORM-D47CCA076DA54F969DF7442D81E4B5A8PC6Q")
  943. .build();
  944. // 任务号, 描述, ID(标题), 任务名称, 适用项目类型
  945. Map<String, ?> compIds1 = UtilMap.map("selectField_lrncf4hk, textField_lrrnqz7w, textField_lrnd3h0r, textField_ltzl9mpc, multiSelectField_ltwjre9s", "selectField_ls9yctsd, textareaField_lo5eakcv, serialNumberField_lo5dmyfb, textField_ls9xvxvk, multiSelectField_ltwjre9s");
  946. syncCheckList(ydParam1, compIds1, "selectField_ls9yctsd", "serialNumberField_lo5dmyfb", "经验库", "associationFormField_lrrn5csf");
  947. break;
  948. case 1:
  949. // IC技术检查表
  950. YDParam ydParam2 = YDParam.builder()
  951. .appType("APP_NOSSCZQ8FFZKHDQOIIDA")
  952. .systemToken("UM6660D1BFTFLFAP7NPJDBJEEAB737FAR1SOLD4")
  953. .formUuid("FORM-B40420AFCDB84AD69FE78D82D6D5CFD2C2D2")
  954. .build();
  955. // 任务号, 描述, ID(标题), 任务名称, 适用项目类型
  956. Map<String, ?> compIds2 = UtilMap.map("selectField_lrncf4hk, textField_lrrnqz7w, textField_lrnd3h0r, textField_ltzl9mpc, multiSelectField_ltwjre9s", "selectField_bclg9y5, textField_k1e08ji, textField_96ikaoh, textField_lsa0d856, multiSelectField_ltwjre9s");
  957. syncCheckList(ydParam2, compIds2, "selectField_bclg9y5", "textField_96ikaoh", "IC技术检查表", "associationFormField_lrrn5csg");
  958. break;
  959. case 2:
  960. // TR评审要素表
  961. YDParam ydParam3 = YDParam.builder()
  962. .formUuid("FORM-5436DC75BC8347D498DFF2617925BA70O066")
  963. .build();
  964. // 任务号, 描述, ID(标题), 验收标准, 等级, 任务名称, 适用项目类型
  965. Map<String, ?> compIds3 = UtilMap.map("selectField_lrncf4hk, textField_lrrnqz7w, textField_lrnd3h0r, textField_cd4f65l, textField_267kho2, textField_ltzl9mpc, multiSelectField_ltwjre9s", "selectField_4gwjfa5, textField_lryetn3g, textField_9x1yps6, textField_cd4f65l, textField_267kho2, textField_ls9xvxvk, multiSelectField_ltwjre9s");
  966. syncCheckList(ydParam3, compIds3, "selectField_4gwjfa5", "textField_9x1yps6", "TR评审要素表", "associationFormField_lrrn5csa");
  967. break;
  968. }
  969. log.info("同步预检项 ##, {}", type);
  970. }
  971. /**
  972. * 提供verifier数据读取服务
  973. */
  974. @Override
  975. public List<Map> syncVerifier(String projectCode) {
  976. List<Map> dataList = ydService.queryFormData_all(YDParam.builder()
  977. .formUuid("FORM-23B67983E91C4ED3B063F9B629D6E964SGHH")
  978. .searchCondition(JSON.toJSONString(UtilMap.map("textField_lt6xd8lm", projectCode)))
  979. .build());
  980. return dataList.stream().map(item -> {
  981. return item;
  982. }).collect(Collectors.toList());
  983. }
  984. @Override
  985. public void test() {
  986. }
  987. /// tmp: 5.22 处理检查项重复数据
  988. private void _test() {
  989. List<Map> dataList = ydService.queryFormData_all(YDParam.builder()
  990. .formUuid("FORM-6E2C0D1197264B8AA23EB3FECAE7344B00BN")
  991. .searchFieldJson(JSON.toJSONString(UtilMap.map("selectField_lqxuswze", "A2328")))
  992. .build());
  993. dataList.forEach(item -> {
  994. List<Map> details = (List<Map>) item.get("tableField_lqxxgj4s");
  995. String formInstanceId = String.valueOf(item.get("formInstanceId"));
  996. if (details.size() == 50) {
  997. YDParam ydParam = YDParam.builder().formUuid("FORM-6E2C0D1197264B8AA23EB3FECAE7344B00BN").formInstanceId(formInstanceId).tableFieldId("tableField_lqxxgj4s").build();
  998. details = ydService.queryDetails(ydParam);
  999. }
  1000. if (!details.stream().filter(row -> "技术评审要素表".equals(row.get("selectField_lrnd3h0s"))).findAny().isPresent()) {
  1001. return;
  1002. }
  1003. details = details.stream().filter(row -> !"IC技术检查表".equals(row.get("selectField_lrnd3h0s"))).collect(Collectors.toList());
  1004. details.forEach(record -> {
  1005. // String tmp = ((List<Map<String, String>>) JSON.parse(String.valueOf(JSON.parse(String.valueOf(record.get("associationFormField_lrrnem5r_id")))))).get(0).get("title");
  1006. // if (tmp.startsWith("TR")) {
  1007. // record.put("selectField_lrnd3h0s", "TR评审要素表");
  1008. // } else if (tmp.contains("-")) {
  1009. // record.put("selectField_lrnd3h0s", "IC技术检查表");
  1010. // } else {
  1011. // record.put("selectField_lrnd3h0s", "经验库");
  1012. // }
  1013. if ("技术评审要素表".equals(record.get("selectField_lrnd3h0s"))) {
  1014. record.put("selectField_lrnd3h0s", "IC技术检查表");
  1015. }
  1016. record.put("associationFormField_lrrnem5r", JSON.parse(String.valueOf(record.get("associationFormField_lrrnem5r_id"))));
  1017. });
  1018. log.info("------, {}, {}, {}", details.size(), formInstanceId, details);
  1019. ydClient.operateData(YDParam.builder()
  1020. .formInstanceId(String.valueOf(item.get("formInstanceId")))
  1021. .updateFormDataJson(JSON.toJSONString(UtilMap.map("tableField_lqxxgj4s", details)))
  1022. .build(), YDConf.FORM_OPERATION.update);
  1023. });
  1024. }
  1025. }