AbstractAdmBaseServiceImpl.java 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511
  1. package com.persagy.proxy.adm.service.impl;
  2. import java.io.IOException;
  3. import java.lang.reflect.Field;
  4. import java.lang.reflect.ParameterizedType;
  5. import java.util.ArrayList;
  6. import java.util.HashMap;
  7. import java.util.List;
  8. import java.util.Map;
  9. import java.util.Set;
  10. import java.util.stream.Collectors;
  11. import cn.hutool.core.bean.BeanUtil;
  12. import org.springframework.beans.factory.annotation.Autowired;
  13. import com.alibaba.fastjson.JSONArray;
  14. import com.alibaba.fastjson.JSONObject;
  15. import com.fasterxml.jackson.databind.ObjectMapper;
  16. import com.fasterxml.jackson.databind.node.ArrayNode;
  17. import com.fasterxml.jackson.databind.node.JsonNodeFactory;
  18. import com.fasterxml.jackson.databind.node.ObjectNode;
  19. import com.persagy.dmp.basic.model.QueryCriteria;
  20. import com.persagy.dmp.common.constant.ValidEnum;
  21. import com.persagy.dmp.common.helper.SpringHelper;
  22. import com.persagy.dmp.common.model.entity.BaseEntity;
  23. import com.persagy.dmp.common.utils.JsonHelper;
  24. import com.persagy.dmp.digital.client.DigitalObjectFacade;
  25. import com.persagy.dmp.digital.entity.ObjectRelation;
  26. import com.persagy.proxy.adm.annotations.CascadeColumn;
  27. import com.persagy.proxy.adm.request.AdmQueryCriteria;
  28. import com.persagy.proxy.adm.request.AdmResponse;
  29. import com.persagy.proxy.adm.service.IAdmBaseService;
  30. import com.persagy.proxy.adm.service.IAdmRelationService;
  31. import com.persagy.proxy.adm.utils.AdmEntityTransferUtil;
  32. import com.persagy.proxy.adm.utils.AdmQueryCriteriaHelper;
  33. import com.persagy.proxy.common.client.DmpRwdClient;
  34. import com.persagy.proxy.common.entity.DmpResult;
  35. import com.persagy.proxy.common.entity.InstanceUrlParam;
  36. import cn.hutool.core.collection.CollUtil;
  37. import cn.hutool.core.map.MapUtil;
  38. import cn.hutool.core.util.ReflectUtil;
  39. import cn.hutool.core.util.StrUtil;
  40. import lombok.extern.slf4j.Slf4j;
  41. /**
  42. * ADM 抽象实现类
  43. * @author Charlie Yu
  44. * @date 2021-08-16
  45. */
  46. @Slf4j
  47. public class AbstractAdmBaseServiceImpl<T> implements IAdmBaseService<T> {
  48. @Autowired
  49. private DmpRwdClient rwdClient;
  50. @Autowired
  51. private ObjectMapper objectMapper;
  52. /**
  53. * 查询
  54. * @param request
  55. * @return
  56. */
  57. @Override
  58. public AdmResponse doQuery(InstanceUrlParam context, AdmQueryCriteria request, Class<T> clazz) {
  59. // 转换为数据中台查询条件
  60. QueryCriteria dmpRequest = AdmQueryCriteriaHelper.toDmpCriteria(request);
  61. // 提供一个钩子方法用于处理已封装好的中台查询参数(因传参有buildingId而处理)
  62. processDmpCriteria(dmpRequest);
  63. // 转换参数
  64. JSONObject para = null;
  65. try {
  66. para = JsonHelper.toJsonObject(dmpRequest);
  67. } catch (IOException e) {
  68. log.error(e.getMessage(), e);
  69. return AdmResponse.failure("参数格式有误");
  70. }
  71. log.info("request:{},dmpRequest:{}, dmp-param: {}", request.toString(), dmpRequest.toString(), para.toString());
  72. // 调用中台查询
  73. DmpResult<JSONArray> dmpResult = rwdClient.queryObject(context, para);
  74. processResultAfterQuery(dmpResult);
  75. List<T> admVOs = AdmEntityTransferUtil.toAdmMultiEntity(dmpResult.getData(), request.getProjection(), clazz);
  76. //List<T> admVOs = AdmEntityTransferUtil.toAdmMultiEntityExtra(dmpResult.getData(), request.getProjection(), clazz);
  77. // 级联查询
  78. processCascade(context, admVOs, request.getName(), request.getCascade());
  79. // 设置返回值
  80. AdmResponse response = AdmResponse.success(admVOs);
  81. Long total = dmpResult.getCount() == null ? null : dmpResult.getCount().longValue();
  82. if(request.isOnlyCount()){
  83. response.setTotal(null);
  84. response.setCount(total);
  85. }else {
  86. response.setTotal(total);
  87. }
  88. response.setPageNumber(request.getPageNumber());
  89. response.setPageSize(request.getPageSize());
  90. return response;
  91. }
  92. /***
  93. * Description: 提供一个钩子方法用于处理已封装好的中台查询参数(因传参有buildingId而处理)
  94. * @param dmpRequest : 中台查询参数
  95. * @return : void
  96. * @author : lijie
  97. * @date :2021/10/28 20:23
  98. * Update By lijie 2021/10/28 20:23
  99. */
  100. protected void processDmpCriteria(QueryCriteria dmpRequest) {
  101. // 空方法,需要时子类重写
  102. }
  103. /**
  104. * 处理级联查询
  105. * @param context
  106. * @param admVOs
  107. * @param name
  108. * @param cascades
  109. */
  110. public void processCascade(InstanceUrlParam context, List admVOs, String name, List<AdmQueryCriteria> cascades) {
  111. if(CollUtil.isEmpty(admVOs) || CollUtil.isEmpty(cascades)) {
  112. return;
  113. }
  114. Map<String, Object> voMap = CollUtil.fieldValueMap(admVOs, BaseEntity.PROP_ID);
  115. Object simpleVO = admVOs.get(0);
  116. for(AdmQueryCriteria cascade:cascades) {
  117. if(StrUtil.isBlank(cascade.getName())){
  118. //处理 级联查询参数name首字母大写cascade": [ {"Name": "property"}]
  119. continue;
  120. }
  121. // 获取关联对象的Class
  122. Field casField = ReflectUtil.getField(simpleVO.getClass(), cascade.getName());
  123. if(casField == null) {
  124. continue;
  125. }
  126. // 获取级联信息
  127. CascadeColumn annotationInfo = casField.getAnnotation(CascadeColumn.class);
  128. if(annotationInfo == null) {
  129. continue;
  130. }
  131. Class clazz = getFieldRealClass(casField);
  132. // 获取关联对象的类型
  133. String objectType = getObjTypeByClass(clazz);
  134. if(StrUtil.isBlank(objectType)) {
  135. continue;
  136. }
  137. cascade.setName(objectType);
  138. // 转换为数据中台查询条件
  139. QueryCriteria dmpRequest = AdmQueryCriteriaHelper.toDmpCriteria(cascade);
  140. // 拼接关系条件,得到关系映射
  141. Map<String, List<String>> idMap = ensureRelationCond(context, dmpRequest.getCriteria(), voMap.keySet(), annotationInfo);
  142. if(MapUtil.isEmpty(idMap)) {
  143. continue;
  144. }
  145. // 查询结果
  146. List vos = queryByCondition(context, dmpRequest, cascade.getProjection(), clazz);
  147. if(CollUtil.isEmpty(vos)) {
  148. continue;
  149. }
  150. // 写回原对象中
  151. writeToVOs(voMap, idMap, vos, casField);
  152. // 递归继续处理
  153. processCascade(context, vos, objectType, cascade.getCascade());
  154. }
  155. }
  156. /**
  157. * 拼接关系条件
  158. * @param context
  159. * @param node
  160. * @param idList
  161. * @param annotationInfo
  162. * @return
  163. */
  164. private Map<String, List<String>> ensureRelationCond(InstanceUrlParam context, ObjectNode node, Set<String> idList, CascadeColumn annotationInfo) {
  165. if(CollUtil.isEmpty(idList)) {
  166. return null;
  167. }
  168. // 查询关系
  169. List<ObjectRelation> relations = queryRelations(context, idList, annotationInfo);
  170. ArrayNode array = node.putObject(BaseEntity.PROP_ID).putArray("$in");
  171. // 拼接查询条件
  172. if(CollUtil.isEmpty(relations)) {
  173. array.add("0");
  174. } else {
  175. List<String> relList = CollUtil.getFieldValues(relations, annotationInfo.selectToObj()?"objTo":"objFrom", String.class);
  176. relList.forEach(id -> array.add(id));
  177. }
  178. if(annotationInfo.selectToObj()){
  179. return groupMapList(relations, "objFrom", "objTo");
  180. }else{
  181. return groupMapList(relations, "objTo", "objFrom");
  182. }
  183. }
  184. /**
  185. * 将集合转换为Map
  186. * @param list 集合
  187. * @param keyField key属性名
  188. * @param valueField value属性名
  189. * @return
  190. */
  191. private static Map<String, List<String>> groupMapList(List<?> list, String keyField, String valueField) {
  192. if(CollUtil.isEmpty(list)) {
  193. return null;
  194. }
  195. Map<String, List<String>> groupMap = new HashMap<>();
  196. for(Object vo:list) {
  197. String key = (String) ReflectUtil.getFieldValue(vo, keyField);
  198. String value = (String) ReflectUtil.getFieldValue(vo, valueField);
  199. if(!StrUtil.isAllNotBlank(key, value)) {
  200. continue;
  201. }
  202. List<String> valueList = groupMap.get(key);
  203. if(CollUtil.isEmpty(valueList)) {
  204. valueList = new ArrayList<>();
  205. groupMap.put(key, valueList);
  206. }
  207. valueList.add(value);
  208. }
  209. return groupMap;
  210. }
  211. /**
  212. * 查询关系
  213. * @param context
  214. * @param idList
  215. * @param anno
  216. * @return
  217. */
  218. private List<ObjectRelation> queryRelations(InstanceUrlParam context, Set<String>idList, CascadeColumn anno) {
  219. // 如果没有任何配置,不处理
  220. if(StrUtil.isBlank(anno.graphCode()) && StrUtil.isBlank(anno.relCode()) &&
  221. StrUtil.isBlank(anno.relValue())) {
  222. return null;
  223. }
  224. // 查询关系
  225. QueryCriteria queryRequest = new QueryCriteria();
  226. ObjectNode criteria = JsonNodeFactory.instance.objectNode();
  227. queryRequest.setCriteria(criteria);
  228. boolean isSelectTo = anno.selectToObj();
  229. criteria.put(BaseEntity.PROP_VALID, ValidEnum.TRUE.getType());
  230. putString(criteria, "graphCode", anno.graphCode());
  231. putString(criteria, "relCode", anno.relCode());
  232. putString(criteria, "relValue", anno.relValue());
  233. ArrayNode array = criteria.putObject(isSelectTo?"objFrom":"objTo").putArray("$in");
  234. idList.forEach(id -> array.add(id));
  235. return SpringHelper.getBean(IAdmRelationService.class).queryByCondition(context, queryRequest);
  236. }
  237. /**
  238. * 添加字符串条件
  239. * @param node
  240. * @param key
  241. * @param value
  242. */
  243. private void putString(ObjectNode node, String key, String value) {
  244. if(StrUtil.isBlank(value)) {
  245. return;
  246. }
  247. node.put(key, value);
  248. }
  249. /**
  250. * 将子集写回到对象中
  251. * @param voMap 原始对象Map
  252. * @param idMap 主键映射<mainId, List<subId>>
  253. * @param vos 子对象
  254. * @param casField 子对象对应属性
  255. */
  256. private <V> void writeToVOs(Map<String, Object> voMap, Map<String, List<String>> idMap, List<V> vos, Field casField) {
  257. if(CollUtil.isEmpty(vos) || MapUtil.isEmpty(idMap)) {
  258. return;
  259. }
  260. // 转换为Map
  261. Map<String, V> subVOMap = CollUtil.fieldValueMap(vos, BaseEntity.PROP_ID);
  262. // 是否为集合
  263. boolean isList = List.class.isAssignableFrom(casField.getType());
  264. // 按对象循环
  265. for(String id:voMap.keySet()) {
  266. Object mainVO = voMap.get(id);
  267. List<String> subIdList = idMap.get(id);
  268. if(CollUtil.isEmpty(subIdList)) {
  269. continue;
  270. }
  271. // 如果是集合
  272. if(isList) {
  273. // 取出对应的对象集合
  274. Map<String, V> currVOMap = MapUtil.filter(subVOMap, subIdList.toArray(new String[0]));
  275. // 2021年11月11日17:30:04,by lijie.解决级联查询已经排好序的列表错乱问题
  276. List<V> currList = vos.stream().filter(vo -> currVOMap.containsKey(BeanUtil.getFieldValue(vo, BaseEntity.PROP_ID)))
  277. .collect(Collectors.toList());
  278. // List<V> voList = CollUtil.newArrayList(currVOMap.values());
  279. ReflectUtil.setFieldValue(mainVO, casField, currList);
  280. } else {
  281. // 非集合取第一个即可
  282. String subId = subIdList.get(0);
  283. V vo = subVOMap.get(subId);
  284. ReflectUtil.setFieldValue(mainVO, casField, vo);
  285. }
  286. }
  287. }
  288. /**
  289. * 调用Client查询
  290. * @param context
  291. * @param criteria
  292. * @param projection
  293. * @param clazz
  294. * @return
  295. */
  296. private <V> List<V> queryByCondition(InstanceUrlParam context, QueryCriteria criteria, List<String> projection, Class<V> clazz) {
  297. // 转换参数
  298. if(criteria == null) {
  299. return null;
  300. }
  301. List<ObjectNode> objects = DigitalObjectFacade.query(context.getGroupCode(), context.getProjectId(),
  302. context.getAppId(), null, criteria);
  303. return AdmEntityTransferUtil.toAdmMultiEntity(objects, projection, clazz);
  304. }
  305. /**
  306. * 获取实际的Class
  307. * @param casField
  308. * @return
  309. */
  310. private Class getFieldRealClass(Field casField) {
  311. // 取到Field
  312. Class clazz = casField.getType();
  313. // 如果是集合,则取实际的对象
  314. if(List.class.isAssignableFrom(clazz)) {
  315. clazz = (Class) ((ParameterizedType) casField.getGenericType()).getActualTypeArguments()[0];
  316. }
  317. return clazz;
  318. }
  319. /**
  320. * 通过Class获取其对象类型
  321. * @param clazz
  322. * @return
  323. */
  324. private String getObjTypeByClass(Class clazz) {
  325. // 获取关联对象的类型
  326. String objectType = null;
  327. try {
  328. objectType = (String) ReflectUtil.getFieldValue(clazz.newInstance(), "objectType");
  329. } catch (Exception e) {
  330. log.error(e.getMessage(), e);
  331. }
  332. return objectType;
  333. }
  334. /**
  335. * 新增
  336. * @param context
  337. * @param clazz
  338. * @param voList
  339. * @return
  340. */
  341. @Override
  342. public List<T> doInsert(InstanceUrlParam context, Class<T> clazz, List<T> voList) {
  343. if(CollUtil.isEmpty(voList)) {
  344. return null;
  345. }
  346. ArrayNode vos = AdmEntityTransferUtil.toDmpMultiEntity(voList);
  347. // 调用中台新增
  348. DmpResult<JSONArray> response = rwdClient.createObject(context, JSONArray.parseArray(vos.toString()));
  349. if(!DmpResult.SUCCESS.equals(response.getResult())) {
  350. throw new RuntimeException("调用中台查询接口出错:" + response.getMessage());
  351. }
  352. // 转换为结果
  353. // 2021年11月3日17:45:46 by lijie 新增一个钩子方法,用于处理floor的properties字段,不然转换为对象时会报转换异常
  354. processResultAfterQuery(response);
  355. return AdmEntityTransferUtil.toAdmMultiEntity(response.getData(), null, clazz);
  356. }
  357. /***
  358. * Description: 一个钩子方法,特殊处理一下楼层对象返回的properties信息点,不然转换为对象时会报转换异常
  359. * @param response : 查询结果
  360. * @return : void
  361. * @author : lijie
  362. * @date :2021/11/3 17:47
  363. * Update By lijie 2021/11/3 17:47
  364. */
  365. protected void processResultAfterQuery(DmpResult<JSONArray> response) {
  366. }
  367. /**
  368. * 修改
  369. * @param context
  370. * @param clazz
  371. * @param voList
  372. * @return
  373. */
  374. @Override
  375. public List<T> doUpdate(InstanceUrlParam context, Class<T> clazz, List<T> voList) {
  376. if(CollUtil.isEmpty(voList)) {
  377. return null;
  378. }
  379. ArrayNode vos = AdmEntityTransferUtil.toDmpMultiEntity(voList);
  380. // 调用中台修改
  381. DmpResult<JSONArray> response = rwdClient.updateObject(context, JSONArray.parseArray(vos.toString()));
  382. if(!DmpResult.SUCCESS.equals(response.getResult())) {
  383. throw new RuntimeException("调用中台查询接口出错:" + response.getMessage());
  384. }
  385. // 转换为结果
  386. processResultAfterQuery(response);
  387. return AdmEntityTransferUtil.toAdmMultiEntity(response.getData(), null, clazz);
  388. }
  389. /**
  390. * 删除
  391. * @param context
  392. * @param voList
  393. * @return
  394. */
  395. @Override
  396. public void doDelete(InstanceUrlParam context, List<T> voList) {
  397. if(CollUtil.isEmpty(voList)) {
  398. return;
  399. }
  400. List<String> idList = CollUtil.getFieldValues(voList, "id", String.class);
  401. // 调用中台修改
  402. DmpResult response = rwdClient.deleteObject(context, idList);
  403. if(!DmpResult.SUCCESS.equals(response.getResult())) {
  404. throw new RuntimeException("调用中台查询接口出错:" + response.getMessage());
  405. }
  406. }
  407. /**
  408. * 根据物理世界查询条件查询对象信息
  409. * @param context
  410. * @param dmpRequest
  411. * @param clazz
  412. * @return
  413. */
  414. @Override
  415. public AdmResponse doQuery(InstanceUrlParam context, QueryCriteria dmpRequest, Class<T> clazz) {
  416. // 转换参数
  417. JSONObject para = null;
  418. try {
  419. para = JsonHelper.toJsonObject(dmpRequest);
  420. } catch (IOException e) {
  421. log.error(e.getMessage(), e);
  422. return null;
  423. }
  424. // 调用中台查询
  425. DmpResult<JSONArray> dmpResult = rwdClient.queryObject(context, para);
  426. processResultAfterQuery(dmpResult);
  427. List<T> admVOs = AdmEntityTransferUtil.toAdmMultiEntity(dmpResult.getData(), null, clazz);
  428. // JSONArray data = dmpResult.getData();
  429. // List<T> admVOs = new ArrayList<>();
  430. // if(CollUtil.isNotEmpty(data)){
  431. // admVOs = data.toJavaList(clazz);
  432. // }
  433. // 设置返回值
  434. AdmResponse response = AdmResponse.success(admVOs);
  435. Long total = dmpResult.getCount() == null ? null : dmpResult.getCount().longValue();
  436. response.setTotal(total);
  437. return response;
  438. }
  439. /**
  440. * 查询标记计算关系
  441. * @param context
  442. * @param dmpRequest
  443. * @param clazz
  444. * @return
  445. */
  446. @Override
  447. public AdmResponse doQueryRelationProjectCal(InstanceUrlParam context, QueryCriteria dmpRequest, Class<T> clazz) {
  448. // 转换参数
  449. JSONObject para = null;
  450. try {
  451. para = JsonHelper.toJsonObject(dmpRequest);
  452. } catch (IOException e) {
  453. log.error(e.getMessage(), e);
  454. return null;
  455. }
  456. // 调用中台查询
  457. DmpResult<JSONArray> dmpResult = rwdClient.queryRelationProjectCal(context, para);
  458. List<T> admVOs = AdmEntityTransferUtil.toAdmMultiEntity(dmpResult.getData(), null, clazz);
  459. // 设置返回值
  460. AdmResponse response = AdmResponse.success(admVOs);
  461. Long total = dmpResult.getCount() == null ? null : dmpResult.getCount().longValue();
  462. response.setTotal(total);
  463. return response;
  464. }
  465. /**
  466. * 修改
  467. * @param context
  468. * @param clazz
  469. * @param voList
  470. * @return
  471. */
  472. @Override
  473. public List<T> doUpdateRelationProjectCal(InstanceUrlParam context, Class<T> clazz, List<T> voList) {
  474. if(CollUtil.isEmpty(voList)) {
  475. return null;
  476. }
  477. ArrayNode vos = AdmEntityTransferUtil.toDmpMultiEntity(voList);
  478. // 调用中台修改
  479. DmpResult<JSONArray> response = rwdClient.updateRelationProjectCal(context, JSONArray.parseArray(vos.toString()));
  480. if(!DmpResult.SUCCESS.equals(response.getResult())) {
  481. throw new RuntimeException("调用中台查询接口出错:" + response.getMessage());
  482. }
  483. // 转换为结果
  484. return AdmEntityTransferUtil.toAdmMultiEntity(response.getData(), null, clazz);
  485. }
  486. }