|  |  |  | 
|---|
|  |  |  | package com.genersoft.iot.vmp.service.impl; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.alibaba.fastjson2.JSONArray; | 
|---|
|  |  |  | import com.alibaba.fastjson2.JSONObject; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.InviteInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.InviteSessionStatus; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.InviteSessionType; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.StreamInfo; | 
|---|
|  |  |  | import com.baomidou.dynamic.datasource.annotation.DS; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.DynamicTask; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.SipConfig; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.UserSetting; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.exception.ControllerException; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.exception.ServiceException; | 
|---|
|  |  |  | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.session.AudioBroadcastManager; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.session.SSRCFactory; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.session.VideoStreamSessionManager; | 
|---|
|  |  |  | 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.ISIPCommander; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommanderForPlatform; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.utils.SipUtils; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.ZLMRESTfulUtils; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.bean.MediaInfo; | 
|---|
|  |  |  | 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.SendRtpPortManager; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.ZLMServerFactory; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.ZlmHttpHookSubscribe; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.HookSubscribeFactory; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.HookSubscribeForRecordMp4; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.HookSubscribeForStreamChange; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.MediaServerItem; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.MediaServer; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.hook.HookParam; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.hook.HookResult; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.hook.OnRecordMp4HookParam; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.hook.OnStreamChangedHookParam; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.*; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.ErrorCallback; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.InviteErrorCode; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.RequestPushStreamMsg; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.bean.SSRCInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.redisMsg.RedisGbPlayMsgListener; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.dao.CloudRecordServiceMapper; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.utils.CloudRecordUtils; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.utils.DateUtil; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.AudioBroadcastResult; | 
|---|
|  |  |  | 
|---|
|  |  |  | 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.context.event.EventListener; | 
|---|
|  |  |  | import org.springframework.scheduling.annotation.Async; | 
|---|
|  |  |  | import org.springframework.stereotype.Service; | 
|---|
|  |  |  | import org.springframework.util.ObjectUtils; | 
|---|
|  |  |  | import org.springframework.web.context.request.async.DeferredResult; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import javax.sdp.*; | 
|---|
|  |  |  | import javax.sip.InvalidArgumentException; | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @SuppressWarnings(value = {"rawtypes", "unchecked"}) | 
|---|
|  |  |  | @Service | 
|---|
|  |  |  | @DS("master") | 
|---|
|  |  |  | public class PlayServiceImpl implements IPlayService { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private final static Logger logger = LoggerFactory.getLogger(PlayServiceImpl.class); | 
|---|
|  |  |  | 
|---|
|  |  |  | private IVideoManagerStorage storager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private SIPCommander cmder; | 
|---|
|  |  |  | private ISIPCommander cmder; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private AudioBroadcastManager audioBroadcastManager; | 
|---|
|  |  |  | 
|---|
|  |  |  | private SendRtpPortManager sendRtpPortManager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ZLMRESTfulUtils zlmresTfulUtils; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IMediaService mediaService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | 
|---|
|  |  |  | private UserSetting userSetting; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private SipConfig sipConfig; | 
|---|
|  |  |  | private IDeviceChannelService channelService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private DynamicTask dynamicTask; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private CloudRecordServiceMapper cloudRecordServiceMapper; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ISIPCommanderForPlatform commanderForPlatform; | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Qualifier("taskExecutor") | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ThreadPoolTaskExecutor taskExecutor; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private RedisGbPlayMsgListener redisGbPlayMsgListener; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ZlmHttpHookSubscribe hookSubscribe; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private SSRCFactory ssrcFactory; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private RedisTemplate<Object, Object> redisTemplate; | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流到来的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @org.springframework.context.event.EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaArrivalEvent event) { | 
|---|
|  |  |  | if ("broadcast".equals(event.getApp())) { | 
|---|
|  |  |  | if (event.getStream().indexOf("_") > 0) { | 
|---|
|  |  |  | String[] streamArray = event.getStream().split("_"); | 
|---|
|  |  |  | if (streamArray.length == 2) { | 
|---|
|  |  |  | String deviceId = streamArray[0]; | 
|---|
|  |  |  | String channelId = streamArray[1]; | 
|---|
|  |  |  | Device device = deviceService.getDevice(deviceId); | 
|---|
|  |  |  | if (device == null) { | 
|---|
|  |  |  | logger.info("[语音对讲/喊话] 未找到设备:{}", deviceId); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if ("broadcast".equals(event.getApp())) { | 
|---|
|  |  |  | if (audioBroadcastManager.exit(deviceId, channelId)) { | 
|---|
|  |  |  | stopAudioBroadcast(deviceId, channelId); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 开启语音对讲通道 | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | audioBroadcastCmd(device, channelId, event.getMediaServer(), | 
|---|
|  |  |  | event.getApp(), event.getStream(), 60, false, (msg) -> { | 
|---|
|  |  |  | logger.info("[语音对讲] 通道建立成功, device: {}, channel: {}", deviceId, channelId); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException e) { | 
|---|
|  |  |  | logger.error("[命令发送失败] 语音对讲: {}", e.getMessage()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else if ("talk".equals(event.getApp())) { | 
|---|
|  |  |  | // 开启语音对讲通道 | 
|---|
|  |  |  | talkCmd(device, channelId, event.getMediaServer(), event.getStream(), (msg) -> { | 
|---|
|  |  |  | logger.info("[语音对讲] 通道建立成功, device: {}, channel: {}", deviceId, channelId); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流离开的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaDepartureEvent event) { | 
|---|
|  |  |  | if ("broadcast".equals(event.getApp()) || "talk".equals(event.getApp())) { | 
|---|
|  |  |  | if (event.getStream().indexOf("_") > 0) { | 
|---|
|  |  |  | String[] streamArray = event.getStream().split("_"); | 
|---|
|  |  |  | if (streamArray.length == 2) { | 
|---|
|  |  |  | String deviceId = streamArray[0]; | 
|---|
|  |  |  | String channelId = streamArray[1]; | 
|---|
|  |  |  | Device device = deviceService.getDevice(deviceId); | 
|---|
|  |  |  | if (device == null) { | 
|---|
|  |  |  | logger.info("[语音对讲/喊话] 未找到设备:{}", deviceId); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if ("broadcast".equals(event.getApp())) { | 
|---|
|  |  |  | stopAudioBroadcast(deviceId, channelId); | 
|---|
|  |  |  | }else if ("talk".equals(event.getApp())) { | 
|---|
|  |  |  | stopTalk(device, channelId, false); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流未找到的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaNotFoundEvent event) { | 
|---|
|  |  |  | if (!"rtp".equals(event.getApp())) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String[] s = event.getStream().split("_"); | 
|---|
|  |  |  | if ((s.length != 2 && s.length != 4)) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String deviceId = s[0]; | 
|---|
|  |  |  | String channelId = s[1]; | 
|---|
|  |  |  | Device device = redisCatchStorage.getDevice(deviceId); | 
|---|
|  |  |  | if (device == null || !device.isOnLine()) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | DeviceChannel deviceChannel = storager.queryChannel(deviceId, channelId); | 
|---|
|  |  |  | if (deviceChannel == null) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (s.length == 2) { | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 预览流未找到, 发起自动点播:{}->{}->{}/{}", event.getMediaServer().getId(), event.getSchema(), event.getApp(), event.getStream()); | 
|---|
|  |  |  | play(event.getMediaServer(), deviceId, channelId, null, null); | 
|---|
|  |  |  | } 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) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String startTime = DateUtil.urlToyyyy_MM_dd_HH_mm_ss(startTimeStr); | 
|---|
|  |  |  | String endTime = DateUtil.urlToyyyy_MM_dd_HH_mm_ss(endTimeStr); | 
|---|
|  |  |  | logger.info("[ZLM HOOK] 回放流未找到, 发起自动点播:{}->{}->{}/{}-{}-{}", | 
|---|
|  |  |  | event.getMediaServer().getId(), event.getSchema(), | 
|---|
|  |  |  | event.getApp(), event.getStream(), | 
|---|
|  |  |  | startTime, endTime | 
|---|
|  |  |  | ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | SSRCInfo ssrcInfo = mediaServerService.openRTPServer(event.getMediaServer(), event.getStream(), null, | 
|---|
|  |  |  | device.isSsrcCheck(), true, 0, false, false, device.getStreamModeForParam()); | 
|---|
|  |  |  | playBack(event.getMediaServer(), ssrcInfo, deviceId, channelId, startTime, endTime, null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public SSRCInfo play(MediaServerItem mediaServerItem, String deviceId, String channelId, String ssrc, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | public SSRCInfo play(MediaServer mediaServerItem, String deviceId, String channelId, String ssrc, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | if (mediaServerItem == null) { | 
|---|
|  |  |  | logger.warn("[点播] 未找到可用的zlm deviceId: {},channelId:{}", deviceId, channelId); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到可用的zlm"); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (device.getStreamMode().equalsIgnoreCase("TCP-ACTIVE") && !mediaServerItem.isRtpEnable()) { | 
|---|
|  |  |  | logger.warn("[点播] 单端口收流时不支持TCP主动方式收流 deviceId: {},channelId:{}", deviceId, channelId); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "单端口收流时不支持TCP主动方式收流"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | DeviceChannel channel = channelService.getOne(deviceId, channelId); | 
|---|
|  |  |  | if (channel == null) { | 
|---|
|  |  |  | logger.warn("[点播] 未找到通道 deviceId: {},channelId:{}", deviceId, channelId); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到通道"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | InviteInfo inviteInfo = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId); | 
|---|
|  |  |  | if (inviteInfo != null ) { | 
|---|
|  |  |  | 
|---|
|  |  |  | return inviteInfo.getSsrcInfo(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String mediaServerId = streamInfo.getMediaServerId(); | 
|---|
|  |  |  | MediaServerItem mediaInfo = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | MediaServer mediaInfo = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Boolean ready = zlmServerFactory.isStreamReady(mediaInfo, "rtp", streamId); | 
|---|
|  |  |  | if (ready != null && ready) { | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String streamId = String.format("%s_%s", device.getDeviceId(), channelId);; | 
|---|
|  |  |  | String streamId = String.format("%s_%s", device.getDeviceId(), channelId); | 
|---|
|  |  |  | SSRCInfo ssrcInfo = mediaServerService.openRTPServer(mediaServerItem, streamId, ssrc, device.isSsrcCheck(),  false, 0, false, false, device.getStreamModeForParam()); | 
|---|
|  |  |  | if (ssrcInfo == null) { | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_RESOURCE_EXHAUSTION.getCode(), InviteErrorCode.ERROR_FOR_RESOURCE_EXHAUSTION.getMsg(), null); | 
|---|
|  |  |  | 
|---|
|  |  |  | null); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | play(mediaServerItem, ssrcInfo, device, channelId, callback); | 
|---|
|  |  |  | play(mediaServerItem, ssrcInfo, device, channel, callback); | 
|---|
|  |  |  | return ssrcInfo; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void talk(MediaServerItem mediaServerItem, Device device, String channelId, String stream, | 
|---|
|  |  |  | private void talk(MediaServer mediaServerItem, Device device, String channelId, String stream, | 
|---|
|  |  |  | ZlmHttpHookSubscribe.Event hookEvent, SipSubscribe.Event errorEvent, | 
|---|
|  |  |  | Runnable timeoutCallback, AudioBroadcastEvent audioEvent) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void play(MediaServerItem mediaServerItem, SSRCInfo ssrcInfo, Device device, String channelId, | 
|---|
|  |  |  | public void play(MediaServer mediaServerItem, SSRCInfo ssrcInfo, Device device, DeviceChannel channel, | 
|---|
|  |  |  | ErrorCallback<Object> callback) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (mediaServerItem == null || ssrcInfo == null) { | 
|---|
|  |  |  | 
|---|
|  |  |  | null); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | logger.info("[点播开始] deviceId: {}, channelId: {},码流类型:{}, 收流端口: {}, STREAM:{}, 收流模式:{}, SSRC: {}, SSRC校验:{}", | 
|---|
|  |  |  | device.getDeviceId(), channelId, device.isSwitchPrimarySubStream() ? "辅码流" : "主码流", ssrcInfo.getPort(), ssrcInfo.getStream(), | 
|---|
|  |  |  | logger.info("[点播开始] deviceId: {}, channelId: {},码流类型:{}, 收流端口: {}, 码流:{}, 收流模式:{}, SSRC: {}, SSRC校验:{}", | 
|---|
|  |  |  | device.getDeviceId(), channel.getChannelId(), channel.getStreamIdentification(), ssrcInfo.getPort(), ssrcInfo.getStream(), | 
|---|
|  |  |  | device.getStreamMode(), ssrcInfo.getSsrc(), device.isSsrcCheck()); | 
|---|
|  |  |  | //端口获取失败的ssrcInfo 没有必要发送点播指令 | 
|---|
|  |  |  | if (ssrcInfo.getPort() <= 0) { | 
|---|
|  |  |  | logger.info("[点播端口分配异常],deviceId={},channelId={},ssrcInfo={}", device.getDeviceId(), channelId, ssrcInfo); | 
|---|
|  |  |  | logger.info("[点播端口分配异常],deviceId={},channelId={},ssrcInfo={}", device.getDeviceId(), channel.getChannelId(), ssrcInfo); | 
|---|
|  |  |  | // 释放ssrc | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_RESOURCE_EXHAUSTION.getCode(), "点播端口分配异常", null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_RESOURCE_EXHAUSTION.getCode(), "点播端口分配异常", null); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 初始化redis中的invite消息状态 | 
|---|
|  |  |  | InviteInfo inviteInfo = InviteInfo.getInviteInfo(device.getDeviceId(), channelId, ssrcInfo.getStream(), ssrcInfo, | 
|---|
|  |  |  | InviteInfo inviteInfo = InviteInfo.getInviteInfo(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream(), ssrcInfo, | 
|---|
|  |  |  | mediaServerItem.getSdpIp(), ssrcInfo.getPort(), device.getStreamMode(), InviteSessionType.PLAY, | 
|---|
|  |  |  | InviteSessionStatus.ready); | 
|---|
|  |  |  | inviteInfo.setSubStream(device.isSwitchPrimarySubStream()); | 
|---|
|  |  |  | inviteStreamService.updateInviteInfo(inviteInfo); | 
|---|
|  |  |  | // 超时处理 | 
|---|
|  |  |  | String timeOutTaskKey = UUID.randomUUID().toString(); | 
|---|
|  |  |  | dynamicTask.startDelay(timeOutTaskKey, () -> { | 
|---|
|  |  |  | // 执行超时任务时查询是否已经成功,成功了则不执行超时任务,防止超时任务取消失败的情况 | 
|---|
|  |  |  | InviteInfo inviteInfoForTimeOut = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channelId); | 
|---|
|  |  |  | InviteInfo inviteInfoForTimeOut = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId()); | 
|---|
|  |  |  | if (inviteInfoForTimeOut == null || inviteInfoForTimeOut.getStreamInfo() == null) { | 
|---|
|  |  |  | logger.info("[点播超时] 收流超时 deviceId: {}, channelId: {},码流类型:{},端口:{}, SSRC: {}", | 
|---|
|  |  |  | device.getDeviceId(), channelId, device.isSwitchPrimarySubStream() ? "辅码流" : "主码流", | 
|---|
|  |  |  | logger.info("[点播超时] 收流超时 deviceId: {}, channelId: {},码流:{},端口:{}, SSRC: {}", | 
|---|
|  |  |  | device.getDeviceId(), channel.getChannelId(), channel.getStreamIdentification(), | 
|---|
|  |  |  | ssrcInfo.getPort(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_STREAM_TIMEOUT.getCode(), InviteErrorCode.ERROR_FOR_STREAM_TIMEOUT.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_STREAM_TIMEOUT.getCode(), InviteErrorCode.ERROR_FOR_STREAM_TIMEOUT.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channelId); | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | cmder.streamByeCmd(device, channelId, ssrcInfo.getStream(), null); | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException | | 
|---|
|  |  |  | SsrcTransactionNotFoundException e) { | 
|---|
|  |  |  | cmder.streamByeCmd(device, channel.getChannelId(), ssrcInfo.getStream(), null); | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException | SsrcTransactionNotFoundException e) { | 
|---|
|  |  |  | logger.error("[点播超时], 发送BYE失败 {}", e.getMessage()); | 
|---|
|  |  |  | } finally { | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  | mediaServerService.closeRTPServer(mediaServerItem, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  | mediaServerService.closeRTPServer(mediaServerItem, ssrcInfo.getStream()); | 
|---|
|  |  |  | // 取消订阅消息监听 | 
|---|
|  |  |  | HookSubscribeForStreamChange hookSubscribe = HookSubscribeFactory.on_stream_changed("rtp", ssrcInfo.getStream(), true, "rtsp", mediaServerItem.getId()); | 
|---|
|  |  |  | subscribe.removeSubscribe(hookSubscribe); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | logger.info("[点播超时] 收流超时 deviceId: {}, channelId: {},码流类型:{},端口:{}, SSRC: {}", | 
|---|
|  |  |  | device.getDeviceId(), channelId, device.isSwitchPrimarySubStream() ? "辅码流" : "主码流", | 
|---|
|  |  |  | logger.info("[点播超时] 收流超时 deviceId: {}, channelId: {},码流:{},端口:{}, SSRC: {}", | 
|---|
|  |  |  | device.getDeviceId(), channel.getChannelId(), channel.getStreamIdentification(), | 
|---|
|  |  |  | ssrcInfo.getPort(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | mediaServerService.closeRTPServer(mediaServerItem.getId(), ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }, userSetting.getPlayTimeout()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | cmder.playStreamCmd(mediaServerItem, ssrcInfo, device, channelId, (mediaServerItemInuse, hookParam ) -> { | 
|---|
|  |  |  | cmder.playStreamCmd(mediaServerItem, ssrcInfo, device, channel, (mediaServerItemInuse, hookParam ) -> { | 
|---|
|  |  |  | logger.info("收到订阅消息: " + hookParam); | 
|---|
|  |  |  | dynamicTask.stop(timeOutTaskKey); | 
|---|
|  |  |  | // hook响应 | 
|---|
|  |  |  | StreamInfo streamInfo = onPublishHandlerForPlay(mediaServerItemInuse, hookParam, device.getDeviceId(), channelId); | 
|---|
|  |  |  | StreamInfo streamInfo = onPublishHandlerForPlay(mediaServerItemInuse, hookParam, device.getDeviceId(), channel.getChannelId()); | 
|---|
|  |  |  | if (streamInfo == null){ | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_STREAM_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_STREAM_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_STREAM_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_STREAM_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | callback.run(InviteErrorCode.SUCCESS.getCode(), InviteErrorCode.SUCCESS.getMsg(), streamInfo); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.SUCCESS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.SUCCESS.getMsg(), | 
|---|
|  |  |  | streamInfo); | 
|---|
|  |  |  | logger.info("[点播成功] deviceId: {}, channelId:{}, 码流类型:{}", device.getDeviceId(), channelId, | 
|---|
|  |  |  | device.isSwitchPrimarySubStream() ? "辅码流" : "主码流"); | 
|---|
|  |  |  | snapOnPlay(mediaServerItemInuse, device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | logger.info("[点播成功] deviceId: {}, channelId:{}, 码流类型:{}", device.getDeviceId(), channel.getChannelId(), | 
|---|
|  |  |  | channel.getStreamIdentification()); | 
|---|
|  |  |  | snapOnPlay(mediaServerItemInuse, device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  | }, (eventResult) -> { | 
|---|
|  |  |  | // 处理收到200ok后的TCP主动连接以及SSRC不一致的问题 | 
|---|
|  |  |  | InviteOKHandler(eventResult, ssrcInfo, mediaServerItem, device, channelId, | 
|---|
|  |  |  | InviteOKHandler(eventResult, ssrcInfo, mediaServerItem, device, channel.getChannelId(), | 
|---|
|  |  |  | timeOutTaskKey, callback, inviteInfo, InviteSessionType.PLAY); | 
|---|
|  |  |  | }, (event) -> { | 
|---|
|  |  |  | logger.info("[点播失败] deviceId: {}, channelId:{}, {}: {}", device.getDeviceId(), channelId, event.statusCode, event.msg); | 
|---|
|  |  |  | logger.info("[点播失败] deviceId: {}, channelId:{}, {}: {}", device.getDeviceId(), channel.getChannelId(), event.statusCode, event.msg); | 
|---|
|  |  |  | dynamicTask.stop(timeOutTaskKey); | 
|---|
|  |  |  | mediaServerService.closeRTPServer(mediaServerItem, ssrcInfo.getStream()); | 
|---|
|  |  |  | // 释放ssrc | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_SIGNALLING_ERROR.getCode(), | 
|---|
|  |  |  | String.format("点播失败, 错误码: %s, %s", event.statusCode, event.msg), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_RESET_SSRC.getCode(), | 
|---|
|  |  |  | String.format("点播失败, 错误码: %s, %s", event.statusCode, event.msg), null); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channelId); | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId()); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } catch (InvalidArgumentException | SipException | ParseException e) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // 释放ssrc | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channel.getChannelId(), ssrcInfo.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_SIP_SENDING_FAILED.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SIP_SENDING_FAILED.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId(), null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SIP_SENDING_FAILED.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SIP_SENDING_FAILED.getMsg(), null); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channelId); | 
|---|
|  |  |  | inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, device.getDeviceId(), channel.getChannelId()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void tcpActiveHandler(Device device, String channelId, String contentString, | 
|---|
|  |  |  | MediaServerItem mediaServerItem, | 
|---|
|  |  |  | MediaServer mediaServerItem, | 
|---|
|  |  |  | String timeOutTaskKey, SSRCInfo ssrcInfo, ErrorCallback<Object> callback){ | 
|---|
|  |  |  | if (!device.getStreamMode().equalsIgnoreCase("TCP-ACTIVE")) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | logger.info("[TCP主动连接对方] deviceId: {}, channelId: {}, 连接对方的地址:{}:{}, 收流模式:{}, SSRC: {}, SSRC校验:{}", device.getDeviceId(), channelId, sdp.getConnection().getAddress(), port, device.getStreamMode(), ssrcInfo.getSsrc(), device.isSsrcCheck()); | 
|---|
|  |  |  | JSONObject jsonObject = zlmresTfulUtils.connectRtpServer(mediaServerItem, sdp.getConnection().getAddress(), port, ssrcInfo.getStream()); | 
|---|
|  |  |  | logger.info("[TCP主动连接对方] 结果: {}", jsonObject); | 
|---|
|  |  |  | Boolean result = mediaServerService.connectRtpServer(mediaServerItem, sdp.getConnection().getAddress(), port, ssrcInfo.getStream()); | 
|---|
|  |  |  | logger.info("[TCP主动连接对方] 结果: {}" , result); | 
|---|
|  |  |  | if (!result) { | 
|---|
|  |  |  | // 主动连接失败,结束流程, 清理数据 | 
|---|
|  |  |  | dynamicTask.stop(timeOutTaskKey); | 
|---|
|  |  |  | mediaServerService.closeRTPServer(mediaServerItem, ssrcInfo.getStream()); | 
|---|
|  |  |  | // 释放ssrc | 
|---|
|  |  |  | mediaServerService.releaseSsrc(mediaServerItem.getId(), ssrcInfo.getSsrc()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | streamSession.remove(device.getDeviceId(), channelId, ssrcInfo.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.BROADCAST, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } catch (SdpException e) { | 
|---|
|  |  |  | logger.error("[TCP主动连接对方] deviceId: {}, channelId: {}, 解析200OK的SDP信息失败", device.getDeviceId(), channelId, e); | 
|---|
|  |  |  | dynamicTask.stop(timeOutTaskKey); | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.PLAY, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | inviteStreamService.call(InviteSessionType.BROADCAST, device.getDeviceId(), channelId, null, | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_SDP_PARSING_EXCEPTIONS.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | * @param channelId            通道 ID | 
|---|
|  |  |  | * @param stream               ssrc | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | private void snapOnPlay(MediaServerItem mediaServerItemInuse, String deviceId, String channelId, String stream) { | 
|---|
|  |  |  | private void snapOnPlay(MediaServer mediaServerItemInuse, String deviceId, String channelId, String stream) { | 
|---|
|  |  |  | String streamUrl; | 
|---|
|  |  |  | if (mediaServerItemInuse.getRtspPort() != 0) { | 
|---|
|  |  |  | streamUrl = String.format("rtsp://127.0.0.1:%s/%s/%s", mediaServerItemInuse.getRtspPort(), "rtp", stream); | 
|---|
|  |  |  | 
|---|
|  |  |  | String fileName = deviceId + "_" + channelId + ".jpg"; | 
|---|
|  |  |  | // 请求截图 | 
|---|
|  |  |  | logger.info("[请求截图]: " + fileName); | 
|---|
|  |  |  | zlmresTfulUtils.getSnap(mediaServerItemInuse, streamUrl, 15, 1, path, fileName); | 
|---|
|  |  |  | mediaServerService.getSnap(mediaServerItemInuse, streamUrl, 15, 1, path, fileName); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public StreamInfo onPublishHandlerForPlay(MediaServerItem mediaServerItem, HookParam hookParam, String deviceId, String channelId) { | 
|---|
|  |  |  | public StreamInfo onPublishHandlerForPlay(MediaServer mediaServerItem, HookParam hookParam, String deviceId, String channelId) { | 
|---|
|  |  |  | StreamInfo streamInfo = null; | 
|---|
|  |  |  | Device device = redisCatchStorage.getDevice(deviceId); | 
|---|
|  |  |  | OnStreamChangedHookParam streamChangedHookParam = (OnStreamChangedHookParam)hookParam; | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private StreamInfo onPublishHandlerForPlayback(MediaServerItem mediaServerItem, HookParam param, String deviceId, String channelId, String startTime, String endTime) { | 
|---|
|  |  |  | private StreamInfo onPublishHandlerForPlayback(MediaServer mediaServerItem, HookParam param, String deviceId, String channelId, String startTime, String endTime) { | 
|---|
|  |  |  | OnStreamChangedHookParam streamChangedHookParam = (OnStreamChangedHookParam) param; | 
|---|
|  |  |  | StreamInfo streamInfo = onPublishHandler(mediaServerItem, streamChangedHookParam, deviceId, channelId); | 
|---|
|  |  |  | if (streamInfo != null) { | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public MediaServerItem getNewMediaServerItem(Device device) { | 
|---|
|  |  |  | public MediaServer getNewMediaServerItem(Device device) { | 
|---|
|  |  |  | if (device == null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | MediaServerItem mediaServerItem; | 
|---|
|  |  |  | MediaServer mediaServerItem; | 
|---|
|  |  |  | if (ObjectUtils.isEmpty(device.getMediaServerId()) || "auto".equals(device.getMediaServerId())) { | 
|---|
|  |  |  | mediaServerItem = mediaServerService.getMediaServerForMinimumLoad(null); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void playBack(String deviceId, String channelId, String startTime, | 
|---|
|  |  |  | String endTime, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | String endTime, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | Device device = storager.queryVideoDevice(deviceId); | 
|---|
|  |  |  | if (device == null) { | 
|---|
|  |  |  | logger.warn("[录像回放] 未找到设备 deviceId: {},channelId:{}", deviceId, channelId); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到设备:" + deviceId); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaServerItem newMediaServerItem = getNewMediaServerItem(device); | 
|---|
|  |  |  | MediaServer newMediaServerItem = getNewMediaServerItem(device); | 
|---|
|  |  |  | if (device.getStreamMode().equalsIgnoreCase("TCP-ACTIVE") && ! newMediaServerItem.isRtpEnable()) { | 
|---|
|  |  |  | logger.warn("[录像回放] 单端口收流时不支持TCP主动方式收流 deviceId: {},channelId:{}", deviceId, channelId); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "单端口收流时不支持TCP主动方式收流"); | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void playBack(MediaServerItem mediaServerItem, SSRCInfo ssrcInfo, | 
|---|
|  |  |  | String deviceId, String channelId, String startTime, | 
|---|
|  |  |  | String endTime, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | public void playBack(MediaServer mediaServerItem, SSRCInfo ssrcInfo, | 
|---|
|  |  |  | String deviceId, String channelId, String startTime, | 
|---|
|  |  |  | String endTime, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | if (mediaServerItem == null || ssrcInfo == null) { | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getMsg(), | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void InviteOKHandler(SipSubscribe.EventResult eventResult, SSRCInfo ssrcInfo, MediaServerItem mediaServerItem, | 
|---|
|  |  |  | private void InviteOKHandler(SipSubscribe.EventResult eventResult, SSRCInfo ssrcInfo, MediaServer mediaServerItem, | 
|---|
|  |  |  | Device device, String channelId, String timeOutTaskKey, ErrorCallback<Object> callback, | 
|---|
|  |  |  | InviteInfo inviteInfo, InviteSessionType inviteSessionType){ | 
|---|
|  |  |  | inviteInfo.setStatus(InviteSessionStatus.ok); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (device == null) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | MediaServerItem newMediaServerItem = this.getNewMediaServerItem(device); | 
|---|
|  |  |  | MediaServer newMediaServerItem = this.getNewMediaServerItem(device); | 
|---|
|  |  |  | if (newMediaServerItem == null) { | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_ASSIST_NOT_READY.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_ASSIST_NOT_READY.getMsg(), | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void download(MediaServerItem mediaServerItem, SSRCInfo ssrcInfo, String deviceId, String channelId, String startTime, String endTime, int downloadSpeed, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | public void download(MediaServer mediaServerItem, SSRCInfo ssrcInfo, String deviceId, String channelId, String startTime, String endTime, int downloadSpeed, ErrorCallback<Object> callback) { | 
|---|
|  |  |  | if (mediaServerItem == null || ssrcInfo == null) { | 
|---|
|  |  |  | callback.run(InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getCode(), | 
|---|
|  |  |  | InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getMsg(), | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 获取当前已下载时长 | 
|---|
|  |  |  | String mediaServerId = inviteInfo.getStreamInfo().getMediaServerId(); | 
|---|
|  |  |  | MediaServerItem mediaServerItem = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | MediaServer mediaServerItem = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | if (mediaServerItem == null) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 查询录像信息时发现节点不存在"); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | 
|---|
|  |  |  | logger.warn("[获取下载进度] 下载已结束"); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject mediaListJson= zlmresTfulUtils.getMediaList(mediaServerItem, "rtp", stream); | 
|---|
|  |  |  | if (mediaListJson == null) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 从zlm查询进度失败"); | 
|---|
|  |  |  | String app = "rtp"; | 
|---|
|  |  |  | MediaInfo mediaInfo = mediaServerService.getMediaInfo(mediaServerItem, app, stream); | 
|---|
|  |  |  | if (mediaInfo == null) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 查询进度失败, 节点Id: {}, {}/{}", mediaServerId, app, stream); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (mediaListJson.getInteger("code") != 0) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 从zlm查询进度出现错误: {}", mediaListJson.getString("msg")); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | JSONArray data = mediaListJson.getJSONArray("data"); | 
|---|
|  |  |  | if (data == null) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 从zlm查询进度时未返回数据"); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | JSONObject mediaJSON = data.getJSONObject(0); | 
|---|
|  |  |  | JSONArray tracks = mediaJSON.getJSONArray("tracks"); | 
|---|
|  |  |  | if (tracks.isEmpty()) { | 
|---|
|  |  |  | logger.warn("[获取下载进度] 从zlm查询进度时未返回数据"); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | JSONObject jsonObject = tracks.getJSONObject(0); | 
|---|
|  |  |  | long duration = jsonObject.getLongValue("duration"); | 
|---|
|  |  |  | if (duration == 0) { | 
|---|
|  |  |  | if (mediaInfo.getDuration() == 0) { | 
|---|
|  |  |  | inviteInfo.getStreamInfo().setProgress(0); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | String startTime = inviteInfo.getStreamInfo().getStartTime(); | 
|---|
|  |  |  | 
|---|
|  |  |  | long start = DateUtil.yyyy_MM_dd_HH_mm_ssToTimestamp(startTime); | 
|---|
|  |  |  | long end = DateUtil.yyyy_MM_dd_HH_mm_ssToTimestamp(endTime); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | BigDecimal currentCount = new BigDecimal(duration); | 
|---|
|  |  |  | BigDecimal currentCount = new BigDecimal(mediaInfo.getDuration()); | 
|---|
|  |  |  | BigDecimal totalCount = new BigDecimal((end - start) * 1000); | 
|---|
|  |  |  | BigDecimal divide = currentCount.divide(totalCount, 2, RoundingMode.HALF_UP); | 
|---|
|  |  |  | double process = divide.doubleValue(); | 
|---|
|  |  |  | 
|---|
|  |  |  | return inviteInfo.getStreamInfo(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private StreamInfo onPublishHandlerForDownload(MediaServerItem mediaServerItemInuse, HookParam hookParam, String deviceId, String channelId, String startTime, String endTime) { | 
|---|
|  |  |  | private StreamInfo onPublishHandlerForDownload(MediaServer mediaServerItemInuse, HookParam hookParam, String deviceId, String channelId, String startTime, String endTime) { | 
|---|
|  |  |  | OnStreamChangedHookParam streamChangedHookParam = (OnStreamChangedHookParam) hookParam; | 
|---|
|  |  |  | StreamInfo streamInfo = onPublishHandler(mediaServerItemInuse, streamChangedHookParam, deviceId, channelId); | 
|---|
|  |  |  | if (streamInfo != null) { | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public StreamInfo onPublishHandler(MediaServerItem mediaServerItem, OnStreamChangedHookParam hookParam, String deviceId, String channelId) { | 
|---|
|  |  |  | StreamInfo streamInfo = mediaService.getStreamInfoByAppAndStream(mediaServerItem, "rtp", hookParam.getStream(), hookParam.getTracks(), null); | 
|---|
|  |  |  | public StreamInfo onPublishHandler(MediaServer mediaServerItem, OnStreamChangedHookParam hookParam, String deviceId, String channelId) { | 
|---|
|  |  |  | MediaInfo mediaInfo = MediaInfo.getInstance(hookParam); | 
|---|
|  |  |  | StreamInfo streamInfo = mediaService.getStreamInfoByAppAndStream(mediaServerItem, "rtp", hookParam.getStream(), mediaInfo, null); | 
|---|
|  |  |  | streamInfo.setDeviceID(deviceId); | 
|---|
|  |  |  | streamInfo.setChannelId(channelId); | 
|---|
|  |  |  | return streamInfo; | 
|---|
|  |  |  | 
|---|
|  |  |  | logger.warn("开启语音广播的时候未找到通道: {}", channelId); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | MediaServerItem mediaServerItem = mediaServerService.getMediaServerForMinimumLoad(null); | 
|---|
|  |  |  | MediaServer mediaServerItem = mediaServerService.getMediaServerForMinimumLoad(null); | 
|---|
|  |  |  | if (broadcastMode == null) { | 
|---|
|  |  |  | broadcastMode = true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public boolean audioBroadcastCmd(Device device, String channelId, MediaServerItem mediaServerItem, String app, String stream, int timeout, boolean isFromPlatform, AudioBroadcastEvent event) throws InvalidArgumentException, ParseException, SipException { | 
|---|
|  |  |  | public boolean audioBroadcastCmd(Device device, String channelId, MediaServer mediaServerItem, String app, String stream, int timeout, boolean isFromPlatform, AudioBroadcastEvent event) throws InvalidArgumentException, ParseException, SipException { | 
|---|
|  |  |  | if (device == null || channelId == null) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | // 发送成功 | 
|---|
|  |  |  | AudioBroadcastCatch audioBroadcastCatch = new AudioBroadcastCatch(device.getDeviceId(), channelId, mediaServerItem, app, stream, event, AudioBroadcastCatchStatus.Ready, isFromPlatform); | 
|---|
|  |  |  | audioBroadcastManager.update(audioBroadcastCatch); | 
|---|
|  |  |  | // 等待invite消息, 超时则结束 | 
|---|
|  |  |  | String key = VideoManagerConstants.BROADCAST_WAITE_INVITE +  device.getDeviceId(); | 
|---|
|  |  |  | if (!SipUtils.isFrontEnd(device.getDeviceId())) { | 
|---|
|  |  |  | key += audioBroadcastCatch.getChannelId(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | dynamicTask.startDelay(key, ()->{ | 
|---|
|  |  |  | logger.info("[语音广播]等待invite消息超时:{}/{}", device.getDeviceId(), channelId); | 
|---|
|  |  |  | stopAudioBroadcast(device.getDeviceId(), channelId); | 
|---|
|  |  |  | }, 2000); | 
|---|
|  |  |  | }, eventResultForError -> { | 
|---|
|  |  |  | // 发送失败 | 
|---|
|  |  |  | logger.error("语音广播发送失败: {}:{}", channelId, eventResultForError.msg); | 
|---|
|  |  |  | 
|---|
|  |  |  | SendRtpItem sendRtpItem = redisCatchStorage.querySendRTPServer(device.getDeviceId(), channelId, null, null); | 
|---|
|  |  |  | if (sendRtpItem != null && sendRtpItem.isOnlyAudio()) { | 
|---|
|  |  |  | // 查询流是否存在,不存在则认为是异常状态 | 
|---|
|  |  |  | MediaServerItem mediaServerServiceOne = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServerServiceOne = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | Boolean streamReady = zlmServerFactory.isStreamReady(mediaServerServiceOne, sendRtpItem.getApp(), sendRtpItem.getStream()); | 
|---|
|  |  |  | if (streamReady) { | 
|---|
|  |  |  | logger.warn("语音广播通道使用中: {}", channelId); | 
|---|
|  |  |  | 
|---|
|  |  |  | SendRtpItem sendRtpItem = redisCatchStorage.querySendRTPServer(deviceId, audioBroadcastCatch.getChannelId(), null, null); | 
|---|
|  |  |  | if (sendRtpItem != null) { | 
|---|
|  |  |  | redisCatchStorage.deleteSendRTPServer(deviceId, sendRtpItem.getChannelId(), null, null); | 
|---|
|  |  |  | MediaServerItem mediaInfo = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | Map<String, Object> param = new HashMap<>(); | 
|---|
|  |  |  | param.put("vhost", "__defaultVhost__"); | 
|---|
|  |  |  | param.put("app", sendRtpItem.getApp()); | 
|---|
|  |  |  | param.put("stream", sendRtpItem.getStream()); | 
|---|
|  |  |  | zlmresTfulUtils.stopSendRtp(mediaInfo, param); | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | mediaServerService.stopSendRtp(mediaServer, sendRtpItem.getApp(), sendRtpItem.getStream(), null); | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | cmder.streamByeCmdForDeviceInvite(device, sendRtpItem.getChannelId(), audioBroadcastCatch.getSipTransactionInfo(), null); | 
|---|
|  |  |  | } catch (InvalidArgumentException | ParseException | SipException | | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | inviteInfo.getStreamInfo().setPause(true); | 
|---|
|  |  |  | inviteStreamService.updateInviteInfo(inviteInfo); | 
|---|
|  |  |  | MediaServerItem mediaServerItem = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServerItem = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | if (null == mediaServerItem) { | 
|---|
|  |  |  | logger.warn("mediaServer 不存在!"); | 
|---|
|  |  |  | throw new ServiceException("mediaServer不存在"); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (!mediaServerItem.isRtpEnable()) { | 
|---|
|  |  |  | streamKey = Long.toHexString(Long.parseLong(inviteInfo.getSsrcInfo().getSsrc())).toUpperCase(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | JSONObject jsonObject = zlmresTfulUtils.pauseRtpCheck(mediaServerItem, streamKey); | 
|---|
|  |  |  | if (jsonObject == null || jsonObject.getInteger("code") != 0) { | 
|---|
|  |  |  | Boolean result = mediaServerService.pauseRtpCheck(mediaServerItem, streamKey); | 
|---|
|  |  |  | if (!result) { | 
|---|
|  |  |  | throw new ServiceException("暂停RTP接收失败"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Device device = storager.queryVideoDevice(inviteInfo.getDeviceId()); | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | inviteInfo.getStreamInfo().setPause(false); | 
|---|
|  |  |  | inviteStreamService.updateInviteInfo(inviteInfo); | 
|---|
|  |  |  | MediaServerItem mediaServerItem = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServerItem = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | if (null == mediaServerItem) { | 
|---|
|  |  |  | logger.warn("mediaServer 不存在!"); | 
|---|
|  |  |  | throw new ServiceException("mediaServer不存在"); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (!mediaServerItem.isRtpEnable()) { | 
|---|
|  |  |  | streamKey = Long.toHexString(Long.parseLong(inviteInfo.getSsrcInfo().getSsrc())).toUpperCase(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | JSONObject jsonObject = zlmresTfulUtils.resumeRtpCheck(mediaServerItem, streamKey); | 
|---|
|  |  |  | if (jsonObject == null || jsonObject.getInteger("code") != 0) { | 
|---|
|  |  |  | boolean result = mediaServerService.resumeRtpCheck(mediaServerItem, streamKey); | 
|---|
|  |  |  | if (!result) { | 
|---|
|  |  |  | throw new ServiceException("继续RTP接收失败"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Device device = storager.queryVideoDevice(inviteInfo.getDeviceId()); | 
|---|
|  |  |  | 
|---|
|  |  |  | public void startPushStream(SendRtpItem sendRtpItem, SIPResponse sipResponse, ParentPlatform platform, CallIdHeader callIdHeader) { | 
|---|
|  |  |  | // 开始发流 | 
|---|
|  |  |  | String is_Udp = sendRtpItem.isTcp() ? "0" : "1"; | 
|---|
|  |  |  | MediaServerItem mediaInfo = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaInfo = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | logger.info("[开始推流] rtp/{}, 目标={}:{},SSRC={}, RTCP={}", sendRtpItem.getStream(), | 
|---|
|  |  |  | sendRtpItem.getIp(), sendRtpItem.getPort(), sendRtpItem.getSsrc(), sendRtpItem.isRtcp()); | 
|---|
|  |  |  | Map<String, Object> param = new HashMap<>(12); | 
|---|
|  |  |  | 
|---|
|  |  |  | logger.info("调用ZLM推流接口, 结果: {}", jsonObject); | 
|---|
|  |  |  | logger.info("RTP推流成功[ {}/{} ],{}->{}, ", param.get("app"), param.get("stream"), jsonObject.getString("local_port"), | 
|---|
|  |  |  | sendRtpItem.isTcpActive()?"被动发流": param.get("dst_url") + ":" + param.get("dst_port")); | 
|---|
|  |  |  | if (sendRtpItem.getPlayType() == InviteStreamType.PUSH && correlationInfo instanceof ParentPlatform) { | 
|---|
|  |  |  | ParentPlatform platform = (ParentPlatform)correlationInfo; | 
|---|
|  |  |  | MessageForPushChannel messageForPushChannel = MessageForPushChannel.getInstance(0, sendRtpItem.getApp(), sendRtpItem.getStream(), | 
|---|
|  |  |  | sendRtpItem.getChannelId(), platform.getServerGBId(), platform.getName(), userSetting.getServerId(), | 
|---|
|  |  |  | sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | messageForPushChannel.setPlatFormIndex(platform.getId()); | 
|---|
|  |  |  | redisCatchStorage.sendPlatformStartPlayMsg(messageForPushChannel); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | logger.error("RTP推流失败: {}, 参数:{}", jsonObject.getString("msg"), JSONObject.toJSONString(param)); | 
|---|
|  |  |  | if (sendRtpItem.isOnlyAudio()) { | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void talkCmd(Device device, String channelId, MediaServerItem mediaServerItem, String stream, AudioBroadcastEvent event) { | 
|---|
|  |  |  | public void talkCmd(Device device, String channelId, MediaServer mediaServerItem, String stream, AudioBroadcastEvent event) { | 
|---|
|  |  |  | if (device == null || channelId == null) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | SendRtpItem sendRtpItem = redisCatchStorage.querySendRTPServer(device.getDeviceId(), channelId, null, null); | 
|---|
|  |  |  | if (sendRtpItem != null && sendRtpItem.isOnlyAudio()) { | 
|---|
|  |  |  | // 查询流是否存在,不存在则认为是异常状态 | 
|---|
|  |  |  | MediaServerItem mediaServer = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | Boolean streamReady = zlmServerFactory.isStreamReady(mediaServer, sendRtpItem.getApp(), sendRtpItem.getStream()); | 
|---|
|  |  |  | if (streamReady) { | 
|---|
|  |  |  | logger.warn("[语音对讲] 正在语音广播,无法开启语音通话: {}", channelId); | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | SendRtpItem sendRtpItem = redisCatchStorage.querySendRTPServer(device.getDeviceId(), channelId, stream, null); | 
|---|
|  |  |  | if (sendRtpItem != null) { | 
|---|
|  |  |  | MediaServerItem mediaServer = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(sendRtpItem.getMediaServerId()); | 
|---|
|  |  |  | Boolean streamReady = zlmServerFactory.isStreamReady(mediaServer, "rtp", sendRtpItem.getReceiveStream()); | 
|---|
|  |  |  | if (streamReady) { | 
|---|
|  |  |  | logger.warn("[语音对讲] 进行中: {}", channelId); | 
|---|
|  |  |  | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaServerItem mediaServer = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (streamIsReady == null || streamIsReady) { | 
|---|
|  |  |  | Map<String, Object> param = new HashMap<>(); | 
|---|
|  |  |  | 
|---|
|  |  |  | if (inviteInfo != null) { | 
|---|
|  |  |  | if (inviteInfo.getStreamInfo() != null) { | 
|---|
|  |  |  | // 已存在线直接截图 | 
|---|
|  |  |  | MediaServerItem mediaServerItemInuse = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | MediaServer mediaServerItemInuse = mediaServerService.getOne(inviteInfo.getStreamInfo().getMediaServerId()); | 
|---|
|  |  |  | String streamUrl; | 
|---|
|  |  |  | if (mediaServerItemInuse.getRtspPort() != 0) { | 
|---|
|  |  |  | streamUrl = String.format("rtsp://127.0.0.1:%s/%s/%s", mediaServerItemInuse.getRtspPort(), "rtp",  inviteInfo.getStreamInfo().getStream()); | 
|---|
|  |  |  | 
|---|
|  |  |  | String path = "snap"; | 
|---|
|  |  |  | // 请求截图 | 
|---|
|  |  |  | logger.info("[请求截图]: " + fileName); | 
|---|
|  |  |  | zlmresTfulUtils.getSnap(mediaServerItemInuse, streamUrl, 15, 1, path, fileName); | 
|---|
|  |  |  | mediaServerService.getSnap(mediaServerItemInuse, streamUrl, 15, 1, path, fileName); | 
|---|
|  |  |  | File snapFile = new File(path + File.separator + fileName); | 
|---|
|  |  |  | if (snapFile.exists()) { | 
|---|
|  |  |  | errorCallback.run(InviteErrorCode.SUCCESS.getCode(), InviteErrorCode.SUCCESS.getMsg(), snapFile.getAbsoluteFile()); | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaServerItem newMediaServerItem = getNewMediaServerItem(device); | 
|---|
|  |  |  | MediaServer newMediaServerItem = getNewMediaServerItem(device); | 
|---|
|  |  |  | play(newMediaServerItem, deviceId, channelId, null, (code, msg, data)->{ | 
|---|
|  |  |  | if (code == InviteErrorCode.SUCCESS.getCode()) { | 
|---|
|  |  |  | InviteInfo inviteInfoForPlay = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId); | 
|---|
|  |  |  | if (inviteInfoForPlay != null && inviteInfoForPlay.getStreamInfo() != null) { | 
|---|
|  |  |  | getSnap(deviceId, channelId, fileName, errorCallback); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | errorCallback.run(InviteErrorCode.FAIL.getCode(), InviteErrorCode.FAIL.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | errorCallback.run(InviteErrorCode.FAIL.getCode(), InviteErrorCode.FAIL.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (code == InviteErrorCode.SUCCESS.getCode()) { | 
|---|
|  |  |  | InviteInfo inviteInfoForPlay = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId); | 
|---|
|  |  |  | if (inviteInfoForPlay != null && inviteInfoForPlay.getStreamInfo() != null) { | 
|---|
|  |  |  | getSnap(deviceId, channelId, fileName, errorCallback); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | errorCallback.run(InviteErrorCode.FAIL.getCode(), InviteErrorCode.FAIL.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | errorCallback.run(InviteErrorCode.FAIL.getCode(), InviteErrorCode.FAIL.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|