ChargingMachineClient.java 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. package com.ruoyi.hard.modbus.tcp;
  2. import com.jwk.spring.boot.autoconfigure.ModbusTcpMasterTemplate;
  3. import com.jwk.spring.boot.modbus4j.ModbusMasterUtil;
  4. import com.serotonin.modbus4j.msg.ReadResponse;
  5. import lombok.extern.slf4j.Slf4j;
  6. import org.springframework.beans.factory.annotation.Autowired;
  7. import org.springframework.beans.factory.annotation.Qualifier;
  8. import org.springframework.stereotype.Service;
  9. import javax.annotation.PostConstruct;
  10. import java.util.Arrays;
  11. import java.util.HashMap;
  12. import java.util.LinkedHashMap;
  13. import java.util.Map;
  14. import static com.ruoyi.hard.modbus.tcp.ChargingMachineClient.CHARGER_ADDRESS_MEANING.*;
  15. /**
  16. * 充电机
  17. *
  18. * @author JWK
  19. * @version 1.0
  20. * @date 2022/9/9 15:03
  21. */
  22. @Slf4j
  23. @Service
  24. public class ChargingMachineClient {
  25. /**
  26. * 充电机(三向车)
  27. */
  28. @Autowired(required = false)
  29. @Qualifier("modbusTcpMasterTemplateFourth")
  30. private ModbusTcpMasterTemplate modbusTcpMasterTemplateFourth;
  31. /**
  32. * 充电机(迷你堆垛)
  33. */
  34. @Autowired(required = false)
  35. @Qualifier("modbusTcpMasterTemplateFifth")
  36. private ModbusTcpMasterTemplate modbusTcpMasterTemplateFifth;
  37. private Map<String, ModbusTcpMasterTemplate> tcpMasterTemplateMap;
  38. @PostConstruct
  39. public void initMap() {
  40. tcpMasterTemplateMap = new HashMap<>();
  41. tcpMasterTemplateMap.put(MACHINE_NO.A.name(), modbusTcpMasterTemplateFourth);
  42. tcpMasterTemplateMap.put(MACHINE_NO.B.name(), modbusTcpMasterTemplateFifth);
  43. }
  44. public ModbusMasterUtil getModbusMasterUtil(MACHINE_NO machineNo) {
  45. return tcpMasterTemplateMap.get(machineNo.name()).getModbusMasterUtil();
  46. }
  47. /**
  48. * 机器编号
  49. */
  50. public enum MACHINE_NO {
  51. A("1",1),
  52. B("2",2);
  53. private String value;
  54. private int no;
  55. MACHINE_NO(String value,Integer no) {
  56. this.value = value;
  57. this.no = no;
  58. }
  59. public String getValue() {
  60. return value;
  61. }
  62. public int getNo() {
  63. return no;
  64. }
  65. /**
  66. * 根据值获得枚举类型 switch
  67. *
  68. * @param value
  69. * @return
  70. */
  71. public static MACHINE_NO getByValue(String value) {
  72. for (MACHINE_NO code : values()) {
  73. if (code.getValue().equals(value)) {
  74. return code;
  75. }
  76. }
  77. return null;
  78. }
  79. /**
  80. * 根据值获得枚举类型 switch
  81. *
  82. * @param no
  83. * @return
  84. */
  85. public static MACHINE_NO getByNo(int no) {
  86. for (MACHINE_NO code : values()) {
  87. if (code.getNo() == no) {
  88. return code;
  89. }
  90. }
  91. return null;
  92. }
  93. }
  94. /**
  95. * 充电机操作
  96. *
  97. * @param operation
  98. * @return
  99. */
  100. public boolean operation(CHARGER_ADDRESS_MEANING operation, MACHINE_NO con) {
  101. switch (operation) {
  102. // 启动
  103. case START:
  104. return start(con);
  105. // 停止
  106. case STOP:
  107. return stop(con);
  108. // 放电
  109. case DISCHARGE:
  110. return discharge(con);
  111. default:
  112. break;
  113. }
  114. return false;
  115. }
  116. /**
  117. * 获取状态类型和状态对应关系表
  118. *
  119. * @return
  120. */
  121. public Map<CHARGER_STATUS, Boolean> getStatusMapping(MACHINE_NO con) {
  122. return ChargingMachineClient.CHARGER_STATUS.getMapping(getStatus(con));
  123. }
  124. /**
  125. * 获取状态描述和状态对应关系表
  126. *
  127. * @return
  128. */
  129. public Map<String, Boolean> getStatusNameMapping(MACHINE_NO con) {
  130. return ChargingMachineClient.CHARGER_STATUS.getNameMapping(getStatus(con));
  131. }
  132. /**
  133. * 获取充电机状态
  134. * 第4个字节(状态代码) 第5个字节(故障代码) 第6个字节(各种状态,有些功能需要硬件支持)
  135. * 返回3个字节 24位
  136. * 假设3个字节为:00000110 00000101 00000100
  137. * 则最终返回的数组:[0, 1, 1, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0]
  138. */
  139. public boolean[] getStatus(MACHINE_NO con) {
  140. ModbusMasterUtil modbusMasterUtil = getModbusMasterUtil(con);
  141. ReadResponse read = modbusMasterUtil.readCoils(1, CHARGER_STATUS.getOffset(), CHARGER_STATUS.getCommand());
  142. if (read == null || read.getBooleanData() == null) {
  143. log.error("获取充电机状态为空!");
  144. return null;
  145. }
  146. if (read.getBooleanData().length != 24) {
  147. log.error("获取充电机状态错误,不足24位:" + Arrays.toString(read.getBooleanData()));
  148. return null;
  149. }
  150. return read.getBooleanData();
  151. }
  152. /**
  153. * 启动充电机
  154. *
  155. * @return
  156. */
  157. private boolean start(MACHINE_NO con) {
  158. ModbusMasterUtil modbusMasterUtil = getModbusMasterUtil(con);
  159. return modbusMasterUtil.writeCoil(1, START.getOffset(), START.getCommand());
  160. }
  161. /**
  162. * 停止充电机
  163. *
  164. * @return
  165. */
  166. private boolean stop(MACHINE_NO con) {
  167. ModbusMasterUtil modbusMasterUtil = getModbusMasterUtil(con);
  168. return modbusMasterUtil.writeCoil(1, STOP.getOffset(), STOP.getCommand());
  169. }
  170. /**
  171. * 充电机放电
  172. *
  173. * @return
  174. */
  175. private boolean discharge(MACHINE_NO con) {
  176. ModbusMasterUtil modbusMasterUtil = getModbusMasterUtil(con);
  177. return modbusMasterUtil.writeCoil(1, DISCHARGE.getOffset(), DISCHARGE.getCommand());
  178. }
  179. /**
  180. * 功能码
  181. */
  182. public enum FUNCTION_CODE {
  183. /***
  184. * 读线圈寄存器
  185. */
  186. READ_01H(0x01),
  187. /***
  188. * 读保持寄存器
  189. */
  190. READ_03H(0x03),
  191. /***
  192. * 写单个线圈寄存器
  193. */
  194. WRITE_05H(0x05),
  195. /***
  196. * 写单个保持寄存器
  197. */
  198. WRITE_06H(0x06);
  199. private Integer value;
  200. FUNCTION_CODE(Integer value) {
  201. this.value = value;
  202. }
  203. public Integer getValue() {
  204. return value;
  205. }
  206. }
  207. /**
  208. * 充电机ModBus地址码
  209. */
  210. public enum CHARGER_ADDRESS_MEANING {
  211. /***
  212. * 启动充电机
  213. */
  214. START(FUNCTION_CODE.WRITE_05H, 0x0007, 0, 0x0007, 0XFF00),
  215. /***
  216. * 停止充电机
  217. */
  218. STOP(FUNCTION_CODE.WRITE_05H, 0x0007, 0, 0x0007, 0X0000),
  219. /**
  220. * 充电机放电
  221. */
  222. DISCHARGE(FUNCTION_CODE.WRITE_05H, 0x0008, 0, 0x0008, 0xFF00),
  223. /**
  224. * 充电机状态
  225. */
  226. CHARGER_STATUS(FUNCTION_CODE.READ_01H, 0x0040, 24, 0x0040, 0X0018),
  227. /**
  228. * 读取电压(/100)
  229. */
  230. READ_VOLTAGE(FUNCTION_CODE.READ_03H, 0x0065, 1, 0x0065, 0X0001),
  231. /**
  232. * 读取电流(/100)
  233. */
  234. READ_CURRENT(FUNCTION_CODE.READ_03H, 0x0066, 1, 0x0066, 0X0001),
  235. /**
  236. * 读取当前容量(/100)
  237. */
  238. READ_CAPACITY(FUNCTION_CODE.READ_03H, 0x0067, 1, 0x0067, 0X0001),
  239. /**
  240. * 时
  241. */
  242. HOURS(FUNCTION_CODE.READ_03H, 0x0068, 1, 0x0068, 0X0001),
  243. /**
  244. * 钟
  245. */
  246. MINUTES(FUNCTION_CODE.READ_03H, 0x0069, 1, 0x0069, 0X0001),
  247. /**
  248. * 秒
  249. */
  250. SECONDS(FUNCTION_CODE.READ_03H, 0x006A, 1, 0x006A, 0X0001),
  251. /**
  252. * 设置电压29.00V
  253. */
  254. SET_VOLTAGE(FUNCTION_CODE.WRITE_06H, 0x0090, 0, 0x0090, 0X0B54),
  255. /**
  256. * 设置电流29.00A
  257. */
  258. SET_CURRENT(FUNCTION_CODE.WRITE_06H, 0x0091, 0, 0x0091, 0X0B54),
  259. /**
  260. * 离线
  261. */
  262. SET_OFFLINE(FUNCTION_CODE.WRITE_06H, 0x0092, 0, 0x0092, 0X000),
  263. /**
  264. * 自动
  265. */
  266. SET_AUTO(FUNCTION_CODE.WRITE_06H, 0x0092, 0, 0x0092, 0X001),
  267. /**
  268. * 在线
  269. */
  270. SET_ONLINE(FUNCTION_CODE.WRITE_06H, 0x0092, 0, 0x0092, 0X002);
  271. /**
  272. * 功能码
  273. */
  274. private FUNCTION_CODE functionCode;
  275. /**
  276. * 读取和写入的开始偏移量
  277. */
  278. private int offset;
  279. /**
  280. * 长度 10进制
  281. */
  282. private int length;
  283. /**
  284. * PLC地址
  285. */
  286. private int address;
  287. /**
  288. * 命令
  289. */
  290. private int command;
  291. CHARGER_ADDRESS_MEANING(FUNCTION_CODE functionCode, int offset, int length, int address, int command) {
  292. this.functionCode = functionCode;
  293. this.offset = offset;
  294. this.length = length;
  295. this.address = address;
  296. this.command = command;
  297. }
  298. public FUNCTION_CODE getFunctionCode() {
  299. return functionCode;
  300. }
  301. public int getOffset() {
  302. return offset;
  303. }
  304. public int getLength() {
  305. return length;
  306. }
  307. public int getAddress() {
  308. return address;
  309. }
  310. public int getCommand() {
  311. return command;
  312. }
  313. }
  314. /**
  315. * 充电机状态:
  316. * 第4个字节(状态代码) 第5个字节(故障代码) 第6个字节(各种状态,有些功能需要硬件支持)
  317. * 假设:第4个字节[00000110] 第5个字节[00000101] 第6个[00000100]
  318. * 放在同一个数组中表示:[0, 1, 1, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0]
  319. */
  320. public enum CHARGER_STATUS {
  321. // 4bit
  322. BIT_04_0("待机", 0),
  323. BIT_04_1("故障", 1),
  324. BIT_04_2("进行", 2),
  325. BIT_04_3("完成", 3),
  326. BIT_04_4("离线", 4),
  327. BIT_04_5("在线", 5),
  328. BIT_04_6("空", 6),
  329. BIT_04_7("空", 7),
  330. // 5bit
  331. BIT_05_0("总故障", 8),
  332. BIT_05_1("过流", 9),
  333. BIT_05_2("过压", 10),
  334. BIT_05_3("短路", 11),
  335. BIT_05_4("电池未接", 12),
  336. BIT_05_5("电池反接", 13),
  337. BIT_05_6("模块通信故障", 14),
  338. BIT_05_7("Can通信超时", 15),
  339. // 6bit
  340. BIT_06_0("归位", 16),
  341. BIT_06_1("正在伸出", 17),
  342. BIT_06_2("正在退回", 18),
  343. BIT_06_3("伸出充电", 19),
  344. BIT_06_4("伸出到底", 20),
  345. BIT_06_5("伸缩故障", 21),
  346. BIT_06_6("压紧", 22),
  347. BIT_06_7("红外到位", 23);
  348. /**
  349. * 判断某位是否为true
  350. *
  351. * @param booleans
  352. * @return
  353. */
  354. public boolean isTrue(boolean[] booleans) {
  355. return booleans[this.getIndex()];
  356. }
  357. /**
  358. * 获取枚举类型和状态对应关系表
  359. *
  360. * @param booleans
  361. * @return
  362. */
  363. public static Map<CHARGER_STATUS, Boolean> getMapping(boolean[] booleans) {
  364. if (booleans == null) {
  365. return null;
  366. }
  367. Map<CHARGER_STATUS, Boolean> map = new LinkedHashMap<>();
  368. for (CHARGER_STATUS value : values()) {
  369. map.put(value, value.isTrue(booleans));
  370. }
  371. return map;
  372. }
  373. /**
  374. * 获取描述和状态对应关系表
  375. *
  376. * @param booleans
  377. * @return
  378. */
  379. public static Map<String, Boolean> getNameMapping(boolean[] booleans) {
  380. if (booleans == null) {
  381. return null;
  382. }
  383. Map<String, Boolean> map = new LinkedHashMap<>();
  384. Map<CHARGER_STATUS, Boolean> mapping = getMapping(booleans);
  385. mapping.forEach((k, v) -> map.put(k.getDescription(), v));
  386. mapping.remove(BIT_04_6.getDescription());
  387. return map;
  388. }
  389. /**
  390. * 状态描述
  391. */
  392. private String description;
  393. /**
  394. * 在数组中下标
  395. */
  396. private int index;
  397. CHARGER_STATUS(String description, int index) {
  398. this.description = description;
  399. this.index = index;
  400. }
  401. public String getDescription() {
  402. return description;
  403. }
  404. public int getIndex() {
  405. return index;
  406. }
  407. }
  408. }