|  |  |  | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.bean.ResultForOnPublish; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.MediaArrivalEvent; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.MediaDepartureEvent; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.MediaNotFoundEvent; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.service.IMediaServerService; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.HookType; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.MediaServer; | 
|---|
|  |  |  | 
|---|
|  |  |  | logger.info("[ZLM HOOK]流无人观看:{}->{}->{}/{}", param.getMediaServerId(), param.getSchema(), | 
|---|
|  |  |  | param.getApp(), param.getStream()); | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | // 国标类型的流 | 
|---|
|  |  |  | if ("rtp".equals(param.getApp())) { | 
|---|
|  |  |  | ret.put("close", userSetting.getStreamOnDemand()); | 
|---|
|  |  |  | // 国标流, 点播/录像回放/录像下载 | 
|---|
|  |  |  | InviteInfo inviteInfo = inviteStreamService.getInviteInfoByStream(null, param.getStream()); | 
|---|
|  |  |  | // 点播 | 
|---|
|  |  |  | if (inviteInfo != null) { | 
|---|
|  |  |  | // 录像下载 | 
|---|
|  |  |  | if (inviteInfo.getType() == InviteSessionType.DOWNLOAD) { | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 收到无人观看说明流也没有在往上级推送 | 
|---|
|  |  |  | if (redisCatchStorage.isChannelSendingRTP(inviteInfo.getChannelId())) { | 
|---|
|  |  |  | List<SendRtpItem> sendRtpItems = redisCatchStorage.querySendRTPServerByChannelId( | 
|---|
|  |  |  | inviteInfo.getChannelId()); | 
|---|
|  |  |  | if (!sendRtpItems.isEmpty()) { | 
|---|
|  |  |  | for (SendRtpItem sendRtpItem : sendRtpItems) { | 
|---|
|  |  |  | ParentPlatform parentPlatform = storager.queryParentPlatByServerGBId(sendRtpItem.getPlatformId()); | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | commanderFroPlatform.streamByeCmd(parentPlatform, sendRtpItem.getCallId()); | 
|---|
|  |  |  | } catch (SipException | InvalidArgumentException | ParseException e) { | 
|---|
|  |  |  | logger.error("[命令发送失败] 国标级联 发送BYE: {}", e.getMessage()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | redisCatchStorage.deleteSendRTPServer(parentPlatform.getServerGBId(), sendRtpItem.getChannelId(), | 
|---|
|  |  |  | sendRtpItem.getCallId(), sendRtpItem.getStream()); | 
|---|
|  |  |  | if (InviteStreamType.PUSH == sendRtpItem.getPlayType()) { | 
|---|
|  |  |  | MessageForPushChannel messageForPushChannel = MessageForPushChannel.getInstance(0, | 
|---|
|  |  |  | sendRtpItem.getApp(), sendRtpItem.getStream(), sendRtpItem.getChannelId(), | 
|---|
|  |  |  | sendRtpItem.getPlatformId(), parentPlatform.getName(), userSetting.getServerId(), sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | messageForPushChannel.setPlatFormIndex(parentPlatform.getId()); | 
|---|
|  |  |  | redisCatchStorage.sendPlatformStopPlayMsg(messageForPushChannel); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Device device = deviceService.getDevice(inviteInfo.getDeviceId()); | 
|---|
|  |  |  | if (device != null) { | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | // 多查询一次防止已经被处理了 | 
|---|
|  |  |  | InviteInfo info = inviteStreamService.getInviteInfo(inviteInfo.getType(), | 
|---|
|  |  |  | inviteInfo.getDeviceId(), inviteInfo.getChannelId(), inviteInfo.getStream()); | 
|---|
|  |  |  | if (info != null) { | 
|---|
|  |  |  | cmder.streamByeCmd(device, inviteInfo.getChannelId(), | 
|---|
|  |  |  | inviteInfo.getStream(), null); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | logger.info("[无人观看] 未找到设备的点播信息: {}, 流:{}", inviteInfo.getDeviceId(), param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException | | 
|---|
|  |  |  | SsrcTransactionNotFoundException e) { | 
|---|
|  |  |  | logger.error("[无人观看]点播, 发送BYE失败 {}", e.getMessage()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | logger.info("[无人观看] 未找到设备: {},流:{}", inviteInfo.getDeviceId(), param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | inviteStreamService.removeInviteInfo(inviteInfo.getType(), inviteInfo.getDeviceId(), | 
|---|
|  |  |  | inviteInfo.getChannelId(), inviteInfo.getStream()); | 
|---|
|  |  |  | storager.stopPlay(inviteInfo.getDeviceId(), inviteInfo.getChannelId()); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | SendRtpItem sendRtpItem = redisCatchStorage.querySendRTPServer(null, null, param.getStream(), null); | 
|---|
|  |  |  | if (sendRtpItem != null && "talk".equals(sendRtpItem.getApp())) { | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else if ("talk".equals(param.getApp()) || "broadcast".equals(param.getApp())) { | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | // 非国标流 推流/拉流代理 | 
|---|
|  |  |  | // 拉流代理 | 
|---|
|  |  |  | StreamProxyItem streamProxyItem = streamProxyService.getStreamProxyByAppAndStream(param.getApp(), param.getStream()); | 
|---|
|  |  |  | if (streamProxyItem != null) { | 
|---|
|  |  |  | if (streamProxyItem.isEnableRemoveNoneReader()) { | 
|---|
|  |  |  | // 无人观看自动移除 | 
|---|
|  |  |  | ret.put("close", true); | 
|---|
|  |  |  | streamProxyService.del(param.getApp(), param.getStream()); | 
|---|
|  |  |  | String url = streamProxyItem.getUrl() != null ? streamProxyItem.getUrl() : streamProxyItem.getSrcUrl(); | 
|---|
|  |  |  | logger.info("[{}/{}]<-[{}] 拉流代理无人观看已经移除", param.getApp(), param.getStream(), url); | 
|---|
|  |  |  | } else if (streamProxyItem.isEnableDisableNoneReader()) { | 
|---|
|  |  |  | // 无人观看停用 | 
|---|
|  |  |  | ret.put("close", true); | 
|---|
|  |  |  | // 修改数据 | 
|---|
|  |  |  | streamProxyService.stop(param.getApp(), param.getStream()); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | // 无人观看不做处理 | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | boolean close = mediaService.closeStreamOnNoneReader(param.getMediaServerId(), param.getApp(), param.getStream(), param.getSchema()); | 
|---|
|  |  |  | ret.put("code", close); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @ResponseBody | 
|---|
|  |  |  | @PostMapping(value = "/on_stream_not_found", produces = "application/json;charset=UTF-8") | 
|---|
|  |  |  | public DeferredResult<HookResult> onStreamNotFound(@RequestBody OnStreamNotFoundHookParam param) { | 
|---|
|  |  |  | public HookResult onStreamNotFound(@RequestBody OnStreamNotFoundHookParam param) { | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 流未找到:{}->{}->{}/{}", param.getMediaServerId(), param.getSchema(), param.getApp(), param.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | DeferredResult<HookResult> defaultResult = new DeferredResult<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaServer mediaInfo = mediaServerService.getOne(param.getMediaServerId()); | 
|---|
|  |  |  | if (!userSetting.isAutoApplyPlay() || mediaInfo == null) { | 
|---|
|  |  |  | defaultResult.setResult(new HookResult(ErrorCode.ERROR404.getCode(), ErrorCode.ERROR404.getMsg())); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); | 
|---|
|  |  |  | if (!userSetting.isAutoApplyPlay() || mediaServer == null) { | 
|---|
|  |  |  | return HookResult.SUCCESS(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if ("rtp".equals(param.getApp())) { | 
|---|
|  |  |  | String[] s = param.getStream().split("_"); | 
|---|
|  |  |  | if ((s.length != 2 && s.length != 4)) { | 
|---|
|  |  |  | defaultResult.setResult(HookResult.SUCCESS()); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String deviceId = s[0]; | 
|---|
|  |  |  | String channelId = s[1]; | 
|---|
|  |  |  | Device device = redisCatchStorage.getDevice(deviceId); | 
|---|
|  |  |  | if (device == null || !device.isOnLine()) { | 
|---|
|  |  |  | defaultResult.setResult(new HookResult(ErrorCode.ERROR404.getCode(), ErrorCode.ERROR404.getMsg())); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | DeviceChannel deviceChannel = storager.queryChannel(deviceId, channelId); | 
|---|
|  |  |  | if (deviceChannel == null) { | 
|---|
|  |  |  | defaultResult.setResult(new HookResult(ErrorCode.ERROR404.getCode(), ErrorCode.ERROR404.getMsg())); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (s.length == 2) { | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 预览流未找到, 发起自动点播:{}->{}->{}/{}", param.getMediaServerId(), param.getSchema(), param.getApp(), param.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | RequestMessage msg = new RequestMessage(); | 
|---|
|  |  |  | String key = DeferredResultHolder.CALLBACK_CMD_PLAY + deviceId + channelId; | 
|---|
|  |  |  | boolean exist = resultHolder.exist(key, null); | 
|---|
|  |  |  | msg.setKey(key); | 
|---|
|  |  |  | String uuid = UUID.randomUUID().toString(); | 
|---|
|  |  |  | msg.setId(uuid); | 
|---|
|  |  |  | DeferredResult<HookResult> result = new DeferredResult<>(userSetting.getPlayTimeout().longValue()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | result.onTimeout(() -> { | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 预览流自动点播, 等待超时"); | 
|---|
|  |  |  | msg.setData(new HookResult(ErrorCode.ERROR100.getCode(), "点播超时")); | 
|---|
|  |  |  | resultHolder.invokeAllResult(msg); | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId); | 
|---|
|  |  |  | storager.stopPlay(deviceId, channelId); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | resultHolder.put(key, uuid, result); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!exist) { | 
|---|
|  |  |  | playService.play(mediaInfo, deviceId, channelId, null, (code, message, data) -> { | 
|---|
|  |  |  | msg.setData(new HookResult(code, message)); | 
|---|
|  |  |  | resultHolder.invokeResult(msg); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } else if (s.length == 4) { | 
|---|
|  |  |  | // 此时为录像回放, 录像回放格式为> 设备ID_通道ID_开始时间_结束时间 | 
|---|
|  |  |  | String startTimeStr = s[2]; | 
|---|
|  |  |  | String endTimeStr = s[3]; | 
|---|
|  |  |  | if (startTimeStr == null || endTimeStr == null || startTimeStr.length() != 14 || endTimeStr.length() != 14) { | 
|---|
|  |  |  | defaultResult.setResult(HookResult.SUCCESS()); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String startTime = DateUtil.urlToyyyy_MM_dd_HH_mm_ss(startTimeStr); | 
|---|
|  |  |  | String endTime = DateUtil.urlToyyyy_MM_dd_HH_mm_ss(endTimeStr); | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 回放流未找到, 发起自动点播:{}->{}->{}/{}-{}-{}", | 
|---|
|  |  |  | param.getMediaServerId(), param.getSchema(), | 
|---|
|  |  |  | param.getApp(), param.getStream(), | 
|---|
|  |  |  | startTime, endTime | 
|---|
|  |  |  | ); | 
|---|
|  |  |  | RequestMessage msg = new RequestMessage(); | 
|---|
|  |  |  | String key = DeferredResultHolder.CALLBACK_CMD_PLAYBACK + deviceId + channelId; | 
|---|
|  |  |  | boolean exist = resultHolder.exist(key, null); | 
|---|
|  |  |  | msg.setKey(key); | 
|---|
|  |  |  | String uuid = UUID.randomUUID().toString(); | 
|---|
|  |  |  | msg.setId(uuid); | 
|---|
|  |  |  | DeferredResult<HookResult> result = new DeferredResult<>(userSetting.getPlayTimeout().longValue()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | result.onTimeout(() -> { | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 回放流自动点播, 等待超时"); | 
|---|
|  |  |  | // 释放rtpserver | 
|---|
|  |  |  | msg.setData(new HookResult(ErrorCode.ERROR100.getCode(), "点播超时")); | 
|---|
|  |  |  | resultHolder.invokeResult(msg); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | resultHolder.put(key, uuid, result); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!exist) { | 
|---|
|  |  |  | SSRCInfo ssrcInfo = mediaServerService.openRTPServer(mediaInfo, param.getStream(), null, | 
|---|
|  |  |  | device.isSsrcCheck(), true, 0, false, false, device.getStreamModeForParam()); | 
|---|
|  |  |  | playService.playBack(mediaInfo, ssrcInfo, deviceId, channelId, startTime, endTime, (code, message, data) -> { | 
|---|
|  |  |  | msg.setData(new HookResult(code, message)); | 
|---|
|  |  |  | resultHolder.invokeResult(msg); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | defaultResult.setResult(HookResult.SUCCESS()); | 
|---|
|  |  |  | return defaultResult; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | // 拉流代理 | 
|---|
|  |  |  | StreamProxyItem streamProxyByAppAndStream = streamProxyService.getStreamProxyByAppAndStream(param.getApp(), param.getStream()); | 
|---|
|  |  |  | if (streamProxyByAppAndStream != null && streamProxyByAppAndStream.isEnableDisableNoneReader()) { | 
|---|
|  |  |  | streamProxyService.start(param.getApp(), param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | DeferredResult<HookResult> result = new DeferredResult<>(); | 
|---|
|  |  |  | result.setResult(HookResult.SUCCESS()); | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | MediaNotFoundEvent mediaNotFoundEvent = MediaNotFoundEvent.getInstance(this, param, mediaServer); | 
|---|
|  |  |  | applicationEventPublisher.publishEvent(mediaNotFoundEvent); | 
|---|
|  |  |  | return HookResult.SUCCESS(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | 
|---|
|  |  |  | ZLMServerConfig zlmServerConfig = JSON.to(ZLMServerConfig.class, jsonObject); | 
|---|
|  |  |  | zlmServerConfig.setIp(request.getRemoteAddr()); | 
|---|
|  |  |  | logger.info("[ZLM HOOK] zlm 启动 " + zlmServerConfig.getGeneralMediaServerId()); | 
|---|
|  |  |  | taskExecutor.execute(() -> { | 
|---|
|  |  |  | List<ZlmHttpHookSubscribe.Event> subscribes = this.subscribe.getSubscribes(HookType.on_server_started); | 
|---|
|  |  |  | if (subscribes != null && !subscribes.isEmpty()) { | 
|---|
|  |  |  | for (ZlmHttpHookSubscribe.Event subscribe : subscribes) { | 
|---|
|  |  |  | subscribe.response(null, zlmServerConfig); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | HookZlmServerStartEvent event = new HookZlmServerStartEvent(this); | 
|---|
|  |  |  | MediaServer mediaServerItem = mediaServerService.getOne(zlmServerConfig.getMediaServerId()); | 
|---|