|  |  | 
 |  |  | package com.genersoft.iot.vmp.vmanager.gb28181.device; | 
 |  |  |  | 
 |  |  | import com.alibaba.fastjson.JSONObject; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.bean.Device; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.event.DeviceOffLineDetector; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.transmit.callback.RequestMessage; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander; | 
 |  |  | import com.genersoft.iot.vmp.service.IDeviceService; | 
 |  |  | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; | 
 |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorager; | 
 |  |  | import com.genersoft.iot.vmp.vmanager.bean.DeviceChannelTree; | 
 |  |  | import com.genersoft.iot.vmp.vmanager.bean.WVPResult; | 
 |  |  | import com.github.pagehelper.PageInfo; | 
 |  |  | import io.swagger.annotations.*; | 
 |  |  | import com.github.xiaoymin.knife4j.annotations.ApiOperationSupport; | 
 |  |  | import io.swagger.annotations.Api; | 
 |  |  | import io.swagger.annotations.ApiImplicitParam; | 
 |  |  | import io.swagger.annotations.ApiImplicitParams; | 
 |  |  | import io.swagger.annotations.ApiOperation; | 
 |  |  | import org.slf4j.Logger; | 
 |  |  | import org.slf4j.LoggerFactory; | 
 |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
 |  |  | 
 |  |  | import org.springframework.web.bind.annotation.*; | 
 |  |  | import org.springframework.web.context.request.async.DeferredResult; | 
 |  |  |  | 
 |  |  | import com.alibaba.fastjson.JSONObject; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.bean.Device; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.event.DeviceOffLineDetector; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder; | 
 |  |  | import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander; | 
 |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorager; | 
 |  |  |  | 
 |  |  | import javax.sip.message.Response; | 
 |  |  | import java.util.List; | 
 |  |  | import java.util.UUID; | 
 |  |  |  | 
 |  |  | @Api(tags = "国标设备查询", value = "国标设备查询") | 
 |  |  | @SuppressWarnings("rawtypes") | 
 |  |  | 
 |  |  |    @Autowired | 
 |  |  |    private DeviceOffLineDetector offLineDetector; | 
 |  |  |  | 
 |  |  |    @Autowired | 
 |  |  |    private IDeviceService deviceService; | 
 |  |  |  | 
 |  |  |    /** | 
 |  |  |     * 使用ID查询国标设备 | 
 |  |  |     * @param deviceId 国标ID | 
 |  |  | 
 |  |  |    @GetMapping("/devices/{deviceId}") | 
 |  |  |    public ResponseEntity<Device> devices(@PathVariable String deviceId){ | 
 |  |  |        | 
 |  |  |       if (logger.isDebugEnabled()) { | 
 |  |  |          logger.debug("查询视频设备API调用,deviceId:" + deviceId); | 
 |  |  |       } | 
 |  |  | //      if (logger.isDebugEnabled()) { | 
 |  |  | //         logger.debug("查询视频设备API调用,deviceId:" + deviceId); | 
 |  |  | //      } | 
 |  |  |        | 
 |  |  |       Device device = storager.queryVideoDevice(deviceId); | 
 |  |  |       return new ResponseEntity<>(device,HttpStatus.OK); | 
 |  |  | 
 |  |  |    @GetMapping("/devices") | 
 |  |  |    public PageInfo<Device> devices(int page, int count){ | 
 |  |  |        | 
 |  |  |       if (logger.isDebugEnabled()) { | 
 |  |  |          logger.debug("查询所有视频设备API调用"); | 
 |  |  |       } | 
 |  |  | //      if (logger.isDebugEnabled()) { | 
 |  |  | //         logger.debug("查询所有视频设备API调用"); | 
 |  |  | //      } | 
 |  |  |        | 
 |  |  |       return storager.queryVideoDeviceList(page, count); | 
 |  |  |    } | 
 |  |  | 
 |  |  |                                     @RequestParam(required = false) String query, | 
 |  |  |                                     @RequestParam(required = false) Boolean online, | 
 |  |  |                                     @RequestParam(required = false) Boolean channelType) { | 
 |  |  |       if (logger.isDebugEnabled()) { | 
 |  |  |          logger.debug("查询视频设备通道API调用"); | 
 |  |  |       } | 
 |  |  | //      if (logger.isDebugEnabled()) { | 
 |  |  | //         logger.debug("查询视频设备通道API调用"); | 
 |  |  | //      } | 
 |  |  |       if (StringUtils.isEmpty(query)) { | 
 |  |  |          query = null; | 
 |  |  |       } | 
 |  |  | 
 |  |  |     */ | 
 |  |  |    @ApiOperation("同步设备通道") | 
 |  |  |    @ApiImplicitParams({ | 
 |  |  |          @ApiImplicitParam(name="deviceId", value = "设备id", required = true ,dataTypeClass = String.class), | 
 |  |  |          @ApiImplicitParam(name="deviceId", value = "设备id", required = true, dataTypeClass = String.class), | 
 |  |  |    }) | 
 |  |  |    @PostMapping("/devices/{deviceId}/sync") | 
 |  |  |    public DeferredResult<ResponseEntity<Device>> devicesSync(@PathVariable String deviceId){ | 
 |  |  |        | 
 |  |  |       if (logger.isDebugEnabled()) { | 
 |  |  |       } | 
 |  |  |          logger.debug("设备通道信息同步API调用,deviceId:" + deviceId); | 
 |  |  |  | 
 |  |  |       } | 
 |  |  |       Device device = storager.queryVideoDevice(deviceId); | 
 |  |  |         cmder.catalogQuery(device, event -> { | 
 |  |  |          Response response = event.getResponse(); | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_CATALOG+deviceId); | 
 |  |  |          msg.setData(String.format("同步通道失败,错误码: %s, %s", response.getStatusCode(), response.getReasonPhrase())); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |       }); | 
 |  |  |         DeferredResult<ResponseEntity<Device>> result = new DeferredResult<ResponseEntity<Device>>(2*1000L); | 
 |  |  |       String key = DeferredResultHolder.CALLBACK_CMD_CATALOG + deviceId; | 
 |  |  |       String uuid = UUID.randomUUID().toString(); | 
 |  |  |       // 默认超时时间为30分钟 | 
 |  |  |       DeferredResult<ResponseEntity<Device>> result = new DeferredResult<ResponseEntity<Device>>(30*60*1000L); | 
 |  |  |       result.onTimeout(()->{ | 
 |  |  |          logger.warn(String.format("设备通道信息同步超时")); | 
 |  |  |          logger.warn("设备[{}]通道信息同步超时", deviceId); | 
 |  |  |          // 释放rtpserver | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_CATALOG+deviceId); | 
 |  |  |          msg.setData("Timeout"); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          WVPResult<Object> wvpResult = new WVPResult<>(); | 
 |  |  |          wvpResult.setCode(-1); | 
 |  |  |          wvpResult.setData(device); | 
 |  |  |          wvpResult.setMsg("更新超时"); | 
 |  |  |          msg.setData(wvpResult); | 
 |  |  |          resultHolder.invokeAllResult(msg); | 
 |  |  |  | 
 |  |  |       }); | 
 |  |  |         resultHolder.put(DeferredResultHolder.CALLBACK_CMD_CATALOG+deviceId, result); | 
 |  |  |       // 等待其他相同请求返回时一起返回 | 
 |  |  |       if (resultHolder.exist(key, null)) { | 
 |  |  |          return result; | 
 |  |  |       } | 
 |  |  |         cmder.catalogQuery(device, event -> { | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          WVPResult<Object> wvpResult = new WVPResult<>(); | 
 |  |  |          wvpResult.setCode(-1); | 
 |  |  |          wvpResult.setData(device); | 
 |  |  |          wvpResult.setMsg(String.format("同步通道失败,错误码: %s, %s", event.statusCode, event.msg)); | 
 |  |  |          msg.setData(wvpResult); | 
 |  |  |          resultHolder.invokeAllResult(msg); | 
 |  |  |       }); | 
 |  |  |  | 
 |  |  |         resultHolder.put(key, uuid, result); | 
 |  |  |         return result; | 
 |  |  |    } | 
 |  |  |  | 
 |  |  | 
 |  |  |                                       @RequestParam(required = false) String online, | 
 |  |  |                                       @RequestParam(required = false) Boolean channelType){ | 
 |  |  |  | 
 |  |  |       if (logger.isDebugEnabled()) { | 
 |  |  |          logger.debug("查询所有视频通道API调用"); | 
 |  |  |       } | 
 |  |  | //      if (logger.isDebugEnabled()) { | 
 |  |  | //         logger.debug("查询所有视频通道API调用"); | 
 |  |  | //      } | 
 |  |  |       DeviceChannel deviceChannel = storager.queryChannel(deviceId,channelId); | 
 |  |  |       if (deviceChannel == null) { | 
 |  |  |          PageInfo<DeviceChannel> deviceChannelPageResult = new PageInfo<>(); | 
 |  |  | 
 |  |  |    @ApiImplicitParams({ | 
 |  |  |          @ApiImplicitParam(name = "deviceId", value = "设备id", required = true, dataTypeClass = String.class), | 
 |  |  |          @ApiImplicitParam(name = "streamMode", value = "数据流传输模式, 取值:" + | 
 |  |  |                "UDP(udp传输),TCP-ACTIVE(tcp主动模式,暂不支持),TCP-PASSIVE(tcp被动模式)"), | 
 |  |  |                "UDP(udp传输),TCP-ACTIVE(tcp主动模式,暂不支持),TCP-PASSIVE(tcp被动模式)", dataTypeClass = String.class), | 
 |  |  |    }) | 
 |  |  |    @PostMapping("/transport/{deviceId}/{streamMode}") | 
 |  |  |    public ResponseEntity<PageInfo> updateTransport(@PathVariable String deviceId, @PathVariable String streamMode){ | 
 |  |  | 
 |  |  |       device.setStreamMode(streamMode); | 
 |  |  |       storager.updateDevice(device); | 
 |  |  |       return new ResponseEntity<>(null,HttpStatus.OK); | 
 |  |  |    } | 
 |  |  |  | 
 |  |  |    /** | 
 |  |  |     * 更新设备信息 | 
 |  |  |     * @param device 设备信息 | 
 |  |  |     * @return | 
 |  |  |     */ | 
 |  |  |    @ApiOperation("更新设备信息") | 
 |  |  |    @ApiImplicitParams({ | 
 |  |  |          @ApiImplicitParam(name = "device", value = "设备信息", required = true, dataTypeClass = Device.class) | 
 |  |  |    }) | 
 |  |  |    @PostMapping("/device/update/") | 
 |  |  |    public ResponseEntity<WVPResult<String>> updateDevice(Device device){ | 
 |  |  |  | 
 |  |  |       if (device != null && device.getDeviceId() != null) { | 
 |  |  |          Device deviceInStore = storager.queryVideoDevice(device.getDeviceId()); | 
 |  |  |          if (!StringUtils.isEmpty(device.getName())) deviceInStore.setName(device.getName()); | 
 |  |  |          if (!StringUtils.isEmpty(device.getCharset())) deviceInStore.setCharset(device.getCharset()); | 
 |  |  |          if (!StringUtils.isEmpty(device.getMediaServerId())) deviceInStore.setMediaServerId(device.getMediaServerId()); | 
 |  |  |  | 
 |  |  |          if ((deviceInStore.getSubscribeCycleForCatalog() <=0 && device.getSubscribeCycleForCatalog() > 0) | 
 |  |  |                || deviceInStore.getSubscribeCycleForCatalog() != device.getSubscribeCycleForCatalog()) { | 
 |  |  |             deviceInStore.setSubscribeCycleForCatalog(device.getSubscribeCycleForCatalog()); | 
 |  |  |             // 开启订阅 | 
 |  |  |             deviceService.addCatalogSubscribe(deviceInStore); | 
 |  |  |          } | 
 |  |  |          if (deviceInStore.getSubscribeCycleForCatalog() > 0 && device.getSubscribeCycleForCatalog() <= 0) { | 
 |  |  |             deviceInStore.setSubscribeCycleForCatalog(device.getSubscribeCycleForCatalog()); | 
 |  |  |             // 取消订阅 | 
 |  |  |             deviceService.removeCatalogSubscribe(deviceInStore); | 
 |  |  |          } | 
 |  |  |  | 
 |  |  |          storager.updateDevice(deviceInStore); | 
 |  |  |          cmder.deviceInfoQuery(deviceInStore); | 
 |  |  |       } | 
 |  |  |       WVPResult<String> result = new WVPResult<>(); | 
 |  |  |       result.setCode(0); | 
 |  |  |       result.setMsg("success"); | 
 |  |  |       return new ResponseEntity<>(result,HttpStatus.OK); | 
 |  |  |    } | 
 |  |  |  | 
 |  |  |    /** | 
 |  |  | 
 |  |  |          logger.debug("设备状态查询API调用"); | 
 |  |  |       } | 
 |  |  |       Device device = storager.queryVideoDevice(deviceId); | 
 |  |  |       String uuid = UUID.randomUUID().toString(); | 
 |  |  |       String key = DeferredResultHolder.CALLBACK_CMD_DEVICESTATUS + deviceId; | 
 |  |  |       cmder.deviceStatusQuery(device, event -> { | 
 |  |  |          Response response = event.getResponse(); | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_DEVICESTATUS + deviceId); | 
 |  |  |          msg.setData(String.format("获取设备状态失败,错误码: %s, %s", response.getStatusCode(), response.getReasonPhrase())); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setData(String.format("获取设备状态失败,错误码: %s, %s", event.statusCode, event.msg)); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |       }); | 
 |  |  |         DeferredResult<ResponseEntity<String>> result = new DeferredResult<ResponseEntity<String>>(2*1000L); | 
 |  |  | 
 |  |  |          logger.warn(String.format("获取设备状态超时")); | 
 |  |  |          // 释放rtpserver | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_DEVICESTATUS + deviceId); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setData("Timeout. Device did not response to this command."); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |       }); | 
 |  |  |       resultHolder.put(DeferredResultHolder.CALLBACK_CMD_DEVICESTATUS + deviceId, result); | 
 |  |  |       resultHolder.put(DeferredResultHolder.CALLBACK_CMD_DEVICESTATUS + deviceId, uuid, result); | 
 |  |  |       return result; | 
 |  |  |    } | 
 |  |  |  | 
 |  |  | 
 |  |  |          logger.debug("设备报警查询API调用"); | 
 |  |  |       } | 
 |  |  |       Device device = storager.queryVideoDevice(deviceId); | 
 |  |  |       String key = DeferredResultHolder.CALLBACK_CMD_ALARM + deviceId; | 
 |  |  |       String uuid = UUID.randomUUID().toString(); | 
 |  |  |       cmder.alarmInfoQuery(device, startPriority, endPriority, alarmMethod, alarmType, startTime, endTime, event -> { | 
 |  |  |          Response response = event.getResponse(); | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_ALARM + deviceId); | 
 |  |  |          msg.setData(String.format("设备报警查询失败,错误码: %s, %s", response.getStatusCode(), response.getReasonPhrase())); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setData(String.format("设备报警查询失败,错误码: %s, %s",event.statusCode, event.msg)); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |       }); | 
 |  |  |         DeferredResult<ResponseEntity<String>> result = new DeferredResult<ResponseEntity<String >> (3 * 1000L); | 
 |  |  | 
 |  |  |          logger.warn(String.format("设备报警查询超时")); | 
 |  |  |          // 释放rtpserver | 
 |  |  |          RequestMessage msg = new RequestMessage(); | 
 |  |  |          msg.setId(DeferredResultHolder.CALLBACK_CMD_ALARM + deviceId); | 
 |  |  |          msg.setId(uuid); | 
 |  |  |          msg.setKey(key); | 
 |  |  |          msg.setData("设备报警查询超时"); | 
 |  |  |          resultHolder.invokeResult(msg); | 
 |  |  |       }); | 
 |  |  |       resultHolder.put(DeferredResultHolder.CALLBACK_CMD_ALARM + deviceId, result); | 
 |  |  |       resultHolder.put(DeferredResultHolder.CALLBACK_CMD_ALARM + deviceId, uuid, result); | 
 |  |  |       return result; | 
 |  |  |    } | 
 |  |  |  | 
 |  |  |  | 
 |  |  |    @GetMapping("/{deviceId}/tree") | 
 |  |  |    @ApiOperation(value = "通道树形结构", notes = "通道树形结构") | 
 |  |  |    public WVPResult<List<DeviceChannelTree>> tree(@PathVariable String deviceId) { | 
 |  |  |       return WVPResult.Data(storager.tree(deviceId)); | 
 |  |  |    } | 
 |  |  | } |