|  |  |  | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.InviteInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.InviteSessionType; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.StreamInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.VideoManagerConstants; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.UserSetting; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.exception.SsrcTransactionNotFoundException; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.bean.*; | 
|---|
|  |  |  | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.StreamProxyItem; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.hook.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.MessageForPushChannel; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.ErrorCode; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.OtherRtpSendInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.StreamContent; | 
|---|
|  |  |  | import org.slf4j.Logger; | 
|---|
|  |  |  | import org.slf4j.LoggerFactory; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Qualifier; | 
|---|
|  |  |  | import org.springframework.data.redis.core.RedisTemplate; | 
|---|
|  |  |  | import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; | 
|---|
|  |  |  | import org.springframework.util.ObjectUtils; | 
|---|
|  |  |  | import org.springframework.web.bind.annotation.*; | 
|---|
|  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ThreadPoolTaskExecutor taskExecutor; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private RedisTemplate<Object, Object> redisTemplate; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 服务器定时上报时间,上报间隔可配置,默认10s上报一次 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | 
|---|
|  |  |  | @PostMapping(value = "/on_server_keepalive", produces = "application/json;charset=UTF-8") | 
|---|
|  |  |  | public HookResult onServerKeepalive(@RequestBody OnServerKeepaliveHookParam param) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //        logger.info("[ZLM HOOK] 收到zlm心跳:" + param.getMediaServerId()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | taskExecutor.execute(() -> { | 
|---|
|  |  |  | List<ZlmHttpHookSubscribe.Event> subscribes = this.subscribe.getSubscribes(HookType.on_server_keepalive); | 
|---|
|  |  |  | JSONObject json = (JSONObject) JSON.toJSON(param); | 
|---|
|  |  |  | if (subscribes != null && subscribes.size() > 0) { | 
|---|
|  |  |  | for (ZlmHttpHookSubscribe.Event subscribe : subscribes) { | 
|---|
|  |  |  | subscribe.response(null, json); | 
|---|
|  |  |  | subscribe.response(null, param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (subscribe != null) { | 
|---|
|  |  |  | MediaServerItem mediaInfo = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | if (mediaInfo != null) { | 
|---|
|  |  |  | subscribe.response(mediaInfo, json); | 
|---|
|  |  |  | subscribe.response(mediaInfo, param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (userSetting.getPushAuthority()) { | 
|---|
|  |  |  | // 推流鉴权 | 
|---|
|  |  |  | if (param.getParams() == null) { | 
|---|
|  |  |  | logger.info("推流鉴权失败: 缺少不要参数:sign=md5(user表的pushKey)"); | 
|---|
|  |  |  | logger.info("推流鉴权失败: 缺少必要参数:sign=md5(user表的pushKey)"); | 
|---|
|  |  |  | return new HookResultForOnPublish(401, "Unauthorized"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Map<String, String> paramMap = urlParamToMap(param.getParams()); | 
|---|
|  |  |  | String sign = paramMap.get("sign"); | 
|---|
|  |  |  | if (sign == null) { | 
|---|
|  |  |  | logger.info("推流鉴权失败: 缺少不要参数:sign=md5(user表的pushKey)"); | 
|---|
|  |  |  | logger.info("推流鉴权失败: 缺少必要参数:sign=md5(user表的pushKey)"); | 
|---|
|  |  |  | return new HookResultForOnPublish(401, "Unauthorized"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 推流自定义播放鉴权码 | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | HookResultForOnPublish result = HookResultForOnPublish.SUCCESS(); | 
|---|
|  |  |  | if (!"rtp".equals(param.getApp())) { | 
|---|
|  |  |  | result.setEnable_audio(true); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | result.setEnable_audio(true); | 
|---|
|  |  |  | taskExecutor.execute(() -> { | 
|---|
|  |  |  | ZlmHttpHookSubscribe.Event subscribe = this.subscribe.sendNotify(HookType.on_publish, json); | 
|---|
|  |  |  | if (subscribe != null) { | 
|---|
|  |  |  | if (mediaInfo != null) { | 
|---|
|  |  |  | subscribe.response(mediaInfo, json); | 
|---|
|  |  |  | subscribe.response(mediaInfo, param); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | new HookResultForOnPublish(1, "zlm not register"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | // 如果是录像下载就设置视频间隔十秒 | 
|---|
|  |  |  | if (ssrcTransactionForAll.get(0).getType() == InviteSessionType.DOWNLOAD) { | 
|---|
|  |  |  | result.setMp4_max_second(10); | 
|---|
|  |  |  | result.setEnable_audio(true); | 
|---|
|  |  |  | result.setEnable_mp4(true); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (param.getApp().equalsIgnoreCase("rtp")) { | 
|---|
|  |  |  | String receiveKey = VideoManagerConstants.WVP_OTHER_RECEIVE_RTP_INFO + userSetting.getServerId() + "_" + param.getStream(); | 
|---|
|  |  |  | OtherRtpSendInfo otherRtpSendInfo = (OtherRtpSendInfo)redisTemplate.opsForValue().get(receiveKey); | 
|---|
|  |  |  | if (otherRtpSendInfo != null) { | 
|---|
|  |  |  | result.setEnable_mp4(true); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | logger.info("[ZLM HOOK]推流鉴权 响应:{}->{}->>>>{}", param.getMediaServerId(), param, result); | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (subscribe != null) { | 
|---|
|  |  |  | subscribe.response(mediaInfo, json); | 
|---|
|  |  |  | subscribe.response(mediaInfo, param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<OnStreamChangedHookParam.MediaTrack> tracks = param.getTracks(); | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | GbStream gbStream = storager.getGbStream(param.getApp(), param.getStream()); | 
|---|
|  |  |  | if (gbStream != null) { | 
|---|
|  |  |  | eventPublisher.catalogEventPublishForStream(null, gbStream, param.isRegist()?CatalogEvent.ON:CatalogEvent.OFF); | 
|---|
|  |  |  | if (userSetting.isUsePushingAsStatus()) { | 
|---|
|  |  |  | eventPublisher.catalogEventPublishForStream(null, gbStream, param.isRegist()?CatalogEvent.ON:CatalogEvent.OFF); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (type != null) { | 
|---|
|  |  |  | // 发送流变化redis消息 | 
|---|
|  |  |  | 
|---|
|  |  |  | @PostMapping(value = "/on_stream_none_reader", produces = "application/json;charset=UTF-8") | 
|---|
|  |  |  | public JSONObject onStreamNoneReader(@RequestBody OnStreamNoneReaderHookParam param) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | logger.info("[ZLM HOOK]流无人观看:{]->{}->{}/{}" + param.getMediaServerId(), param.getSchema(), | 
|---|
|  |  |  | 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()); | 
|---|
|  |  |  | // 国标流, 点播/录像回放/录像下载 | 
|---|
|  |  |  | //            StreamInfo streamInfoForPlayCatch = redisCatchStorage.queryPlayByStreamId(param.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 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.querySendRTPServerByChnnelId( | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | redisCatchStorage.deleteSendRTPServer(parentPlatform.getServerGBId(), sendRtpItem.getChannelId(), | 
|---|
|  |  |  | sendRtpItem.getCallId(), sendRtpItem.getStreamId()); | 
|---|
|  |  |  | if (InviteStreamType.PUSH == sendRtpItem.getPlayType()) { | 
|---|
|  |  |  | MessageForPushChannel messageForPushChannel = MessageForPushChannel.getInstance(0, | 
|---|
|  |  |  | sendRtpItem.getApp(), sendRtpItem.getStreamId(), 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 { | 
|---|
|  |  |  | if (inviteStreamService.getInviteInfo(inviteInfo.getType(), inviteInfo.getDeviceId(), inviteInfo.getChannelId(), inviteInfo.getStream()) != null) { | 
|---|
|  |  |  | InviteInfo info = inviteStreamService.getInviteInfo(inviteInfo.getType(), | 
|---|
|  |  |  | inviteInfo.getDeviceId(), inviteInfo.getChannelId(), inviteInfo.getStream()); | 
|---|
|  |  |  | if (info != null) { | 
|---|
|  |  |  | cmder.streamByeCmd(device, inviteInfo.getChannelId(), | 
|---|
|  |  |  | inviteInfo.getStream(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | storager.stopPlay(inviteInfo.getDeviceId(), inviteInfo.getChannelId()); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 录像回放 | 
|---|
|  |  |  | StreamInfo streamInfoForPlayBackCatch = redisCatchStorage.queryPlayback(null, null, | 
|---|
|  |  |  | param.getStream(), null); | 
|---|
|  |  |  | if (streamInfoForPlayBackCatch != null) { | 
|---|
|  |  |  | if (streamInfoForPlayBackCatch.isPause()) { | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | Device device = deviceService.getDevice(streamInfoForPlayBackCatch.getDeviceID()); | 
|---|
|  |  |  | if (device != null) { | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | cmder.streamByeCmd(device, streamInfoForPlayBackCatch.getChannelId(), | 
|---|
|  |  |  | streamInfoForPlayBackCatch.getStream(), null); | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException | | 
|---|
|  |  |  | SsrcTransactionNotFoundException e) { | 
|---|
|  |  |  | logger.error("[无人观看]回放, 发送BYE失败 {}", e.getMessage()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | redisCatchStorage.stopPlayback(streamInfoForPlayBackCatch.getDeviceID(), | 
|---|
|  |  |  | streamInfoForPlayBackCatch.getChannelId(), streamInfoForPlayBackCatch.getStream(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 录像下载 | 
|---|
|  |  |  | StreamInfo streamInfoForDownload = redisCatchStorage.queryDownload(null, null, | 
|---|
|  |  |  | param.getStream(), null); | 
|---|
|  |  |  | // 进行录像下载时无人观看不断流 | 
|---|
|  |  |  | if (streamInfoForDownload != null) { | 
|---|
|  |  |  | ret.put("close", false); | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | // 非国标流 推流/拉流代理 | 
|---|
|  |  |  | // 拉流代理 | 
|---|
|  |  |  | StreamProxyItem streamProxyItem = streamProxyService.getStreamProxyByAppAndStream(param.getApp(), param.getStream()); | 
|---|
|  |  |  | if (streamProxyItem != null) { | 
|---|
|  |  |  | if (streamProxyItem.isEnable_remove_none_reader()) { | 
|---|
|  |  |  | if (streamProxyItem.isEnableRemoveNoneReader()) { | 
|---|
|  |  |  | // 无人观看自动移除 | 
|---|
|  |  |  | ret.put("close", true); | 
|---|
|  |  |  | streamProxyService.del(param.getApp(), param.getStream()); | 
|---|
|  |  |  | String url = streamProxyItem.getUrl() != null ? streamProxyItem.getUrl() : streamProxyItem.getSrc_url(); | 
|---|
|  |  |  | String url = streamProxyItem.getUrl() != null ? streamProxyItem.getUrl() : streamProxyItem.getSrcUrl(); | 
|---|
|  |  |  | logger.info("[{}/{}]<-[{}] 拉流代理无人观看已经移除", param.getApp(), param.getStream(), url); | 
|---|
|  |  |  | } else if (streamProxyItem.isEnable_disable_none_reader()) { | 
|---|
|  |  |  | } else if (streamProxyItem.isEnableDisableNoneReader()) { | 
|---|
|  |  |  | // 无人观看停用 | 
|---|
|  |  |  | ret.put("close", true); | 
|---|
|  |  |  | // 修改数据 | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 推流具有主动性,暂时不做处理 | 
|---|
|  |  |  | // TODO 推流具有主动性,暂时不做处理 | 
|---|
|  |  |  | //         StreamPushItem streamPushItem = streamPushService.getPush(app, streamId); | 
|---|
|  |  |  | //         if (streamPushItem != null) { | 
|---|
|  |  |  | //            // TODO 发送停止 | 
|---|
|  |  |  | 
|---|
|  |  |  | resultHolder.put(key, uuid, result); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!exist) { | 
|---|
|  |  |  | playService.play(mediaInfo, deviceId, channelId, (code, message, data) -> { | 
|---|
|  |  |  | playService.play(mediaInfo, deviceId, channelId, null, (code, message, data) -> { | 
|---|
|  |  |  | msg.setData(new HookResult(code, message)); | 
|---|
|  |  |  | resultHolder.invokeResult(msg); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | // 拉流代理 | 
|---|
|  |  |  | StreamProxyItem streamProxyByAppAndStream = streamProxyService.getStreamProxyByAppAndStream(param.getApp(), param.getStream()); | 
|---|
|  |  |  | if (streamProxyByAppAndStream != null && streamProxyByAppAndStream.isEnable_disable_none_reader()) { | 
|---|
|  |  |  | if (streamProxyByAppAndStream != null && streamProxyByAppAndStream.isEnableDisableNoneReader()) { | 
|---|
|  |  |  | streamProxyService.start(param.getApp(), param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | DeferredResult<HookResult> result = new DeferredResult<>(); | 
|---|
|  |  |  | 
|---|
|  |  |  | List<ZlmHttpHookSubscribe.Event> subscribes = this.subscribe.getSubscribes(HookType.on_server_started); | 
|---|
|  |  |  | if (subscribes != null && subscribes.size() > 0) { | 
|---|
|  |  |  | for (ZlmHttpHookSubscribe.Event subscribe : subscribes) { | 
|---|
|  |  |  | subscribe.response(null, jsonObject); | 
|---|
|  |  |  | subscribe.response(null, zlmServerConfig); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | mediaServerService.zlmServerOnline(zlmServerConfig); | 
|---|
|  |  |  | 
|---|
|  |  |  | List<ZlmHttpHookSubscribe.Event> subscribes = this.subscribe.getSubscribes(HookType.on_rtp_server_timeout); | 
|---|
|  |  |  | if (subscribes != null && subscribes.size() > 0) { | 
|---|
|  |  |  | for (ZlmHttpHookSubscribe.Event subscribe : subscribes) { | 
|---|
|  |  |  | subscribe.response(null, json); | 
|---|
|  |  |  | subscribe.response(null, param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|