北斗定位
Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.

218 lines
8.7KB

  1. package com.telpo.dipperposition.service.impl;
  2. import com.alibaba.fastjson.JSONObject;
  3. import com.telpo.dipperposition.common.*;
  4. import com.telpo.dipperposition.config.NettyServerConfig;
  5. import com.telpo.dipperposition.config.PositionConfig;
  6. import com.telpo.dipperposition.entity.mongo.IpProvinceEntity;
  7. import com.telpo.dipperposition.service.IDipperAstPosAsyncTaskService;
  8. import com.telpo.dipperposition.service.IpProvinceService;
  9. import lombok.extern.slf4j.Slf4j;
  10. import org.apache.commons.lang3.ObjectUtils;
  11. import org.apache.logging.log4j.util.PropertiesUtil;
  12. import org.springframework.beans.factory.annotation.Autowired;
  13. import org.springframework.scheduling.annotation.Async;
  14. import org.springframework.stereotype.Service;
  15. import org.springframework.util.ResourceUtils;
  16. import java.io.FileNotFoundException;
  17. import java.io.UnsupportedEncodingException;
  18. import java.util.List;
  19. /**
  20. * @program: DipperAstPosAsyncTaskServiceImpl
  21. * @description: 系统预先基于省份的省会城市的经纬度作为辅助信息,
  22. * * 根据设备请求的IP地址,从高德IP定位服务获取相关的省份,再匹配相应的位置信息作为辅助位置信息。
  23. * * 如果匹配不到省份,则以武汉中心作为辅助为位置信息。
  24. * * 关于IP与省份的关系保存到缓存中,用于下次使用时,先在缓存中获取匹配信息,匹配不到,再请求高德IP定位服务。
  25. * * 高德IP定位服务:https://lbs.amap.com/api/webservice/guide/api/ipconfig。
  26. * @author: king
  27. * @create: 2021-01-10 14:01
  28. */
  29. @Service
  30. @Slf4j
  31. public class DipperAstPosAsyncTaskServiceImpl implements IDipperAstPosAsyncTaskService {
  32. @Autowired
  33. private RedisUtil redisUtil;
  34. @Autowired
  35. private OkHttpUtil okHttpUtil;
  36. @Autowired
  37. private PositionConfig positionConfig;
  38. @Autowired
  39. private IpProvinceService ipProvinceService;
  40. private String centerProvince;
  41. private String centerProvinceFilePath;
  42. private String ipPositionRequestPath;
  43. private String ipPositionRequestKey;
  44. // 从CSV文件读取省会城市中心点位置信息
  45. private void getPosFromFile(String centerAddress) {
  46. // 不存在说明token是已过期了
  47. String centerProvinceName = "";
  48. String centerProvinceLonAndAlt = "";
  49. String appCsvPath = "";
  50. try {
  51. appCsvPath = ResourceUtils.getFile("resources"+this.centerProvinceFilePath).getPath();
  52. log.debug(appCsvPath);
  53. } catch (Exception e) {
  54. log.debug("cannot find path");
  55. }
  56. List<String> centerAddressSets = CSVUtil.readCSV(appCsvPath);
  57. for (String centerAddressSet:centerAddressSets) {
  58. String[] centerAddressSetArray = centerAddressSet.split(",");
  59. if (centerAddressSetArray.length < 3) {
  60. log.warn("CSV数据格式错误");
  61. } else {
  62. centerProvinceName = centerAddressSetArray[3];
  63. centerProvinceLonAndAlt = centerAddressSetArray[1]+","+centerAddressSetArray[2];
  64. redisUtil.set(centerProvinceName, centerProvinceLonAndAlt, 0);
  65. }
  66. }
  67. }
  68. // 根据IP获取省会信息
  69. private String getIpPositionProvince(String ipAddress) {
  70. // 关于IP与省份的关系保存到缓存中
  71. // 使用时,先在缓存中获取匹配信息
  72. // 用mongodb实现
  73. IpProvinceEntity ipProvinceEntity = ipProvinceService.getIpProvince(ipAddress);
  74. if (ipProvinceEntity == null) {
  75. // 匹配不到,再请求高德IP定位服务。
  76. JSONObject userObj = new JSONObject();
  77. userObj.put("ip", ipAddress);
  78. userObj.put("key", ipPositionRequestKey);
  79. JSONObject json = okHttpUtil.postRequestWithJson(ipPositionRequestPath, null, userObj);
  80. if (ObjectUtils.isNotEmpty(json)) {
  81. String province = (String) json.get("province");
  82. if (ObjectUtils.isEmpty(province)) {
  83. log.debug("json is :" + json.toString());
  84. return null;
  85. }
  86. return province;
  87. } else {
  88. // 意外错误
  89. log.debug("ip address is null");
  90. return null;
  91. }
  92. } else {
  93. return ipProvinceEntity.getProvince();
  94. }
  95. }
  96. // 将IP对应的省会保存到mongoDB
  97. @Async("asyncServiceExecutor")
  98. public void createIPProvince(String ipAddress, String province) {
  99. log.debug("异步创建推送失败任务记录!");
  100. try {
  101. IpProvinceEntity ipProvinceEntity = ipProvinceService.getIpProvince(ipAddress);
  102. if (ipProvinceEntity == null) {
  103. ipProvinceEntity.setIp(ipAddress);
  104. ipProvinceEntity.setProvince(province);
  105. ipProvinceService.saveIpProvince(ipProvinceEntity);
  106. }
  107. } catch (Exception e) {
  108. log.error("创建推送失败记录异常:", e);
  109. }
  110. }
  111. /*
  112. * 获取定位辅助信息
  113. * @param ipAddress
  114. */
  115. @Override
  116. @Async("asyncServiceExecutor")
  117. public String pushAstPos(String ipAddress) {
  118. this.ipPositionRequestKey = positionConfig.getIpPositionRequestKey();
  119. this.ipPositionRequestPath = positionConfig.getIpPositionRequestPath();
  120. this.centerProvince = positionConfig.getCenterProvince();
  121. this.centerProvinceFilePath = positionConfig.getCenterProvinceFilePath();
  122. // (1) 获取省会城市信息
  123. String centerAddress = getIpPositionProvince(ipAddress);
  124. if (ObjectUtils.isEmpty(centerAddress) || centerAddress.equals("0")) {
  125. log.warn("IP地址非法,无法获取辅助位置信息!");
  126. // 返回武汉的定位数据
  127. centerAddress = this.centerProvince;
  128. } else {
  129. // 保存到mongoDB
  130. createIPProvince(ipAddress, centerAddress);
  131. }
  132. String lonAndAlt;
  133. if (redisUtil.hasKey(centerAddress)) {
  134. // 获取省会城市定位信息
  135. lonAndAlt= (String) redisUtil.get(centerAddress);
  136. } else {
  137. // 请求高德IP定位服务
  138. this.getPosFromFile(centerAddress);
  139. lonAndAlt = (String) redisUtil.get(centerAddress);
  140. }
  141. // (2) 处理返回结果
  142. if (lonAndAlt == null) {
  143. // null处理
  144. log.error("系统错误,请联系系统管理员。");
  145. return null;
  146. //return;
  147. }
  148. // push to GNNS Server
  149. String pushResult = getCmdOfPos(lonAndAlt);
  150. return pushResult;
  151. }
  152. // 组装命令发送给设备
  153. private String getCmdOfPos(String astPos) {
  154. // 时间和位置不是从服务器获取,而是本地生成
  155. String[] astPosArray = astPos.split(",");
  156. String lan = astPosArray[0].trim();
  157. String alt = astPosArray[1].trim();
  158. double lanValue = Double.parseDouble(lan) * 10000000;
  159. long lanLongValue = Double.doubleToLongBits(lanValue);
  160. if (lanLongValue < 0) {
  161. lanLongValue = lanLongValue + 4294967295L + 1;
  162. }
  163. double altValue = Double.parseDouble(alt) * 10000000;
  164. long altLongValue = Double.doubleToLongBits(altValue);
  165. if (altLongValue < 0) {
  166. altLongValue = altLongValue + 4294967295L + 1;
  167. }
  168. // 数值换算举例(以经度举例。纬度、高度、位置精度换算方法一致):
  169. // (1)经度数值为 113.431,则换算方法如下:
  170. // 113.431/比例因子 = 1134310000(十进制)
  171. //  439C3270(十六进制)
  172. //  经度数据填入 70 32 9C 43(小端模式)
  173. // (2)经度数值为-113.431,则换算方法如下:
  174. // 113.431/比例因子 = 1134310000(十进制)
  175. //  439C3270(十六进制)
  176. //  FFFFFFFF - 439C3270 + 1= BC63CD90(补码)
  177. // 经度数据填入 90 CD 63 BC(小端模式)
  178. // 指令(十六进制)
  179. // 举例: 23 3E 04 01 10 00 70 32 9C 43 D0 B2 CE 0D 70 17 00 00 40 0D 03 00 CA 95
  180. // 其中
  181. // 23 3E 为同步头
  182. // 04 01 为识别码
  183. // 10 00 表示长度为 16
  184. // 70 32 9C 43 表示注入的辅助经度为 113.431 度
  185. // D0 B2 CE 0D 表示注入的辅助纬度为 23.165 度
  186. // 00 2F 为校验和
  187. // astTimeCmd 组装
  188. String astTimeCmd = "233E0401";
  189. astTimeCmd += "1000";
  190. astTimeCmd += HexConvert.encodeHEX(lanLongValue);
  191. astTimeCmd += HexConvert.encodeHEX(altLongValue);
  192. //String hexIn = HexConvert.convertStringToHex(astTimeCmd) + HexConvert.makeChecksum(astTimeCmd);
  193. String hexIn = astTimeCmd + HexConvert.makeChecksum(astTimeCmd);
  194. //return sendResult;
  195. return hexIn;
  196. }
  197. }