XmlUtil.java 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. package com.genersoft.iot.vmp.gb28181.utils;
  2. import com.alibaba.fastjson.JSONArray;
  3. import com.alibaba.fastjson.JSONObject;
  4. import com.genersoft.iot.vmp.gb28181.bean.Device;
  5. import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
  6. import org.dom4j.Attribute;
  7. import org.dom4j.Document;
  8. import org.dom4j.DocumentException;
  9. import org.dom4j.Element;
  10. import org.dom4j.io.SAXReader;
  11. import org.slf4j.Logger;
  12. import org.slf4j.LoggerFactory;
  13. import org.springframework.util.StringUtils;
  14. import javax.sip.RequestEvent;
  15. import javax.sip.message.Request;
  16. import java.io.ByteArrayInputStream;
  17. import java.io.StringReader;
  18. import java.util.*;
  19. /**
  20. * 基于dom4j的工具包
  21. *
  22. *
  23. */
  24. public class XmlUtil {
  25. /**
  26. * 日志服务
  27. */
  28. private static Logger LOG = LoggerFactory.getLogger(XmlUtil.class);
  29. /**
  30. * 解析XML为Document对象
  31. *
  32. * @param xml 被解析的XMl
  33. *
  34. * @return Document
  35. */
  36. public static Element parseXml(String xml) {
  37. Document document = null;
  38. //
  39. StringReader sr = new StringReader(xml);
  40. SAXReader saxReader = new SAXReader();
  41. try {
  42. document = saxReader.read(sr);
  43. } catch (DocumentException e) {
  44. LOG.error("解析失败", e);
  45. }
  46. return null == document ? null : document.getRootElement();
  47. }
  48. /**
  49. * 获取element对象的text的值
  50. *
  51. * @param em 节点的对象
  52. * @param tag 节点的tag
  53. * @return 节点
  54. */
  55. public static String getText(Element em, String tag) {
  56. if (null == em) {
  57. return null;
  58. }
  59. Element e = em.element(tag);
  60. //
  61. return null == e ? null : e.getText().trim();
  62. }
  63. /**
  64. * 递归解析xml节点,适用于 多节点数据
  65. *
  66. * @param node node
  67. * @param nodeName nodeName
  68. * @return List<Map<String, Object>>
  69. */
  70. public static List<Map<String, Object>> listNodes(Element node, String nodeName) {
  71. if (null == node) {
  72. return null;
  73. }
  74. // 初始化返回
  75. List<Map<String, Object>> listMap = new ArrayList<Map<String, Object>>();
  76. // 首先获取当前节点的所有属性节点
  77. List<Attribute> list = node.attributes();
  78. Map<String, Object> map = null;
  79. // 遍历属性节点
  80. for (Attribute attribute : list) {
  81. if (nodeName.equals(node.getName())) {
  82. if (null == map) {
  83. map = new HashMap<String, Object>();
  84. listMap.add(map);
  85. }
  86. // 取到的节点属性放到map中
  87. map.put(attribute.getName(), attribute.getValue());
  88. }
  89. }
  90. // 遍历当前节点下的所有节点 ,nodeName 要解析的节点名称
  91. // 使用递归
  92. Iterator<Element> iterator = node.elementIterator();
  93. while (iterator.hasNext()) {
  94. Element e = iterator.next();
  95. listMap.addAll(listNodes(e, nodeName));
  96. }
  97. return listMap;
  98. }
  99. /**
  100. * xml转json
  101. *
  102. * @param element
  103. * @param json
  104. */
  105. public static void node2Json(Element element, JSONObject json) {
  106. // 如果是属性
  107. for (Object o : element.attributes()) {
  108. Attribute attr = (Attribute) o;
  109. if (!StringUtils.isEmpty(attr.getValue())) {
  110. json.put("@" + attr.getName(), attr.getValue());
  111. }
  112. }
  113. List<Element> chdEl = element.elements();
  114. if (chdEl.isEmpty() && !StringUtils.isEmpty(element.getText())) {// 如果没有子元素,只有一个值
  115. json.put(element.getName(), element.getText());
  116. }
  117. for (Element e : chdEl) { // 有子元素
  118. if (!e.elements().isEmpty()) { // 子元素也有子元素
  119. JSONObject chdjson = new JSONObject();
  120. node2Json(e, chdjson);
  121. Object o = json.get(e.getName());
  122. if (o != null) {
  123. JSONArray jsona = null;
  124. if (o instanceof JSONObject) { // 如果此元素已存在,则转为jsonArray
  125. JSONObject jsono = (JSONObject) o;
  126. json.remove(e.getName());
  127. jsona = new JSONArray();
  128. jsona.add(jsono);
  129. jsona.add(chdjson);
  130. }
  131. if (o instanceof JSONArray) {
  132. jsona = (JSONArray) o;
  133. jsona.add(chdjson);
  134. }
  135. json.put(e.getName(), jsona);
  136. } else {
  137. if (!chdjson.isEmpty()) {
  138. json.put(e.getName(), chdjson);
  139. }
  140. }
  141. } else { // 子元素没有子元素
  142. for (Object o : element.attributes()) {
  143. Attribute attr = (Attribute) o;
  144. if (!StringUtils.isEmpty(attr.getValue())) {
  145. json.put("@" + attr.getName(), attr.getValue());
  146. }
  147. }
  148. if (!e.getText().isEmpty()) {
  149. json.put(e.getName(), e.getText());
  150. }
  151. }
  152. }
  153. }
  154. public static Element getRootElement(RequestEvent evt) throws DocumentException {
  155. return getRootElement(evt, "gb2312");
  156. }
  157. public static Element getRootElement(RequestEvent evt, String charset) throws DocumentException {
  158. Request request = evt.getRequest();
  159. return getRootElement(request.getRawContent(), charset);
  160. }
  161. public static Element getRootElement(byte[] content, String charset) throws DocumentException {
  162. if (charset == null) {
  163. charset = "gb2312";
  164. }
  165. SAXReader reader = new SAXReader();
  166. reader.setEncoding(charset);
  167. Document xml = reader.read(new ByteArrayInputStream(content));
  168. return xml.getRootElement();
  169. }
  170. public static DeviceChannel channelContentHander(Element itemDevice, Device device){
  171. Element channdelNameElement = itemDevice.element("Name");
  172. String channelName = channdelNameElement != null ? channdelNameElement.getTextTrim().toString() : "";
  173. Element statusElement = itemDevice.element("Status");
  174. String status = statusElement != null ? statusElement.getTextTrim().toString() : "ON";
  175. DeviceChannel deviceChannel = new DeviceChannel();
  176. deviceChannel.setName(channelName);
  177. Element channdelIdElement = itemDevice.element("DeviceID");
  178. String channelId = channdelIdElement != null ? channdelIdElement.getTextTrim().toString() : "";
  179. deviceChannel.setChannelId(channelId);
  180. // ONLINE OFFLINE HIKVISION DS-7716N-E4 NVR的兼容性处理
  181. if (status.equals("ON") || status.equals("On") || status.equals("ONLINE") || status.equals("OK")) {
  182. deviceChannel.setStatus(1);
  183. }
  184. if (status.equals("OFF") || status.equals("Off") || status.equals("OFFLINE")) {
  185. deviceChannel.setStatus(0);
  186. }
  187. deviceChannel.setManufacture(XmlUtil.getText(itemDevice, "Manufacturer"));
  188. deviceChannel.setModel(XmlUtil.getText(itemDevice, "Model"));
  189. deviceChannel.setOwner(XmlUtil.getText(itemDevice, "Owner"));
  190. deviceChannel.setCivilCode(XmlUtil.getText(itemDevice, "CivilCode"));
  191. deviceChannel.setBlock(XmlUtil.getText(itemDevice, "Block"));
  192. deviceChannel.setAddress(XmlUtil.getText(itemDevice, "Address"));
  193. String businessGroupID = XmlUtil.getText(itemDevice, "BusinessGroupID");
  194. if (XmlUtil.getText(itemDevice, "Parental") == null
  195. || XmlUtil.getText(itemDevice, "Parental").equals("")) {
  196. if (deviceChannel.getChannelId().length() <= 10
  197. || (deviceChannel.getChannelId().length() == 20 && (
  198. Integer.parseInt(deviceChannel.getChannelId().substring(10, 13)) == 215
  199. || Integer.parseInt(deviceChannel.getChannelId().substring(10, 13)) == 216
  200. )
  201. )
  202. ) {
  203. deviceChannel.setParental(1);
  204. }else {
  205. deviceChannel.setParental(0);
  206. }
  207. } else {
  208. // 由于海康会错误的发送65535作为这里的取值,所以这里除非是0否则认为是1
  209. deviceChannel.setParental(Integer.parseInt(XmlUtil.getText(itemDevice, "Parental")) == 1?1:0);
  210. }
  211. /**
  212. * 行政区划展示设备树与业务分组展示设备树是两种不同的模式
  213. * 行政区划展示设备树 各个目录之间主要靠deviceId做关联,摄像头通过CivilCode指定其属于那个行政区划;都是不超过十位的编号; 结构如下:
  214. * 河北省
  215. * --> 石家庄市
  216. * --> 摄像头
  217. * --> 正定县
  218. * --> 摄像头
  219. * --> 摄像头
  220. *
  221. * 业务分组展示设备树是顶级是业务分组,其下的虚拟组织靠BusinessGroupID指定其所属的业务分组;摄像头通过ParentId来指定其所属于的虚拟组织:
  222. * 业务分组
  223. * --> 虚拟组织
  224. * --> 摄像头
  225. * --> 虚拟组织
  226. * --> 摄像头
  227. * --> 摄像头
  228. */
  229. String parentId = XmlUtil.getText(itemDevice, "ParentID");
  230. if (parentId != null) {
  231. if (parentId.contains("/")) {
  232. String lastParentId = parentId.substring(parentId.lastIndexOf("/") + 1);
  233. deviceChannel.setParentId(lastParentId);
  234. }else {
  235. deviceChannel.setParentId(parentId);
  236. }
  237. }else {
  238. if (deviceChannel.getChannelId().length() <= 10) { // 此时为行政区划, 上下级行政区划使用DeviceId关联
  239. deviceChannel.setParentId(deviceChannel.getChannelId().substring(0, deviceChannel.getChannelId().length() - 2));
  240. }else if (deviceChannel.getChannelId().length() == 20) {
  241. if (Integer.parseInt(deviceChannel.getChannelId().substring(10, 13)) == 216) { // 虚拟组织
  242. deviceChannel.setParentId(businessGroupID);
  243. }else if (Integer.parseInt(device.getDeviceId().substring(10, 13) )== 118) {//NVR 如果上级设备编号是NVR则直接将NVR的编号设置给通道的上级编号
  244. deviceChannel.setParentId(device.getDeviceId());
  245. }else if (deviceChannel.getCivilCode() != null) {
  246. // 设备, 无parentId的20位是使用CivilCode表示上级的设备,
  247. // 注:215 业务分组是需要有parentId的
  248. deviceChannel.setParentId(deviceChannel.getCivilCode());
  249. }
  250. }else {
  251. deviceChannel.setParentId(deviceChannel.getDeviceId());
  252. }
  253. }
  254. if (XmlUtil.getText(itemDevice, "SafetyWay") == null
  255. || XmlUtil.getText(itemDevice, "SafetyWay") == "") {
  256. deviceChannel.setSafetyWay(0);
  257. } else {
  258. deviceChannel.setSafetyWay(Integer.parseInt(XmlUtil.getText(itemDevice, "SafetyWay")));
  259. }
  260. if (XmlUtil.getText(itemDevice, "RegisterWay") == null
  261. || XmlUtil.getText(itemDevice, "RegisterWay") == "") {
  262. deviceChannel.setRegisterWay(1);
  263. } else {
  264. deviceChannel.setRegisterWay(Integer.parseInt(XmlUtil.getText(itemDevice, "RegisterWay")));
  265. }
  266. deviceChannel.setCertNum(XmlUtil.getText(itemDevice, "CertNum"));
  267. if (XmlUtil.getText(itemDevice, "Certifiable") == null
  268. || XmlUtil.getText(itemDevice, "Certifiable") == "") {
  269. deviceChannel.setCertifiable(0);
  270. } else {
  271. deviceChannel.setCertifiable(Integer.parseInt(XmlUtil.getText(itemDevice, "Certifiable")));
  272. }
  273. if (XmlUtil.getText(itemDevice, "ErrCode") == null
  274. || XmlUtil.getText(itemDevice, "ErrCode") == "") {
  275. deviceChannel.setErrCode(0);
  276. } else {
  277. deviceChannel.setErrCode(Integer.parseInt(XmlUtil.getText(itemDevice, "ErrCode")));
  278. }
  279. deviceChannel.setEndTime(XmlUtil.getText(itemDevice, "EndTime"));
  280. deviceChannel.setSecrecy(XmlUtil.getText(itemDevice, "Secrecy"));
  281. deviceChannel.setIpAddress(XmlUtil.getText(itemDevice, "IPAddress"));
  282. if (XmlUtil.getText(itemDevice, "Port") == null || XmlUtil.getText(itemDevice, "Port") == "") {
  283. deviceChannel.setPort(0);
  284. } else {
  285. deviceChannel.setPort(Integer.parseInt(XmlUtil.getText(itemDevice, "Port")));
  286. }
  287. deviceChannel.setPassword(XmlUtil.getText(itemDevice, "Password"));
  288. if (NumericUtil.isDouble(XmlUtil.getText(itemDevice, "Longitude"))) {
  289. deviceChannel.setLongitude(Double.parseDouble(XmlUtil.getText(itemDevice, "Longitude")));
  290. } else {
  291. deviceChannel.setLongitude(0.00);
  292. }
  293. if (NumericUtil.isDouble(XmlUtil.getText(itemDevice, "Latitude"))) {
  294. deviceChannel.setLatitude(Double.parseDouble(XmlUtil.getText(itemDevice, "Latitude")));
  295. } else {
  296. deviceChannel.setLatitude(0.00);
  297. }
  298. if (deviceChannel.getLongitude()*deviceChannel.getLatitude() > 0) {
  299. if ("WGS84".equals(device.getGeoCoordSys())) {
  300. deviceChannel.setLongitudeWgs84(deviceChannel.getLongitude());
  301. deviceChannel.setLatitudeWgs84(deviceChannel.getLatitude());
  302. Double[] position = Coordtransform.WGS84ToGCJ02(deviceChannel.getLongitude(), deviceChannel.getLatitude());
  303. deviceChannel.setLongitudeGcj02(position[0]);
  304. deviceChannel.setLatitudeGcj02(position[1]);
  305. }else if ("GCJ02".equals(device.getGeoCoordSys())) {
  306. deviceChannel.setLongitudeGcj02(deviceChannel.getLongitude());
  307. deviceChannel.setLatitudeGcj02(deviceChannel.getLatitude());
  308. Double[] position = Coordtransform.GCJ02ToWGS84(deviceChannel.getLongitude(), deviceChannel.getLatitude());
  309. deviceChannel.setLongitudeWgs84(position[0]);
  310. deviceChannel.setLatitudeWgs84(position[1]);
  311. }else {
  312. deviceChannel.setLongitudeGcj02(0.00);
  313. deviceChannel.setLatitudeGcj02(0.00);
  314. deviceChannel.setLongitudeWgs84(0.00);
  315. deviceChannel.setLatitudeWgs84(0.00);
  316. }
  317. }else {
  318. deviceChannel.setLongitudeGcj02(deviceChannel.getLongitude());
  319. deviceChannel.setLatitudeGcj02(deviceChannel.getLatitude());
  320. deviceChannel.setLongitudeWgs84(deviceChannel.getLongitude());
  321. deviceChannel.setLatitudeWgs84(deviceChannel.getLatitude());
  322. }
  323. if (XmlUtil.getText(itemDevice, "PTZType") == null || "".equals(XmlUtil.getText(itemDevice, "PTZType"))) {
  324. //兼容INFO中的信息
  325. Element info = itemDevice.element("Info");
  326. if(XmlUtil.getText(info, "PTZType") == null || "".equals(XmlUtil.getText(info, "PTZType"))){
  327. deviceChannel.setPTZType(0);
  328. }else{
  329. deviceChannel.setPTZType(Integer.parseInt(XmlUtil.getText(info, "PTZType")));
  330. }
  331. } else {
  332. deviceChannel.setPTZType(Integer.parseInt(XmlUtil.getText(itemDevice, "PTZType")));
  333. }
  334. deviceChannel.setHasAudio(true); // 默认含有音频,播放时再检查是否有音频及是否AAC
  335. return deviceChannel;
  336. }
  337. }