HuaGaoServiceImpl.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. package com.malk.huagao.service.impl;
  2. import cn.hutool.core.date.DateUtil;
  3. import cn.hutool.core.util.NumberUtil;
  4. import com.alibaba.fastjson.JSON;
  5. import com.alibaba.fastjson.JSONObject;
  6. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  7. import com.malk.huagao.entity.YdNonProdPrice;
  8. import com.malk.huagao.entity.YdProdWorkHoursPrice;
  9. import com.malk.huagao.entity.YdUserInput;
  10. import com.malk.huagao.mapper.YdNonProdPriceMapper;
  11. import com.malk.huagao.mapper.YdProdWorkHoursPriceMapper;
  12. import com.malk.huagao.mapper.YdUserInputMapper;
  13. import com.malk.huagao.service.HuaGaoService;
  14. import com.malk.server.aliwork.YDConf;
  15. import com.malk.server.aliwork.YDParam;
  16. import com.malk.server.common.McException;
  17. import com.malk.server.dingtalk.DDR;
  18. import com.malk.server.dingtalk.DDR_New;
  19. import com.malk.service.aliwork.YDClient;
  20. import com.malk.service.aliwork.YDService;
  21. import com.malk.service.dingtalk.DDClient;
  22. import com.malk.service.dingtalk.DDClient_Attendance;
  23. import com.malk.service.dingtalk.DDClient_Contacts;
  24. import com.malk.service.fxiaoke.FXKClient;
  25. import com.malk.utils.UtilDateTime;
  26. import com.malk.utils.UtilMap;
  27. import lombok.extern.slf4j.Slf4j;
  28. import org.springframework.beans.factory.annotation.Autowired;
  29. import org.springframework.beans.factory.annotation.Value;
  30. import org.springframework.stereotype.Service;
  31. import java.text.ParseException;
  32. import java.text.SimpleDateFormat;
  33. import java.time.LocalDateTime;
  34. import java.util.*;
  35. @Slf4j
  36. @Service
  37. public class HuaGaoServiceImpl implements HuaGaoService {
  38. @Autowired
  39. private YDService ydService;
  40. @Autowired
  41. private YDClient ydClient;
  42. @Value("${dingtalk.appKey}")
  43. private String appKey;
  44. @Value("${dingtalk.appSecret}")
  45. private String appSecret;
  46. @Autowired
  47. private DDClient_Contacts ddClient_contacts;
  48. @Autowired
  49. private DDClient_Attendance ddClientAttendance;
  50. @Autowired
  51. private DDClient ddClient;
  52. @Autowired
  53. private FXKClient fxkClient;
  54. @Autowired
  55. private YdUserInputMapper ydUserInputMapper;
  56. @Autowired
  57. private YdNonProdPriceMapper ydNonProdPriceMapper;
  58. @Autowired
  59. private YdProdWorkHoursPriceMapper ydProdWorkHoursPriceMapper;
  60. @Autowired
  61. private static final long DEPT_ID = 971482089L; // 合作加盟校区
  62. /// 创建企业账号: loginI/userId 都为 SHR 工号
  63. private Map crcreateUser_dingTalk(String name, String randomAccount, String randomPassword) {
  64. Map param = new HashMap();
  65. param.put("access_token", ddClient.getAccessToken());
  66. Map body = new HashMap();
  67. body.put("exclusive_account", "true");
  68. body.put("exclusive_account_type", "dingtalk");
  69. body.put("login_id", randomAccount);
  70. body.put("init_password", randomPassword);
  71. body.put("name", name);
  72. body.put("userid", randomAccount);
  73. body.put("dept_id_list", DEPT_ID);
  74. Object result = DDR.doPost("https://oapi.dingtalk.com/topapi/v2/user/create", (Map) null, param, body).getResult();
  75. return (Map) result;
  76. }
  77. /**
  78. * 同步钉钉企业账号
  79. */
  80. @Override
  81. public void syncDingTalk_exclusive() {
  82. List<Map> dataList = ydService.queryFormData_all(YDParam.builder()
  83. .formUuid("FORM-0086C4597047459BBB37C79E404745C9CK0C")
  84. .build());
  85. log.info("钉钉企业账号同步开始,共{}条数据", dataList.size());
  86. dataList.forEach(data -> {
  87. if ("".equals(UtilMap.getString(data, "textField_m1mw64v4"))) {
  88. String message = "";
  89. String randomAccount = generateRandomAccount();
  90. String randomPassword = generateRandomSixDigitPassword();
  91. try {
  92. Map textField_m21lkesk = this.crcreateUser_dingTalk(UtilMap.getString(data, "textField_m21lkesk"), randomAccount, randomPassword);
  93. System.out.println("");
  94. } catch (McException e) {
  95. message = e.getMessage();
  96. log.error(e.getMessage(), e);
  97. }
  98. ydClient.operateData(YDParam.builder()
  99. .formInstanceId(UtilMap.getString(data, "instanceId"))
  100. .useLatestVersion(true)
  101. .updateFormDataJson(JSON.toJSONString(UtilMap.map("textField_m1mw64v4, textField_m1mw64v5", randomAccount, randomPassword)))
  102. .build(), YDConf.FORM_OPERATION.update);
  103. }
  104. });
  105. }
  106. /**
  107. * 重置企业账号密码
  108. */
  109. @Override
  110. public void resetPwd(Map data) {
  111. ddClient_contacts.updateUser_dingTalk(ddClient.getAccessToken(), UtilMap.getString(data, "userId"), Arrays.asList(DEPT_ID), UtilMap.map("init_password", UtilMap.getString(data, "password")));
  112. }
  113. SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  114. @Override
  115. public void getCheckWorkDateDetails(Map data) {
  116. List<String> users = new ArrayList<>();
  117. users.add("284156461536271475");
  118. String[] strings = users.toArray(new String[0]);
  119. log.info("users:{}", JSON.toJSONString(strings));
  120. List<Map> mapList = ddClientAttendance.getAttColumnVal(ddClient.getAccessToken(appKey, appSecret),"284156461536271475", Arrays.asList("82772106,89349019"),"2025-01-20 08:00:00", "2025-01-21 08:00:00");
  121. System.out.println("====="+mapList);
  122. // mapList.get(0)
  123. String FORMtime = "2025-01-01 00:00:00";
  124. String TOtime = "2025-01-31 00:00:00";
  125. // dateFormat.format(FORMtime);
  126. // dateFormat.format(TOtime);
  127. //工作时长(实际工时)
  128. // Step 1. 提取原始数据到日期-字段ID-值的映射
  129. Map<String, Map<Long, Double>> dateData = new HashMap<>();
  130. for (Map<String, Object> element : mapList) {
  131. // 获取字段ID
  132. Map<String, Object> columnVo = (Map<String, Object>) element.get("column_vo");
  133. Long id = ((Number) columnVo.get("id")).longValue();
  134. // 遍历该字段的所有日期数据
  135. List<Map<String, String>> columnVals = (List<Map<String, String>>) element.get("column_vals");
  136. for (Map<String, String> entry : columnVals) {
  137. String fullDate = entry.get("date");
  138. // System.out.println("fullDate:====="+fullDate);
  139. String date = fullDate.split(" ")[0]; // 提取日期部分
  140. Double value = Double.parseDouble(entry.get("value"));
  141. // 存储到日期映射中
  142. dateData.computeIfAbsent(date, k -> new HashMap<>())
  143. .put(id, value);
  144. }
  145. }
  146. // Step 2. 按日期处理业务逻辑
  147. Map<String, Double> finalResult = new LinkedHashMap<>(); // 保持日期顺序
  148. for (Map.Entry<String, Map<Long, Double>> entry : dateData.entrySet()) {
  149. String date = entry.getKey();
  150. Map<Long, Double> idValues = entry.getValue();
  151. // 获取目标字段的值
  152. Double targetValue = idValues.get(89349019L); // ID=89349019
  153. Double refValue = idValues.get(82772106L); // ID=82772106
  154. // 逻辑判断
  155. if (targetValue == null || refValue == null) {
  156. System.err.println("日期 " + date + " 缺少必要字段数据,跳过计算");
  157. continue;
  158. }
  159. double result;
  160. if (targetValue >= 8) {
  161. result = targetValue; // 直接返回
  162. } else {
  163. result = targetValue + (refValue / 60); // 组合计算
  164. }
  165. finalResult.put(date, result);
  166. }
  167. // Step 3. 打印最终结果
  168. System.out.println("===== 计算结果 =====");
  169. finalResult.forEach((date, value) -> {
  170. System.out.printf("日期: %s \t 结果值: %.1f%n", date, value);
  171. });
  172. }
  173. @Override
  174. public void syncKqData(LocalDateTime startTime, LocalDateTime endTime) {
  175. String stTime=UtilDateTime.formatLocal(startTime,UtilDateTime.DATE_TIME_PATTERN);
  176. String edTime=UtilDateTime.formatLocal(endTime,UtilDateTime.DATE_TIME_PATTERN);
  177. List<Map> list=ydService.queryAllFormData(YDParam.builder().formUuid("FORM-5061F7AE543B429C8241EC730A6F31653NNK").build());
  178. for(Map map:list){
  179. Map formData= UtilMap.getMap(map,"formData");
  180. String deptId=String.valueOf(UtilMap.getList(formData,"departmentSelectField_m8e4zaju_id").get(0));
  181. String price=UtilMap.getString(formData,"numberField_m8e4zajw_value");
  182. List<String> users=ddClient_contacts.listDepartmentUserId(ddClient.getAccessToken(),Long.parseLong(deptId));
  183. for (String user : users) {
  184. List<Map> valList=ddClientAttendance.getAttColumnVal(ddClient.getAccessToken(),user, Arrays.asList("82772106","89349019","82772125"),stTime,edTime);
  185. if(valList!=null&&valList.size()==3){
  186. Map<String,Map<String,String>> dateMap=toDateMap(valList);
  187. for (String date:dateMap.keySet()) {
  188. Map<String,String> dataMap=dateMap.get(date);
  189. String workTime="休息".equals(dataMap.get("82772125"))?
  190. dataMap.get("89349019"): NumberUtil.add(dataMap.get("89349019"),dataMap.get("82772106")).toString();
  191. saveData(user,date,workTime,price);
  192. //同步到数据库
  193. LambdaQueryWrapper<YdUserInput> ydUserInputLambdaQueryWrapper = new LambdaQueryWrapper<>();
  194. ydUserInputLambdaQueryWrapper.eq(YdUserInput::getUserId,user)
  195. .eq(YdUserInput::getDate, date);
  196. YdUserInput ydUserInput = ydUserInputMapper.selectOne(ydUserInputLambdaQueryWrapper);
  197. if (Objects.nonNull(ydUserInput)){
  198. ydUserInput.setWorkHours(workTime);
  199. ydUserInput.setPrice(price);
  200. ydUserInput.setTotalPrice(NumberUtil.mul(workTime,price));
  201. ydUserInputMapper.update(ydUserInput,ydUserInputLambdaQueryWrapper);
  202. }else {
  203. try {
  204. ydUserInput = new YdUserInput();
  205. ydUserInput.setUserId(user);
  206. SimpleDateFormat formatter = new SimpleDateFormat(UtilDateTime.DATE_TIME_PATTERN);
  207. ydUserInput.setDate(formatter.parse(date));
  208. ydUserInput.setWorkHours(workTime);
  209. ydUserInput.setPrice(price);
  210. ydUserInput.setTotalPrice(NumberUtil.mul(workTime,price));
  211. ydUserInputMapper.insert(ydUserInput);
  212. }catch (ParseException e){
  213. log.info("无法解析日期字符串: {}",e.getMessage());
  214. }
  215. }
  216. }
  217. }
  218. }
  219. }
  220. }
  221. @Override
  222. public void upsertNonProdPrice(Map map) {
  223. String formInstId = UtilMap.getString(map, "formInstId");
  224. Map formData = ydClient.queryData(YDParam.builder()
  225. .formInstId(formInstId)
  226. .build(), YDConf.FORM_QUERY.retrieve_id).getFormData();
  227. String type = UtilMap.getString(formData, "textField_m8fljzks");
  228. String form = UtilMap.getString(formData, "textField_m8fljzkt");
  229. String position = UtilMap.getString(formData, "textField_m8fljzku");
  230. String materialCodeLeft = UtilMap.getString(formData, "textField_m8fljzkv");
  231. String price = UtilMap.getString(formData, "numberField_m8fljzkw");
  232. LambdaQueryWrapper<YdNonProdPrice> ydNonProdPriceLambdaQueryWrapper = new LambdaQueryWrapper<>();
  233. ydNonProdPriceLambdaQueryWrapper.eq(YdNonProdPrice::getType,type)
  234. .eq(YdNonProdPrice::getForm,form)
  235. .eq(YdNonProdPrice::getPosition,position)
  236. .eq(YdNonProdPrice::getMaterialCodeLeft,materialCodeLeft);
  237. YdNonProdPrice ydNonProdPrice = ydNonProdPriceMapper.selectOne(ydNonProdPriceLambdaQueryWrapper);
  238. if (Objects.nonNull(ydNonProdPrice)){
  239. ydNonProdPrice.setPrice(price);
  240. ydNonProdPriceMapper.update(ydNonProdPrice,ydNonProdPriceLambdaQueryWrapper);
  241. }else {
  242. ydNonProdPrice = new YdNonProdPrice();
  243. ydNonProdPrice.setType(type);
  244. ydNonProdPrice.setForm(form);
  245. ydNonProdPrice.setPosition(position);
  246. ydNonProdPrice.setMaterialCodeLeft(materialCodeLeft);
  247. ydNonProdPrice.setPrice(price);
  248. ydNonProdPriceMapper.insert(ydNonProdPrice);
  249. }
  250. }
  251. @Override
  252. public void upsertWorkHoursPrice(Map map) {
  253. String formInstId = UtilMap.getString(map, "formInstId");
  254. Map formData = ydClient.queryData(YDParam.builder()
  255. .formInstId(formInstId)
  256. .build(), YDConf.FORM_QUERY.retrieve_id).getFormData();
  257. String type = UtilMap.getString(formData, "selectField_m8zib4co");
  258. String price = UtilMap.getString(formData, "numberField_m8zib4cp");
  259. LambdaQueryWrapper<YdProdWorkHoursPrice> ydProdWorkHoursPriceLambdaQueryWrapper = new LambdaQueryWrapper<>();
  260. ydProdWorkHoursPriceLambdaQueryWrapper.eq(YdProdWorkHoursPrice::getType,type);
  261. YdProdWorkHoursPrice ydProdWorkHoursPrice = ydProdWorkHoursPriceMapper.selectOne(ydProdWorkHoursPriceLambdaQueryWrapper);
  262. if (Objects.nonNull(ydProdWorkHoursPrice)){
  263. ydProdWorkHoursPrice.setPrice(price);
  264. ydProdWorkHoursPriceMapper.update(ydProdWorkHoursPrice,ydProdWorkHoursPriceLambdaQueryWrapper);
  265. }else {
  266. ydProdWorkHoursPrice = new YdProdWorkHoursPrice();
  267. ydProdWorkHoursPrice.setType(type);
  268. ydProdWorkHoursPrice.setPrice(price);
  269. ydProdWorkHoursPriceMapper.insert(ydProdWorkHoursPrice);
  270. }
  271. }
  272. /*@Override
  273. public void syncUserInput(LocalDateTime startTime, LocalDateTime endTime) {
  274. List<Map> dataList = getYdFormDataList("FORM-78BF5DE20E244369BD475C11A5CA48A0ZAQJ", "", YDConf.FORM_QUERY.retrieve_list_all);
  275. for (Map data : dataList) {
  276. Map formData = UtilMap.getMap(data, "formData");
  277. YdUserInput ydUserInput = new YdUserInput();
  278. ydUserInput.setUser(UtilMap.getList(formData, "employeeField_m8cu0g58").get(0).toString());
  279. ydUserInput.setUserId(UtilMap.getList(formData, "employeeField_m8cu0g58_id").get(0).toString());
  280. ydUserInput.setDate(new Date(UtilMap.getLong(formData,"dateField_m8e53oef")));
  281. ydUserInput.setWorkHours(UtilMap.getString(formData,"numberField_m8ctvm2e"));
  282. ydUserInput.setPrice(UtilMap.getString(formData,"numberField_m8e53oeg"));
  283. ydUserInput.setTotalPrice(UtilMap.getString(formData,"numberField_m8e53oeh"));
  284. ydUserInputMapper.insert(ydUserInput);
  285. }
  286. }*/
  287. private List<Map> getYdFormDataList(String formUuid, String searchCondition, YDConf.FORM_QUERY formQuery) {
  288. List<Map> list = new ArrayList<>();
  289. DDR_New ddrNew = new DDR_New();
  290. int pageNumber = 1;
  291. int pageSize = 100;
  292. do {
  293. ddrNew = ydClient.queryData(YDParam.builder().formUuid(formUuid)
  294. .searchCondition(searchCondition)
  295. .pageNumber(pageNumber)
  296. .pageSize(pageSize).build(), formQuery);
  297. list.addAll((List<Map>) ddrNew.getData());
  298. pageNumber++;
  299. }while (ddrNew.getTotalCount() > ddrNew.getPageNumber() * pageSize);
  300. return list;
  301. }
  302. private Map<String,Map<String,String>> toDateMap(List<Map> valList){
  303. Map<String,Map<String,String>> result = new HashMap<>();
  304. for(Map map:valList){
  305. String columnId=UtilMap.getString(UtilMap.getMap(map,"column_vo"),"id");
  306. List<Map> list=UtilMap.getList(map,"column_vals");
  307. for(Map map1:list){
  308. String date=UtilMap.getString(map1,"date");
  309. String value=UtilMap.getString(map1,"value");
  310. value=columnId.equals("82772106")?NumberUtil.round(NumberUtil.div(value,"60"),2).toString():value; // 工作时长 分钟转小时
  311. if(result.containsKey(date)){
  312. result.get(date).put(columnId,value);
  313. }else{
  314. result.put(date,UtilMap.map(columnId,value));
  315. }
  316. }
  317. }
  318. return result;
  319. }
  320. private void saveData(String userId,String date,String gs,String price){
  321. Long dateTime= UtilDateTime.getLocalDateTimeTimeStamp(UtilDateTime.parseLocalDateTime(date));
  322. Map saveMap=UtilMap.map("employeeField_m8cu0g58, dateField_m8e53oef, numberField_m8ctvm2e, numberField_m8e53oeg, numberField_m8e53oeh",
  323. Arrays.asList(userId),dateTime,gs,price,NumberUtil.mul(gs,price));
  324. ydClient.operateData(YDParam.builder().formUuid("FORM-78BF5DE20E244369BD475C11A5CA48A0ZAQJ")
  325. .searchCondition(JSONObject.toJSONString(UtilMap.map("employeeField_m8cu0g58, dateField_m8e53oef", userId,Arrays.asList(dateTime,dateTime))))
  326. .formDataJson(JSONObject.toJSONString(saveMap)).build(), YDConf.FORM_OPERATION.upsert);
  327. }
  328. /**
  329. * 生成一个由大写字母和数字组成的八位随机账号。
  330. *
  331. * @return 生成的随机账号字符串
  332. */
  333. public static String generateRandomAccount() {
  334. // 可选字符集合:大写字母A-Z和数字0-9
  335. final String CHARS = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  336. final int LENGTH = 8; // 账号长度
  337. Random random = new Random();
  338. StringBuilder sb = new StringBuilder(LENGTH);
  339. for (int i = 0; i < LENGTH; i++) {
  340. // 随机选择一个字符
  341. int index = random.nextInt(CHARS.length());
  342. char randomChar = CHARS.charAt(index);
  343. sb.append(randomChar);
  344. }
  345. return sb.toString();
  346. }
  347. /**
  348. * 生成一个六位数的随机密码。
  349. *
  350. * @return 生成的随机密码字符串
  351. */
  352. public static String generateRandomSixDigitPassword() {
  353. Random random = new Random();
  354. int min = 100000; // 六位数最小值
  355. int max = 999999; // 六位数最大值
  356. // 生成一个介于[min, max]之间的随机数
  357. int randomNumber = random.nextInt(max - min + 1) + min;
  358. // 将生成的随机数转换为字符串
  359. return String.valueOf(randomNumber);
  360. }
  361. }