| | |
| | | package com.genersoft.iot.vmp.web; |
| | | |
| | | import com.alibaba.fastjson.JSON; |
| | | import com.alibaba.fastjson.JSONArray; |
| | | import com.alibaba.fastjson.JSONObject; |
| | | import com.genersoft.iot.vmp.common.StreamInfo; |
| | | import com.genersoft.iot.vmp.gb28181.bean.Device; |
| | | import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel; |
| | | import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander; |
| | | import com.genersoft.iot.vmp.media.zlm.ZLMRESTfulUtils; |
| | | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; |
| | | import com.genersoft.iot.vmp.storager.IVideoManagerStorager; |
| | | import com.genersoft.iot.vmp.vmanager.play.PlayController; |
| | | import com.genersoft.iot.vmp.vmanager.gb28181.play.PlayController; |
| | | import org.slf4j.Logger; |
| | | import org.slf4j.LoggerFactory; |
| | | import org.springframework.beans.factory.annotation.Autowired; |
| | | import org.springframework.beans.factory.annotation.Value; |
| | | import org.springframework.http.HttpStatus; |
| | | import org.springframework.http.ResponseEntity; |
| | | import org.springframework.web.bind.annotation.*; |
| | | import org.springframework.web.context.request.async.DeferredResult; |
| | | |
| | | /** |
| | | * 兼容LiveGBS的API:实时直播 |
| | | * API兼容:实时直播 |
| | | */ |
| | | @SuppressWarnings(value = {"rawtypes", "unchecked"}) |
| | | @CrossOrigin |
| | | @RestController |
| | | @RequestMapping(value = "/api/v1/stream") |
| | |
| | | @Autowired |
| | | private IVideoManagerStorager storager; |
| | | |
| | | private boolean closeWaitRTPInfo = false; |
| | | @Autowired |
| | | private IRedisCatchStorage redisCatchStorage; |
| | | |
| | | |
| | | // @Autowired |
| | | // private ZLMRESTfulUtils zlmresTfulUtils; |
| | | |
| | | |
| | | @Autowired |
| | | private ZLMRESTfulUtils zlmresTfulUtils; |
| | | private PlayController playController; |
| | | |
| | | /** |
| | | * 实时直播 - 开始直播 |
| | |
| | | * @return |
| | | */ |
| | | @RequestMapping(value = "/start") |
| | | private JSONObject start(String serial , |
| | | @RequestParam(required = false)Integer channel , |
| | | @RequestParam(required = false)String code, |
| | | @RequestParam(required = false)String cdn, |
| | | @RequestParam(required = false)String audio, |
| | | @RequestParam(required = false)String transport, |
| | | @RequestParam(required = false)String checkchannelstatus , |
| | | @RequestParam(required = false)String transportmode, |
| | | @RequestParam(required = false)String timeout |
| | | private DeferredResult<JSONObject> start(String serial , |
| | | @RequestParam(required = false)Integer channel , |
| | | @RequestParam(required = false)String code, |
| | | @RequestParam(required = false)String cdn, |
| | | @RequestParam(required = false)String audio, |
| | | @RequestParam(required = false)String transport, |
| | | @RequestParam(required = false)String checkchannelstatus , |
| | | @RequestParam(required = false)String transportmode, |
| | | @RequestParam(required = false)String timeout |
| | | |
| | | ){ |
| | | int getEncoding = closeWaitRTPInfo? 1: 0; |
| | | DeferredResult<JSONObject> resultDeferredResult = new DeferredResult<JSONObject>(); |
| | | Device device = storager.queryVideoDevice(serial); |
| | | |
| | | if (device == null ) { |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","device[ " + serial + " ]未找到"); |
| | | return result; |
| | | resultDeferredResult.setResult(result); |
| | | }else if (device.getOnline() == 0) { |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","device[ " + code + " ]offline"); |
| | | return result; |
| | | resultDeferredResult.setResult(result); |
| | | } |
| | | resultDeferredResult.onTimeout(()->{ |
| | | logger.info("播放等待超时"); |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","timeout"); |
| | | resultDeferredResult.setResult(result); |
| | | |
| | | // 清理RTP server |
| | | }); |
| | | |
| | | DeviceChannel deviceChannel = storager.queryChannel(serial, code); |
| | | if (deviceChannel == null) { |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","channel[ " + code + " ]未找到"); |
| | | return result; |
| | | resultDeferredResult.setResult(result); |
| | | }else if (deviceChannel.getStatus() == 0) { |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","channel[ " + code + " ]offline"); |
| | | return result; |
| | | resultDeferredResult.setResult(result); |
| | | } |
| | | DeferredResult<ResponseEntity<String>> play = playController.play(serial, code); |
| | | |
| | | // 查询是否已经在播放 |
| | | StreamInfo streamInfo = storager.queryPlayByDevice(device.getDeviceId(), code); |
| | | if (streamInfo == null) { |
| | | logger.debug("streamInfo 等于null, 重新点播"); |
| | | // streamInfo = cmder.playStreamCmd(device, code); |
| | | }else { |
| | | logger.debug("streamInfo 不等于null, 向流媒体查询是否正在推流"); |
| | | String streamId = streamInfo.getStreamId(); |
| | | JSONObject rtpInfo = zlmresTfulUtils.getRtpInfo(streamId); |
| | | if (rtpInfo.getBoolean("exist")) { |
| | | logger.debug("向流媒体查询正在推流, 直接返回: " + streamInfo.getRtsp()); |
| | | JSONObject result = new JSONObject(); |
| | | result.put("StreamID", streamInfo.getStreamId()); |
| | | result.put("DeviceID", device.getDeviceId()); |
| | | result.put("ChannelID", code); |
| | | result.put("ChannelName", deviceChannel.getName()); |
| | | result.put("ChannelCustomName", ""); |
| | | result.put("FLV", streamInfo.getFlv()); |
| | | result.put("WS_FLV", streamInfo.getWs_flv()); |
| | | result.put("RTMP", streamInfo.getRtmp()); |
| | | result.put("HLS", streamInfo.getHls()); |
| | | result.put("RTSP", streamInfo.getRtsp()); |
| | | result.put("CDN", ""); |
| | | result.put("SnapURL", ""); |
| | | result.put("Transport", device.getTransport()); |
| | | result.put("StartAt", ""); |
| | | result.put("Duration", ""); |
| | | result.put("SourceVideoCodecName", ""); |
| | | result.put("SourceVideoWidth", ""); |
| | | result.put("SourceVideoHeight", ""); |
| | | result.put("SourceVideoFrameRate", ""); |
| | | result.put("SourceAudioCodecName", ""); |
| | | result.put("SourceAudioSampleRate", ""); |
| | | result.put("AudioEnable", ""); |
| | | result.put("Ondemand", ""); |
| | | result.put("InBytes", ""); |
| | | result.put("InBitRate", ""); |
| | | result.put("OutBytes", ""); |
| | | result.put("NumOutputs", ""); |
| | | result.put("CascadeSize", ""); |
| | | result.put("RelaySize", ""); |
| | | result.put("ChannelPTZType", 0); |
| | | return result; |
| | | } else { |
| | | logger.debug("向流媒体查询没有推流, 重新点播"); |
| | | storager.stopPlay(streamInfo); |
| | | // streamInfo = cmder.playStreamCmd(device, code); |
| | | } |
| | | } |
| | | |
| | | if (logger.isDebugEnabled()) { |
| | | logger.debug(String.format("设备预览 API调用,deviceId:%s ,channelId:%s",serial, code)); |
| | | logger.debug("设备预览 API调用,streamId:"+streamInfo.getStreamId()); |
| | | } |
| | | boolean lockFlag = true; |
| | | long startTime = System.currentTimeMillis(); |
| | | while (lockFlag) { |
| | | try { |
| | | if (System.currentTimeMillis() - startTime > 10 * 1000) { |
| | | storager.stopPlay(streamInfo); |
| | | logger.info("播放等待超时"); |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","timeout"); |
| | | return result; |
| | | } else { |
| | | |
| | | StreamInfo streamInfoNow = storager.queryPlayByDevice(serial, code); |
| | | logger.debug("正在向流媒体查询"); |
| | | if (streamInfoNow != null && streamInfoNow.getFlv() != null) { |
| | | streamInfo = streamInfoNow; |
| | | logger.debug("向流媒体查询到: " + streamInfoNow.getRtsp()); |
| | | lockFlag = false; |
| | | continue; |
| | | } else { |
| | | Thread.sleep(2000); |
| | | continue; |
| | | } |
| | | } |
| | | } catch (InterruptedException e) { |
| | | e.printStackTrace(); |
| | | } |
| | | } |
| | | if(streamInfo!=null) { |
| | | play.setResultHandler((Object o)->{ |
| | | ResponseEntity<String> responseEntity = (ResponseEntity)o; |
| | | StreamInfo streamInfo = JSON.parseObject(responseEntity.getBody(), StreamInfo.class); |
| | | JSONObject result = new JSONObject(); |
| | | result.put("StreamID", streamInfo.getStreamId()); |
| | | result.put("DeviceID", device.getDeviceId()); |
| | |
| | | result.put("CascadeSize", ""); |
| | | result.put("RelaySize", ""); |
| | | result.put("ChannelPTZType", 0); |
| | | return result; |
| | | } else { |
| | | logger.warn("设备预览API调用失败!"); |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","调用失败"); |
| | | return result; |
| | | } |
| | | resultDeferredResult.setResult(result); |
| | | }); |
| | | return resultDeferredResult; |
| | | } |
| | | |
| | | /** |
| | |
| | | @RequestParam(required = false)String check_outputs |
| | | |
| | | ){ |
| | | StreamInfo streamInfo = storager.queryPlayByDevice(serial, code); |
| | | |
| | | StreamInfo streamInfo = redisCatchStorage.queryPlayByDevice(serial, code); |
| | | if (streamInfo == null) { |
| | | JSONObject result = new JSONObject(); |
| | | result.put("error","未找到流信息"); |
| | | return result; |
| | | } |
| | | cmder.streamByeCmd(streamInfo.getStreamId()); |
| | | storager.stopPlay(streamInfo); |
| | | cmder.streamByeCmd(serial, code); |
| | | redisCatchStorage.stopPlay(streamInfo); |
| | | storager.stopPlay(streamInfo.getDeviceID(), streamInfo.getChannelId()); |
| | | return null; |
| | | } |
| | | |