| | |
| | | package com.genersoft.iot.vmp.media.service.impl; |
| | | |
| | | import com.alibaba.fastjson2.JSON; |
| | | import com.alibaba.fastjson2.JSONArray; |
| | | import com.alibaba.fastjson2.JSONObject; |
| | | import com.baomidou.dynamic.datasource.annotation.DS; |
| | | import com.genersoft.iot.vmp.common.CommonCallback; |
| | | import com.genersoft.iot.vmp.common.StreamInfo; |
| | | import com.genersoft.iot.vmp.common.VideoManagerConstants; |
| | | import com.genersoft.iot.vmp.conf.DynamicTask; |
| | | import com.genersoft.iot.vmp.conf.SipConfig; |
| | | import com.genersoft.iot.vmp.conf.MediaConfig; |
| | | import com.genersoft.iot.vmp.conf.UserSetting; |
| | | import com.genersoft.iot.vmp.conf.exception.ControllerException; |
| | | import com.genersoft.iot.vmp.gb28181.event.EventPublisher; |
| | | import com.genersoft.iot.vmp.gb28181.bean.SendRtpItem; |
| | | import com.genersoft.iot.vmp.gb28181.session.SSRCFactory; |
| | | import com.genersoft.iot.vmp.media.bean.MediaInfo; |
| | | import com.genersoft.iot.vmp.media.bean.MediaServer; |
| | | import com.genersoft.iot.vmp.media.event.media.MediaArrivalEvent; |
| | | import com.genersoft.iot.vmp.media.event.media.MediaDepartureEvent; |
| | | import com.genersoft.iot.vmp.media.event.mediaServer.MediaServerChangeEvent; |
| | | import com.genersoft.iot.vmp.media.event.mediaServer.MediaServerDeleteEvent; |
| | | import com.genersoft.iot.vmp.media.service.IMediaNodeServerService; |
| | | import com.genersoft.iot.vmp.media.zlm.*; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.MediaServerItem; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.ServerKeepaliveData; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.ZLMServerConfig; |
| | | import com.genersoft.iot.vmp.service.IInviteStreamService; |
| | | import com.genersoft.iot.vmp.media.service.IMediaServerService; |
| | | import com.genersoft.iot.vmp.media.zlm.SendRtpPortManager; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.StreamAuthorityInfo; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.hook.OriginType; |
| | | import com.genersoft.iot.vmp.service.IInviteStreamService; |
| | | import com.genersoft.iot.vmp.service.bean.MediaServerLoad; |
| | | import com.genersoft.iot.vmp.service.bean.SSRCInfo; |
| | | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; |
| | |
| | | import com.genersoft.iot.vmp.utils.JsonUtil; |
| | | import com.genersoft.iot.vmp.utils.redis.RedisUtil; |
| | | import com.genersoft.iot.vmp.vmanager.bean.ErrorCode; |
| | | import com.genersoft.iot.vmp.vmanager.bean.WVPResult; |
| | | import okhttp3.OkHttpClient; |
| | | import okhttp3.Request; |
| | | import okhttp3.Response; |
| | | 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.context.ApplicationEventPublisher; |
| | | import org.springframework.context.event.EventListener; |
| | | import org.springframework.data.redis.core.RedisTemplate; |
| | | import org.springframework.jdbc.datasource.DataSourceTransactionManager; |
| | | import org.springframework.scheduling.annotation.Async; |
| | | import org.springframework.stereotype.Service; |
| | | import org.springframework.transaction.TransactionDefinition; |
| | | import org.springframework.transaction.TransactionStatus; |
| | | import org.springframework.util.ObjectUtils; |
| | | |
| | | import java.io.File; |
| | | import java.time.LocalDateTime; |
| | | import java.util.*; |
| | | |
| | |
| | | |
| | | private final static Logger logger = LoggerFactory.getLogger(MediaServerServiceImpl.class); |
| | | |
| | | private final String zlmKeepaliveKeyPrefix = "zlm-keepalive_"; |
| | | |
| | | @Autowired |
| | | private SipConfig sipConfig; |
| | | |
| | | @Autowired |
| | | private SSRCFactory ssrcFactory; |
| | | |
| | | @Value("${server.ssl.enabled:false}") |
| | | private boolean sslEnabled; |
| | | |
| | | @Value("${server.port}") |
| | | private Integer serverPort; |
| | | |
| | | @Autowired |
| | | private UserSetting userSetting; |
| | | |
| | | @Autowired |
| | | private SendRtpPortManager sendRtpPortManager; |
| | | |
| | | @Autowired |
| | | private AssistRESTfulUtils assistRESTfulUtils; |
| | | |
| | | @Autowired |
| | | private MediaServerMapper mediaServerMapper; |
| | | |
| | | @Autowired |
| | | private DataSourceTransactionManager dataSourceTransactionManager; |
| | | |
| | | @Autowired |
| | | private TransactionDefinition transactionDefinition; |
| | | |
| | | |
| | | @Autowired |
| | | private ZLMServerFactory zlmServerFactory; |
| | | |
| | | @Autowired |
| | | private EventPublisher publisher; |
| | | |
| | | @Autowired |
| | | private DynamicTask dynamicTask; |
| | | |
| | | @Autowired |
| | | private IRedisCatchStorage redisCatchStorage; |
| | |
| | | @Autowired |
| | | private Map<String, IMediaNodeServerService> nodeServerServiceMap; |
| | | |
| | | @Autowired |
| | | private ApplicationEventPublisher applicationEventPublisher; |
| | | |
| | | @Autowired |
| | | private MediaConfig mediaConfig; |
| | | |
| | | @Autowired |
| | | private SendRtpPortManager sendRtpPortManager; |
| | | |
| | | |
| | | |
| | | /** |
| | | * 流到来的处理 |
| | | */ |
| | | @Async("taskExecutor") |
| | | @org.springframework.context.event.EventListener |
| | | public void onApplicationEvent(MediaArrivalEvent event) { |
| | | if ("rtsp".equals(event.getSchema())) { |
| | | logger.info("流变化:注册 app->{}, stream->{}", event.getApp(), event.getStream()); |
| | | addCount(event.getMediaServer().getId()); |
| | | String type = OriginType.values()[event.getMediaInfo().getOriginType()].getType(); |
| | | redisCatchStorage.addStream(event.getMediaServer(), type, event.getApp(), event.getStream(), event.getMediaInfo()); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 流离开的处理 |
| | | */ |
| | | @Async("taskExecutor") |
| | | @EventListener |
| | | public void onApplicationEvent(MediaDepartureEvent event) { |
| | | if ("rtsp".equals(event.getSchema())) { |
| | | logger.info("流变化:注销, app->{}, stream->{}", event.getApp(), event.getStream()); |
| | | removeCount(event.getMediaServer().getId()); |
| | | MediaInfo mediaInfo = redisCatchStorage.getStreamInfo( |
| | | event.getApp(), event.getStream(), event.getMediaServer().getId()); |
| | | if (mediaInfo == null) { |
| | | return; |
| | | } |
| | | String type = OriginType.values()[mediaInfo.getOriginType()].getType(); |
| | | redisCatchStorage.removeStream(mediaInfo.getMediaServer().getId(), type, event.getApp(), event.getStream()); |
| | | } |
| | | |
| | | } |
| | | |
| | | |
| | | /** |
| | | * 初始化 |
| | | */ |
| | | @Override |
| | | public void updateVmServer(List<MediaServerItem> mediaServerItemList) { |
| | | public void updateVmServer(List<MediaServer> mediaServerList) { |
| | | logger.info("[媒体服务节点] 缓存初始化 "); |
| | | for (MediaServerItem mediaServerItem : mediaServerItemList) { |
| | | if (ObjectUtils.isEmpty(mediaServerItem.getId())) { |
| | | for (MediaServer mediaServer : mediaServerList) { |
| | | if (ObjectUtils.isEmpty(mediaServer.getId())) { |
| | | continue; |
| | | } |
| | | // 更新 |
| | | if (!ssrcFactory.hasMediaServerSSRC(mediaServerItem.getId())) { |
| | | ssrcFactory.initMediaServerSSRC(mediaServerItem.getId(), null); |
| | | if (!ssrcFactory.hasMediaServerSSRC(mediaServer.getId())) { |
| | | ssrcFactory.initMediaServerSSRC(mediaServer.getId(), null); |
| | | } |
| | | // 查询redis是否存在此mediaServer |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + mediaServerItem.getId(); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + ":" + mediaServer.getId(); |
| | | Boolean hasKey = redisTemplate.hasKey(key); |
| | | if (hasKey != null && ! hasKey) { |
| | | redisTemplate.opsForValue().set(key, mediaServerItem); |
| | | redisTemplate.opsForValue().set(key, mediaServer); |
| | | } |
| | | } |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public SSRCInfo openRTPServer(MediaServerItem mediaServerItem, String streamId, String presetSsrc, boolean ssrcCheck, |
| | | boolean isPlayback, Integer port, Boolean onlyAuto, Boolean reUsePort, Integer tcpMode) { |
| | | if (mediaServerItem == null || mediaServerItem.getId() == null) { |
| | | logger.info("[openRTPServer] 失败, mediaServerItem == null || mediaServerItem.getId() == null"); |
| | | public SSRCInfo openRTPServer(MediaServer mediaServer, String streamId, String presetSsrc, boolean ssrcCheck, |
| | | boolean isPlayback, Integer port, Boolean onlyAuto, Boolean disableAudio, Boolean reUsePort, Integer tcpMode) { |
| | | if (mediaServer == null || mediaServer.getId() == null) { |
| | | logger.info("[openRTPServer] 失败, mediaServer == null || mediaServer.getId() == null"); |
| | | return null; |
| | | } |
| | | // 获取mediaServer可用的ssrc |
| | |
| | | ssrc = presetSsrc; |
| | | }else { |
| | | if (isPlayback) { |
| | | ssrc = ssrcFactory.getPlayBackSsrc(mediaServerItem.getId()); |
| | | ssrc = ssrcFactory.getPlayBackSsrc(mediaServer.getId()); |
| | | }else { |
| | | ssrc = ssrcFactory.getPlaySsrc(mediaServerItem.getId()); |
| | | ssrc = ssrcFactory.getPlaySsrc(mediaServer.getId()); |
| | | } |
| | | } |
| | | |
| | |
| | | logger.warn("[openRTPServer] 平台对接时下级可能自定义ssrc,但是tcp模式zlm收流目前无法更新ssrc,可能收流超时,此时请使用udp收流或者关闭ssrc校验"); |
| | | } |
| | | int rtpServerPort; |
| | | if (mediaServerItem.isRtpEnable()) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | if (mediaServer.isRtpEnable()) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[openRTPServer] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[openRTPServer] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return null; |
| | | } |
| | | rtpServerPort = mediaNodeServerService.createRTPServer(mediaServerItem, streamId, ssrcCheck ? Long.parseLong(ssrc) : 0, port, onlyAuto, reUsePort, tcpMode); |
| | | rtpServerPort = mediaNodeServerService.createRTPServer(mediaServer, streamId, ssrcCheck ? Long.parseLong(ssrc) : 0, port, onlyAuto, disableAudio, reUsePort, tcpMode); |
| | | } else { |
| | | rtpServerPort = mediaServerItem.getRtpProxyPort(); |
| | | rtpServerPort = mediaServer.getRtpProxyPort(); |
| | | } |
| | | return new SSRCInfo(rtpServerPort, ssrc, streamId); |
| | | } |
| | | |
| | | @Override |
| | | public SSRCInfo openRTPServer(MediaServerItem mediaServerItem, String streamId, String ssrc, boolean ssrcCheck, boolean isPlayback, Integer port, Boolean onlyAuto) { |
| | | return openRTPServer(mediaServerItem, streamId, ssrc, ssrcCheck, isPlayback, port, onlyAuto, null, 0); |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public void closeRTPServer(MediaServerItem mediaServerItem, String streamId) { |
| | | if (mediaServerItem == null) { |
| | | public void closeRTPServer(MediaServer mediaServer, String streamId) { |
| | | if (mediaServer == null) { |
| | | return; |
| | | } |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[closeRTPServer] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[closeRTPServer] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | mediaNodeServerService.closeRtpServer(mediaServerItem, streamId); |
| | | mediaNodeServerService.closeRtpServer(mediaServer, streamId); |
| | | } |
| | | |
| | | @Override |
| | | public void closeRTPServer(MediaServerItem mediaServerItem, String streamId, CommonCallback<Boolean> callback) { |
| | | if (mediaServerItem == null) { |
| | | public void closeRTPServer(MediaServer mediaServer, String streamId, CommonCallback<Boolean> callback) { |
| | | if (mediaServer == null) { |
| | | callback.run(false); |
| | | return; |
| | | } |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[closeRTPServer] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[closeRTPServer] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | mediaNodeServerService.closeRtpServer(mediaServerItem, streamId, callback); |
| | | mediaNodeServerService.closeRtpServer(mediaServer, streamId, callback); |
| | | } |
| | | |
| | | @Override |
| | | public void closeRTPServer(String mediaServerId, String streamId) { |
| | | MediaServerItem mediaServerItem = this.getOne(mediaServerId); |
| | | if (mediaServerItem == null) { |
| | | MediaServer mediaServer = this.getOne(mediaServerId); |
| | | if (mediaServer == null) { |
| | | return; |
| | | } |
| | | if (mediaServerItem.isRtpEnable()) { |
| | | closeRTPServer(mediaServerItem, streamId); |
| | | if (mediaServer.isRtpEnable()) { |
| | | closeRTPServer(mediaServer, streamId); |
| | | } |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[closeRTPServer] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[closeRTPServer] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | mediaNodeServerService.closeStreams(mediaServerItem, "rtp", streamId); |
| | | mediaNodeServerService.closeStreams(mediaServer, "rtp", streamId); |
| | | } |
| | | |
| | | @Override |
| | | public Boolean updateRtpServerSSRC(MediaServerItem mediaServerItem, String streamId, String ssrc) { |
| | | if (mediaServerItem == null) { |
| | | public Boolean updateRtpServerSSRC(MediaServer mediaServer, String streamId, String ssrc) { |
| | | if (mediaServer == null) { |
| | | return false; |
| | | } |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[updateRtpServerSSRC] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[updateRtpServerSSRC] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.updateRtpServerSSRC(mediaServerItem, streamId, ssrc); |
| | | return mediaNodeServerService.updateRtpServerSSRC(mediaServer, streamId, ssrc); |
| | | } |
| | | |
| | | @Override |
| | | public void releaseSsrc(String mediaServerItemId, String ssrc) { |
| | | MediaServerItem mediaServerItem = getOne(mediaServerItemId); |
| | | if (mediaServerItem == null || ssrc == null) { |
| | | public void releaseSsrc(String mediaServerId, String ssrc) { |
| | | MediaServer mediaServer = getOne(mediaServerId); |
| | | if (mediaServer == null || ssrc == null) { |
| | | return; |
| | | } |
| | | ssrcFactory.releaseSsrc(mediaServerItemId, ssrc); |
| | | ssrcFactory.releaseSsrc(mediaServerId, ssrc); |
| | | } |
| | | |
| | | /** |
| | | * 媒体服务节点 重启后重置他的推流信息, TODO 给正在使用的设备发送停止命令 |
| | | */ |
| | | @Override |
| | | public void clearRTPServer(MediaServerItem mediaServerItem) { |
| | | ssrcFactory.reset(mediaServerItem.getId()); |
| | | public void clearRTPServer(MediaServer mediaServer) { |
| | | ssrcFactory.reset(mediaServer.getId()); |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public void update(MediaServerItem mediaSerItem) { |
| | | public void update(MediaServer mediaSerItem) { |
| | | mediaServerMapper.update(mediaSerItem); |
| | | MediaServerItem mediaServerItemInRedis = getOne(mediaSerItem.getId()); |
| | | MediaServerItem mediaServerItemInDataBase = mediaServerMapper.queryOne(mediaSerItem.getId()); |
| | | if (mediaServerItemInRedis == null || !ssrcFactory.hasMediaServerSSRC(mediaSerItem.getId())) { |
| | | ssrcFactory.initMediaServerSSRC(mediaServerItemInDataBase.getId(),null); |
| | | MediaServer mediaServerInRedis = getOne(mediaSerItem.getId()); |
| | | MediaServer mediaServerInDataBase = mediaServerMapper.queryOne(mediaSerItem.getId()); |
| | | if (mediaServerInDataBase == null) { |
| | | return; |
| | | } |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + mediaServerItemInDataBase.getId(); |
| | | redisTemplate.opsForValue().set(key, mediaServerItemInDataBase); |
| | | if (mediaSerItem.isStatus()) { |
| | | resetOnlineServerItem(mediaSerItem); |
| | | mediaServerInDataBase.setStatus(mediaSerItem.isStatus()); |
| | | if (mediaServerInRedis == null || !ssrcFactory.hasMediaServerSSRC(mediaServerInDataBase.getId())) { |
| | | ssrcFactory.initMediaServerSSRC(mediaServerInDataBase.getId(),null); |
| | | } |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + ":" + mediaServerInDataBase.getId(); |
| | | redisTemplate.opsForValue().set(key, mediaServerInDataBase); |
| | | if (mediaServerInDataBase.isStatus()) { |
| | | resetOnlineServerItem(mediaServerInDataBase); |
| | | }else { |
| | | // 发送事件 |
| | | MediaServerChangeEvent event = new MediaServerChangeEvent(this); |
| | | event.setMediaServerItemList(mediaServerInDataBase); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public List<MediaServerItem> getAll() { |
| | | List<MediaServerItem> result = new ArrayList<>(); |
| | | List<Object> mediaServerKeys = RedisUtil.scan(redisTemplate, String.format("%S*", VideoManagerConstants.MEDIA_SERVER_PREFIX+ userSetting.getServerId() + "_" )); |
| | | String onlineKey = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | public List<MediaServer> getAllOnlineList() { |
| | | List<MediaServer> result = new ArrayList<>(); |
| | | List<Object> mediaServerKeys = RedisUtil.scan(redisTemplate, String.format("%S*", VideoManagerConstants.MEDIA_SERVER_PREFIX+ userSetting.getServerId() + ":" )); |
| | | String onlineKey = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | for (Object mediaServerKey : mediaServerKeys) { |
| | | String key = (String) mediaServerKey; |
| | | MediaServerItem mediaServerItem = JsonUtil.redisJsonToObject(redisTemplate, key, MediaServerItem.class); |
| | | if (Objects.isNull(mediaServerItem)) { |
| | | MediaServer mediaServer = JsonUtil.redisJsonToObject(redisTemplate, key, MediaServer.class); |
| | | if (Objects.isNull(mediaServer)) { |
| | | continue; |
| | | } |
| | | // 检查状态 |
| | | Double aDouble = redisTemplate.opsForZSet().score(onlineKey, mediaServerItem.getId()); |
| | | Double aDouble = redisTemplate.opsForZSet().score(onlineKey, mediaServer.getId()); |
| | | if (aDouble != null) { |
| | | mediaServerItem.setStatus(true); |
| | | mediaServer.setStatus(true); |
| | | } |
| | | result.add(mediaServerItem); |
| | | result.add(mediaServer); |
| | | } |
| | | result.sort((serverItem1, serverItem2)->{ |
| | | int sortResult = 0; |
| | |
| | | return result; |
| | | } |
| | | |
| | | @Override |
| | | public List<MediaServer> getAll() { |
| | | List<MediaServer> mediaServerList = mediaServerMapper.queryAll(); |
| | | if (mediaServerList.isEmpty()) { |
| | | return new ArrayList<>(); |
| | | } |
| | | for (MediaServer mediaServer : mediaServerList) { |
| | | MediaServer mediaServerInRedis = getOne(mediaServer.getId()); |
| | | if (mediaServerInRedis != null) { |
| | | mediaServer.setStatus(mediaServerInRedis.isStatus()); |
| | | } |
| | | } |
| | | return mediaServerList; |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public List<MediaServerItem> getAllFromDatabase() { |
| | | public List<MediaServer> getAllFromDatabase() { |
| | | return mediaServerMapper.queryAll(); |
| | | } |
| | | |
| | | @Override |
| | | public List<MediaServerItem> getAllOnline() { |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | public List<MediaServer> getAllOnline() { |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | Set<Object> mediaServerIdSet = redisTemplate.opsForZSet().reverseRange(key, 0, -1); |
| | | |
| | | List<MediaServerItem> result = new ArrayList<>(); |
| | | List<MediaServer> result = new ArrayList<>(); |
| | | if (mediaServerIdSet != null && mediaServerIdSet.size() > 0) { |
| | | for (Object mediaServerId : mediaServerIdSet) { |
| | | String mediaServerIdStr = (String) mediaServerId; |
| | | String serverKey = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + mediaServerIdStr; |
| | | result.add((MediaServerItem) redisTemplate.opsForValue().get(serverKey)); |
| | | String serverKey = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + ":" + mediaServerIdStr; |
| | | result.add((MediaServer) redisTemplate.opsForValue().get(serverKey)); |
| | | } |
| | | } |
| | | Collections.reverse(result); |
| | |
| | | /** |
| | | * 获取单个媒体服务节点服务器 |
| | | * @param mediaServerId 服务id |
| | | * @return MediaServerItem |
| | | * @return mediaServer |
| | | */ |
| | | @Override |
| | | public MediaServerItem getOne(String mediaServerId) { |
| | | public MediaServer getOne(String mediaServerId) { |
| | | if (mediaServerId == null) { |
| | | return null; |
| | | } |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + mediaServerId; |
| | | return JsonUtil.redisJsonToObject(redisTemplate, key, MediaServerItem.class); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + ":" + mediaServerId; |
| | | return JsonUtil.redisJsonToObject(redisTemplate, key, MediaServer.class); |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public MediaServerItem getDefaultMediaServer() { |
| | | public MediaServer getDefaultMediaServer() { |
| | | return mediaServerMapper.queryDefault(); |
| | | } |
| | | |
| | | @Override |
| | | public void clearMediaServerForOnline() { |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | redisTemplate.delete(key); |
| | | } |
| | | |
| | | @Override |
| | | public void add(MediaServerItem mediaServerItem) { |
| | | mediaServerItem.setCreateTime(DateUtil.getNow()); |
| | | mediaServerItem.setUpdateTime(DateUtil.getNow()); |
| | | mediaServerItem.setHookAliveInterval(30f); |
| | | if (mediaServerItem.getType() == null) { |
| | | logger.info("[添加媒体节点] 失败, mediaServerItem的类型:为空"); |
| | | public void add(MediaServer mediaServer) { |
| | | mediaServer.setCreateTime(DateUtil.getNow()); |
| | | mediaServer.setUpdateTime(DateUtil.getNow()); |
| | | if (mediaServer.getHookAliveInterval() == null || mediaServer.getHookAliveInterval() == 0F) { |
| | | mediaServer.setHookAliveInterval(10F); |
| | | } |
| | | if (mediaServer.getType() == null) { |
| | | logger.info("[添加媒体节点] 失败, mediaServer的类型:为空"); |
| | | return; |
| | | } |
| | | if (mediaServerMapper.queryOne(mediaServerItem.getId()) != null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(),"保存失败,媒体服务ID [ " + mediaServerItem.getId() + " ] 已存在,请修改媒体服务器配置"); |
| | | if (mediaServerMapper.queryOne(mediaServer.getId()) != null) { |
| | | logger.info("[添加媒体节点] 失败, 媒体服务ID已存在,请修改媒体服务器配置, {}", mediaServer.getId()); |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(),"保存失败,媒体服务ID [ " + mediaServer.getId() + " ] 已存在,请修改媒体服务器配置"); |
| | | } |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[添加媒体节点] 失败, mediaServerItem的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | logger.info("[添加媒体节点] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | if (mediaNodeServerService.checkNodeId(mediaServerItem)) { |
| | | mediaServerMapper.add(mediaServerItem); |
| | | mediaNodeServerService.online(mediaServerItem); |
| | | |
| | | mediaServerMapper.add(mediaServer); |
| | | if (mediaServer.isStatus()) { |
| | | mediaNodeServerService.online(mediaServer); |
| | | }else { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(),"保存失败,媒体服务ID [ " + mediaServerItem.getId() + " ] 已存在,请修改媒体服务器配置"); |
| | | // 发送事件 |
| | | MediaServerChangeEvent event = new MediaServerChangeEvent(this); |
| | | event.setMediaServerItemList(mediaServer); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | public int addToDatabase(MediaServerItem mediaSerItem) { |
| | | return mediaServerMapper.add(mediaSerItem); |
| | | } |
| | | |
| | | @Override |
| | | public int updateToDatabase(MediaServerItem mediaSerItem) { |
| | | int result = 0; |
| | | if (mediaSerItem.isDefaultServer()) { |
| | | TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition); |
| | | int delResult = mediaServerMapper.delDefault(); |
| | | if (delResult == 0) { |
| | | logger.error("移除数据库默认媒体服务节点节点失败"); |
| | | //事务回滚 |
| | | dataSourceTransactionManager.rollback(transactionStatus); |
| | | return 0; |
| | | } |
| | | result = mediaServerMapper.add(mediaSerItem); |
| | | dataSourceTransactionManager.commit(transactionStatus); //手动提交 |
| | | }else { |
| | | result = mediaServerMapper.update(mediaSerItem); |
| | | } |
| | | return result; |
| | | } |
| | | |
| | | /** |
| | | * 处理媒体服务节点上线 |
| | | * @param zlmServerConfig 媒体服务节点上线携带的参数 |
| | | */ |
| | | @Override |
| | | public void zlmServerOnline(ZLMServerConfig zlmServerConfig) { |
| | | |
| | | MediaServerItem serverItem = mediaServerMapper.queryOne(zlmServerConfig.getGeneralMediaServerId()); |
| | | if (serverItem == null) { |
| | | logger.warn("[未注册的媒体服务节点] 拒接接入:{}来自{}:{}", zlmServerConfig.getGeneralMediaServerId(), zlmServerConfig.getIp(),zlmServerConfig.getHttpPort() ); |
| | | logger.warn("请检查媒体服务节点的ID配置是否与WVP的<media.id>一致"); |
| | | return; |
| | | }else { |
| | | logger.info("[媒体服务节点] 正在连接 : {} -> {}:{}", |
| | | zlmServerConfig.getGeneralMediaServerId(), zlmServerConfig.getIp(), zlmServerConfig.getHttpPort()); |
| | | } |
| | | serverItem.setHookAliveInterval(zlmServerConfig.getHookAliveInterval()); |
| | | if (serverItem.getHttpPort() == 0) { |
| | | serverItem.setHttpPort(zlmServerConfig.getHttpPort()); |
| | | } |
| | | if (serverItem.getHttpSSlPort() == 0) { |
| | | serverItem.setHttpSSlPort(zlmServerConfig.getHttpSSLport()); |
| | | } |
| | | if (serverItem.getRtmpPort() == 0) { |
| | | serverItem.setRtmpPort(zlmServerConfig.getRtmpPort()); |
| | | } |
| | | if (serverItem.getRtmpSSlPort() == 0) { |
| | | serverItem.setRtmpSSlPort(zlmServerConfig.getRtmpSslPort()); |
| | | } |
| | | if (serverItem.getRtspPort() == 0) { |
| | | serverItem.setRtspPort(zlmServerConfig.getRtspPort()); |
| | | } |
| | | if (serverItem.getRtspSSLPort() == 0) { |
| | | serverItem.setRtspSSLPort(zlmServerConfig.getRtspSSlport()); |
| | | } |
| | | if (serverItem.getRtpProxyPort() == 0) { |
| | | serverItem.setRtpProxyPort(zlmServerConfig.getRtpProxyPort()); |
| | | } |
| | | serverItem.setStatus(true); |
| | | |
| | | if (ObjectUtils.isEmpty(serverItem.getId())) { |
| | | logger.warn("[未注册的媒体服务节点] serverItem缺少ID, 无法接入:{}:{}", zlmServerConfig.getIp(),zlmServerConfig.getHttpPort() ); |
| | | return; |
| | | } |
| | | mediaServerMapper.update(serverItem); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + zlmServerConfig.getGeneralMediaServerId(); |
| | | if (!ssrcFactory.hasMediaServerSSRC(serverItem.getId())) { |
| | | ssrcFactory.initMediaServerSSRC(zlmServerConfig.getGeneralMediaServerId(), null); |
| | | } |
| | | redisTemplate.opsForValue().set(key, serverItem); |
| | | resetOnlineServerItem(serverItem); |
| | | |
| | | |
| | | if (serverItem.isAutoConfig()) { |
| | | setZLMConfig(serverItem, "0".equals(zlmServerConfig.getHookEnable())); |
| | | } |
| | | final String zlmKeepaliveKey = zlmKeepaliveKeyPrefix + serverItem.getId(); |
| | | dynamicTask.stop(zlmKeepaliveKey); |
| | | dynamicTask.startDelay(zlmKeepaliveKey, new KeepAliveTimeoutRunnable(serverItem), (serverItem.getHookAliveInterval().intValue() + 5) * 1000); |
| | | publisher.mediaServerOnlineEventPublish(serverItem.getId()); |
| | | |
| | | logger.info("[媒体服务节点] 连接成功 {} - {}:{} ", |
| | | zlmServerConfig.getGeneralMediaServerId(), zlmServerConfig.getIp(), zlmServerConfig.getHttpPort()); |
| | | } |
| | | |
| | | class KeepAliveTimeoutRunnable implements Runnable{ |
| | | |
| | | private MediaServerItem serverItem; |
| | | |
| | | public KeepAliveTimeoutRunnable(MediaServerItem serverItem) { |
| | | this.serverItem = serverItem; |
| | | } |
| | | |
| | | @Override |
| | | public void run() { |
| | | logger.info("[媒体服务节点心跳到期]:" + serverItem.getId()); |
| | | // 发起http请求验证zlm是否确实无法连接,如果确实无法连接则发送离线事件,否则不作处理 |
| | | JSONObject mediaServerConfig = zlmresTfulUtils.getMediaServerConfig(serverItem); |
| | | if (mediaServerConfig != null && mediaServerConfig.getInteger("code") == 0) { |
| | | logger.info("[媒体服务节点心跳到期]:{}验证后媒体服务节点仍在线,恢复心跳信息,请检查媒体服务节点是否可以正常向wvp发送心跳", serverItem.getId()); |
| | | // 添加媒体服务节点信息 |
| | | updateMediaServerKeepalive(serverItem.getId(), null); |
| | | }else { |
| | | publisher.zlmOfflineEventPublish(serverItem.getId()); |
| | | } |
| | | } |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public void zlmServerOffline(String mediaServerId) { |
| | | delete(mediaServerId); |
| | | final String zlmKeepaliveKey = zlmKeepaliveKeyPrefix + mediaServerId; |
| | | dynamicTask.stop(zlmKeepaliveKey); |
| | | } |
| | | |
| | | @Override |
| | | public void resetOnlineServerItem(MediaServerItem serverItem) { |
| | | public void resetOnlineServerItem(MediaServer serverItem) { |
| | | // 更新缓存 |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | // 使用zset的分数作为当前并发量, 默认值设置为0 |
| | | if (redisTemplate.opsForZSet().score(key, serverItem.getId()) == null) { // 不存在则设置默认值 已存在则重置 |
| | | redisTemplate.opsForZSet().add(key, serverItem.getId(), 0L); |
| | |
| | | } |
| | | } |
| | | |
| | | private int getMediaList(MediaServerItem serverItem) { |
| | | private int getMediaList(MediaServer serverItem) { |
| | | |
| | | return 0; |
| | | } |
| | |
| | | if (mediaServerId == null) { |
| | | return; |
| | | } |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | redisTemplate.opsForZSet().incrementScore(key, mediaServerId, 1); |
| | | |
| | | } |
| | | |
| | | @Override |
| | | public void removeCount(String mediaServerId) { |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | redisTemplate.opsForZSet().incrementScore(key, mediaServerId, - 1); |
| | | } |
| | | |
| | | /** |
| | | * 获取负载最低的节点 |
| | | * @return MediaServerItem |
| | | * @return mediaServer |
| | | */ |
| | | @Override |
| | | public MediaServerItem getMediaServerForMinimumLoad(Boolean hasAssist) { |
| | | String key = VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(); |
| | | public MediaServer getMediaServerForMinimumLoad(Boolean hasAssist) { |
| | | String key = VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(); |
| | | Long size = redisTemplate.opsForZSet().zCard(key); |
| | | if (size == null || size == 0) { |
| | | logger.info("获取负载最低的节点时无在线节点"); |
| | |
| | | // 获取分数最低的,及并发最低的 |
| | | Set<Object> objects = redisTemplate.opsForZSet().range(key, 0, -1); |
| | | ArrayList<Object> mediaServerObjectS = new ArrayList<>(objects); |
| | | MediaServerItem mediaServerItem = null; |
| | | MediaServer mediaServer = null; |
| | | if (hasAssist == null) { |
| | | String mediaServerId = (String)mediaServerObjectS.get(0); |
| | | mediaServerItem = getOne(mediaServerId); |
| | | mediaServer = getOne(mediaServerId); |
| | | }else if (hasAssist) { |
| | | for (Object mediaServerObject : mediaServerObjectS) { |
| | | String mediaServerId = (String)mediaServerObject; |
| | | MediaServerItem serverItem = getOne(mediaServerId); |
| | | MediaServer serverItem = getOne(mediaServerId); |
| | | if (serverItem.getRecordAssistPort() > 0) { |
| | | mediaServerItem = serverItem; |
| | | mediaServer = serverItem; |
| | | break; |
| | | } |
| | | } |
| | | }else if (!hasAssist) { |
| | | for (Object mediaServerObject : mediaServerObjectS) { |
| | | String mediaServerId = (String)mediaServerObject; |
| | | MediaServerItem serverItem = getOne(mediaServerId); |
| | | MediaServer serverItem = getOne(mediaServerId); |
| | | if (serverItem.getRecordAssistPort() == 0) { |
| | | mediaServerItem = serverItem; |
| | | mediaServer = serverItem; |
| | | break; |
| | | } |
| | | } |
| | | } |
| | | |
| | | return mediaServerItem; |
| | | return mediaServer; |
| | | } |
| | | |
| | | /** |
| | | * 对媒体服务节点服务器进行基础配置 |
| | | * @param mediaServerItem 服务ID |
| | | * @param restart 是否重启媒体服务节点 |
| | | */ |
| | | @Override |
| | | public void setZLMConfig(MediaServerItem mediaServerItem, boolean restart) { |
| | | logger.info("[媒体服务节点] 正在设置 :{} -> {}:{}", |
| | | mediaServerItem.getId(), mediaServerItem.getIp(), mediaServerItem.getHttpPort()); |
| | | String protocol = sslEnabled ? "https" : "http"; |
| | | String hookPrefix = String.format("%s://%s:%s/index/hook", protocol, mediaServerItem.getHookIp(), serverPort); |
| | | |
| | | Map<String, Object> param = new HashMap<>(); |
| | | param.put("api.secret",mediaServerItem.getSecret()); // -profile:v Baseline |
| | | if (mediaServerItem.getRtspPort() != 0) { |
| | | param.put("ffmpeg.snap", "%s -rtsp_transport tcp -i %s -y -f mjpeg -frames:v 1 %s"); |
| | | } |
| | | param.put("hook.enable","1"); |
| | | param.put("hook.on_flow_report",""); |
| | | param.put("hook.on_play",String.format("%s/on_play", hookPrefix)); |
| | | param.put("hook.on_http_access",""); |
| | | param.put("hook.on_publish", String.format("%s/on_publish", hookPrefix)); |
| | | param.put("hook.on_record_ts",""); |
| | | param.put("hook.on_rtsp_auth",""); |
| | | param.put("hook.on_rtsp_realm",""); |
| | | param.put("hook.on_server_started",String.format("%s/on_server_started", hookPrefix)); |
| | | param.put("hook.on_shell_login",""); |
| | | param.put("hook.on_stream_changed",String.format("%s/on_stream_changed", hookPrefix)); |
| | | param.put("hook.on_stream_none_reader",String.format("%s/on_stream_none_reader", hookPrefix)); |
| | | param.put("hook.on_stream_not_found",String.format("%s/on_stream_not_found", hookPrefix)); |
| | | param.put("hook.on_server_keepalive",String.format("%s/on_server_keepalive", hookPrefix)); |
| | | param.put("hook.on_send_rtp_stopped",String.format("%s/on_send_rtp_stopped", hookPrefix)); |
| | | param.put("hook.on_rtp_server_timeout",String.format("%s/on_rtp_server_timeout", hookPrefix)); |
| | | param.put("hook.on_record_mp4",String.format("%s/on_record_mp4", hookPrefix)); |
| | | param.put("hook.timeoutSec","20"); |
| | | // 推流断开后可以在超时时间内重新连接上继续推流,这样播放器会接着播放。 |
| | | // 置0关闭此特性(推流断开会导致立即断开播放器) |
| | | // 此参数不应大于播放器超时时间 |
| | | // 优化此消息以更快的收到流注销事件 |
| | | param.put("protocol.continue_push_ms", "3000" ); |
| | | // 最多等待未初始化的Track时间,单位毫秒,超时之后会忽略未初始化的Track, 设置此选项优化那些音频错误的不规范流, |
| | | // 等zlm支持给每个rtpServer设置关闭音频的时候可以不设置此选项 |
| | | if (mediaServerItem.isRtpEnable() && !ObjectUtils.isEmpty(mediaServerItem.getRtpPortRange())) { |
| | | param.put("rtp_proxy.port_range", mediaServerItem.getRtpPortRange().replace(",", "-")); |
| | | } |
| | | |
| | | if (!ObjectUtils.isEmpty(mediaServerItem.getRecordPath())) { |
| | | File recordPathFile = new File(mediaServerItem.getRecordPath()); |
| | | param.put("protocol.mp4_save_path", recordPathFile.getParentFile().getPath()); |
| | | param.put("protocol.downloadRoot", recordPathFile.getParentFile().getPath()); |
| | | param.put("record.appName", recordPathFile.getName()); |
| | | } |
| | | |
| | | JSONObject responseJSON = zlmresTfulUtils.setServerConfig(mediaServerItem, param); |
| | | |
| | | if (responseJSON != null && responseJSON.getInteger("code") == 0) { |
| | | if (restart) { |
| | | logger.info("[媒体服务节点] 设置成功,开始重启以保证配置生效 {} -> {}:{}", |
| | | mediaServerItem.getId(), mediaServerItem.getIp(), mediaServerItem.getHttpPort()); |
| | | zlmresTfulUtils.restartServer(mediaServerItem); |
| | | }else { |
| | | logger.info("[媒体服务节点] 设置成功 {} -> {}:{}", |
| | | mediaServerItem.getId(), mediaServerItem.getIp(), mediaServerItem.getHttpPort()); |
| | | } |
| | | |
| | | |
| | | }else { |
| | | logger.info("[媒体服务节点] 设置媒体服务节点失败 {} -> {}:{}", |
| | | mediaServerItem.getId(), mediaServerItem.getIp(), mediaServerItem.getHttpPort()); |
| | | } |
| | | |
| | | |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public MediaServerItem checkMediaServer(String ip, int port, String secret) { |
| | | public MediaServer checkMediaServer(String ip, int port, String secret, String type) { |
| | | if (mediaServerMapper.queryOneByHostAndPort(ip, port) != null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "此连接已存在"); |
| | | } |
| | | MediaServerItem mediaServerItem = new MediaServerItem(); |
| | | mediaServerItem.setIp(ip); |
| | | mediaServerItem.setHttpPort(port); |
| | | mediaServerItem.setSecret(secret); |
| | | JSONObject responseJSON = zlmresTfulUtils.getMediaServerConfig(mediaServerItem); |
| | | if (responseJSON == null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "连接失败"); |
| | | |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(type); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[closeRTPServer] 失败, mediaServer的类型: {},未找到对应的实现类", type); |
| | | return null; |
| | | } |
| | | JSONArray data = responseJSON.getJSONArray("data"); |
| | | ZLMServerConfig zlmServerConfig = JSON.parseObject(JSON.toJSONString(data.get(0)), ZLMServerConfig.class); |
| | | if (zlmServerConfig == null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "读取配置失败"); |
| | | MediaServer mediaServer = mediaNodeServerService.checkMediaServer(ip, port, secret); |
| | | if (mediaServer != null) { |
| | | if (mediaServerMapper.queryOne(mediaServer.getId()) != null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "媒体服务ID [" + mediaServer.getId() + " ] 已存在,请修改媒体服务器配置"); |
| | | } |
| | | } |
| | | if (mediaServerMapper.queryOne(zlmServerConfig.getGeneralMediaServerId()) != null) { |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "媒体服务ID [" + zlmServerConfig.getGeneralMediaServerId() + " ] 已存在,请修改媒体服务器配置"); |
| | | } |
| | | mediaServerItem.setHttpSSlPort(zlmServerConfig.getHttpPort()); |
| | | mediaServerItem.setRtmpPort(zlmServerConfig.getRtmpPort()); |
| | | mediaServerItem.setRtmpSSlPort(zlmServerConfig.getRtmpSslPort()); |
| | | mediaServerItem.setRtspPort(zlmServerConfig.getRtspPort()); |
| | | mediaServerItem.setRtspSSLPort(zlmServerConfig.getRtspSSlport()); |
| | | mediaServerItem.setRtpProxyPort(zlmServerConfig.getRtpProxyPort()); |
| | | mediaServerItem.setStreamIp(ip); |
| | | mediaServerItem.setHookIp(sipConfig.getIp().split(",")[0]); |
| | | mediaServerItem.setSdpIp(ip); |
| | | return mediaServerItem; |
| | | return mediaServer; |
| | | } |
| | | |
| | | @Override |
| | |
| | | |
| | | @Override |
| | | public void delete(String id) { |
| | | redisTemplate.opsForZSet().remove(VideoManagerConstants.MEDIA_SERVERS_ONLINE_PREFIX + userSetting.getServerId(), id); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + id; |
| | | redisTemplate.delete(key); |
| | | } |
| | | @Override |
| | | public void deleteDb(String id){ |
| | | //同步删除数据库中的数据 |
| | | mediaServerMapper.delOne(id); |
| | | redisTemplate.opsForZSet().remove(VideoManagerConstants.ONLINE_MEDIA_SERVERS_PREFIX + userSetting.getServerId(), id); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + ":" + id; |
| | | redisTemplate.delete(key); |
| | | // 发送节点移除通知 |
| | | MediaServerDeleteEvent event = new MediaServerDeleteEvent(this); |
| | | event.setMediaServerId(id); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | |
| | | @Override |
| | | public void updateMediaServerKeepalive(String mediaServerId, ServerKeepaliveData data) { |
| | | MediaServerItem mediaServerItem = getOne(mediaServerId); |
| | | if (mediaServerItem == null) { |
| | | // 缓存不存在,从数据库查询,如果数据库不存在则是错误的 |
| | | mediaServerItem = getOneFromDatabase(mediaServerId); |
| | | if (mediaServerItem == null) { |
| | | logger.warn("[更新媒体服务节点 保活信息] 流媒体{}尚未加入使用,请检查节点中是否含有此流媒体 ", mediaServerId); |
| | | return; |
| | | } |
| | | // 媒体服务节点连接重试 |
| | | logger.warn("[更新媒体服务节点 保活信息]尝试链接zml id {}", mediaServerId); |
| | | ssrcFactory.initMediaServerSSRC(mediaServerItem.getId(), null); |
| | | String key = VideoManagerConstants.MEDIA_SERVER_PREFIX + userSetting.getServerId() + "_" + mediaServerItem.getId(); |
| | | redisTemplate.opsForValue().set(key, mediaServerItem); |
| | | resetOnlineServerItem(mediaServerItem); |
| | | clearRTPServer(mediaServerItem); |
| | | } |
| | | final String zlmKeepaliveKey = zlmKeepaliveKeyPrefix + mediaServerItem.getId(); |
| | | dynamicTask.stop(zlmKeepaliveKey); |
| | | dynamicTask.startDelay(zlmKeepaliveKey, new KeepAliveTimeoutRunnable(mediaServerItem), (mediaServerItem.getHookAliveInterval().intValue() + 5) * 1000); |
| | | } |
| | | |
| | | private MediaServerItem getOneFromDatabase(String mediaServerId) { |
| | | public MediaServer getOneFromDatabase(String mediaServerId) { |
| | | return mediaServerMapper.queryOne(mediaServerId); |
| | | } |
| | | |
| | | @Override |
| | | public void syncCatchFromDatabase() { |
| | | List<MediaServerItem> allInCatch = getAll(); |
| | | List<MediaServerItem> allInDatabase = mediaServerMapper.queryAll(); |
| | | Map<String, MediaServerItem> mediaServerItemMap = new HashMap<>(); |
| | | List<MediaServer> allInCatch = getAllOnlineList(); |
| | | List<MediaServer> allInDatabase = mediaServerMapper.queryAll(); |
| | | Map<String, MediaServer> mediaServerMap = new HashMap<>(); |
| | | |
| | | for (MediaServerItem mediaServerItem : allInDatabase) { |
| | | mediaServerItemMap.put(mediaServerItem.getId(), mediaServerItem); |
| | | for (MediaServer mediaServer : allInDatabase) { |
| | | mediaServerMap.put(mediaServer.getId(), mediaServer); |
| | | } |
| | | for (MediaServerItem mediaServerItem : allInCatch) { |
| | | for (MediaServer mediaServer : allInCatch) { |
| | | // 清除数据中不存在但redis缓存数据 |
| | | if (!mediaServerItemMap.containsKey(mediaServerItem.getId())) { |
| | | delete(mediaServerItem.getId()); |
| | | if (!mediaServerMap.containsKey(mediaServer.getId())) { |
| | | delete(mediaServer.getId()); |
| | | } |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | public MediaServerLoad getLoad(MediaServerItem mediaServerItem) { |
| | | public MediaServerLoad getLoad(MediaServer mediaServer) { |
| | | MediaServerLoad result = new MediaServerLoad(); |
| | | result.setId(mediaServerItem.getId()); |
| | | result.setPush(redisCatchStorage.getPushStreamCount(mediaServerItem.getId())); |
| | | result.setProxy(redisCatchStorage.getProxyStreamCount(mediaServerItem.getId())); |
| | | result.setId(mediaServer.getId()); |
| | | result.setPush(redisCatchStorage.getPushStreamCount(mediaServer.getId())); |
| | | result.setProxy(redisCatchStorage.getProxyStreamCount(mediaServer.getId())); |
| | | |
| | | result.setGbReceive(inviteStreamService.getStreamInfoCount(mediaServerItem.getId())); |
| | | result.setGbSend(redisCatchStorage.getGbSendCount(mediaServerItem.getId())); |
| | | result.setGbReceive(inviteStreamService.getStreamInfoCount(mediaServer.getId())); |
| | | result.setGbSend(redisCatchStorage.getGbSendCount(mediaServer.getId())); |
| | | return result; |
| | | } |
| | | |
| | | @Override |
| | | public List<MediaServerItem> getAllWithAssistPort() { |
| | | public List<MediaServer> getAllWithAssistPort() { |
| | | return mediaServerMapper.queryAllWithAssistPort(); |
| | | } |
| | | |
| | | |
| | | @Override |
| | | public boolean stopSendRtp(MediaServer mediaInfo, String app, String stream, String ssrc) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaInfo.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[stopSendRtp] 失败, mediaServer的类型: {},未找到对应的实现类", mediaInfo.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.stopSendRtp(mediaInfo, app, stream, ssrc); |
| | | } |
| | | |
| | | @Override |
| | | public boolean initStopSendRtp(MediaServer mediaInfo, String app, String stream, String ssrc) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaInfo.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[stopSendRtp] 失败, mediaServer的类型: {},未找到对应的实现类", mediaInfo.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.initStopSendRtp(mediaInfo, app, stream, ssrc); |
| | | } |
| | | |
| | | @Override |
| | | public boolean deleteRecordDirectory(MediaServer mediaServer, String app, String stream, String date, String fileName) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[stopSendRtp] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.deleteRecordDirectory(mediaServer, app, stream, date, fileName); |
| | | } |
| | | |
| | | @Override |
| | | public List<StreamInfo> getMediaList(MediaServer mediaServer, String app, String stream, String callId) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[getMediaList] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return new ArrayList<>(); |
| | | } |
| | | return mediaNodeServerService.getMediaList(mediaServer, app, stream, callId); |
| | | } |
| | | |
| | | @Override |
| | | public Boolean connectRtpServer(MediaServer mediaServer, String address, int port, String stream) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[connectRtpServer] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.connectRtpServer(mediaServer, address, port, stream); |
| | | } |
| | | |
| | | @Override |
| | | public void getSnap(MediaServer mediaServer, String streamUrl, int timeoutSec, int expireSec, String path, String fileName) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[getSnap] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | mediaNodeServerService.getSnap(mediaServer, streamUrl, timeoutSec, expireSec, path, fileName); |
| | | } |
| | | |
| | | @Override |
| | | public MediaInfo getMediaInfo(MediaServer mediaServer, String app, String stream) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[getMediaInfo] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return null; |
| | | } |
| | | return mediaNodeServerService.getMediaInfo(mediaServer, app, stream); |
| | | } |
| | | |
| | | @Override |
| | | public Boolean pauseRtpCheck(MediaServer mediaServer, String streamKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[pauseRtpCheck] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.pauseRtpCheck(mediaServer, streamKey); |
| | | } |
| | | |
| | | @Override |
| | | public boolean resumeRtpCheck(MediaServer mediaServer, String streamKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[pauseRtpCheck] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.resumeRtpCheck(mediaServer, streamKey); |
| | | } |
| | | |
| | | @Override |
| | | public String getFfmpegCmd(MediaServer mediaServer, String cmdKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[getFfmpegCmd] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return null; |
| | | } |
| | | return mediaNodeServerService.getFfmpegCmd(mediaServer, cmdKey); |
| | | } |
| | | |
| | | @Override |
| | | public void closeStreams(MediaServer mediaServer, String app, String stream) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[closeStreams] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return; |
| | | } |
| | | mediaNodeServerService.closeStreams(mediaServer, app, stream); |
| | | } |
| | | |
| | | @Override |
| | | public WVPResult<String> addFFmpegSource(MediaServer mediaServer, String srcUrl, String dstUrl, int timeoutMs, boolean enableAudio, boolean enableMp4, String ffmpegCmdKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[addFFmpegSource] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return WVPResult.fail(ErrorCode.ERROR400); |
| | | } |
| | | return mediaNodeServerService.addFFmpegSource(mediaServer, srcUrl, dstUrl, timeoutMs, enableAudio, enableMp4, ffmpegCmdKey); |
| | | } |
| | | |
| | | @Override |
| | | public WVPResult<String> addStreamProxy(MediaServer mediaServer, String app, String stream, String url, boolean enableAudio, boolean enableMp4, String rtpType) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[addStreamProxy] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return WVPResult.fail(ErrorCode.ERROR400); |
| | | } |
| | | return mediaNodeServerService.addStreamProxy(mediaServer, app, stream, url, enableAudio, enableMp4, rtpType); |
| | | } |
| | | |
| | | @Override |
| | | public Boolean delFFmpegSource(MediaServer mediaServer, String streamKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[delFFmpegSource] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.delFFmpegSource(mediaServer, streamKey); |
| | | } |
| | | |
| | | @Override |
| | | public Boolean delStreamProxy(MediaServer mediaServerItem, String streamKey) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServerItem.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[delStreamProxy] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServerItem.getType()); |
| | | return false; |
| | | } |
| | | return mediaNodeServerService.delStreamProxy(mediaServerItem, streamKey); |
| | | } |
| | | |
| | | @Override |
| | | public Map<String, String> getFFmpegCMDs(MediaServer mediaServer) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[getFFmpegCMDs] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return new HashMap<>(); |
| | | } |
| | | return mediaNodeServerService.getFFmpegCMDs(mediaServer); |
| | | } |
| | | |
| | | @Override |
| | | public StreamInfo getStreamInfoByAppAndStream(MediaServer mediaServerItem, String app, String stream, MediaInfo mediaInfo, String callId) { |
| | | return getStreamInfoByAppAndStream(mediaServerItem, app, stream, mediaInfo, null, callId, true); |
| | | } |
| | | |
| | | @Override |
| | | public StreamInfo getStreamInfoByAppAndStreamWithCheck(String app, String stream, String mediaServerId, String addr, boolean authority) { |
| | | StreamInfo streamInfo = null; |
| | | if (mediaServerId == null) { |
| | | mediaServerId = mediaConfig.getId(); |
| | | } |
| | | MediaServer mediaInfo = getOne(mediaServerId); |
| | | if (mediaInfo == null) { |
| | | return null; |
| | | } |
| | | String calld = null; |
| | | StreamAuthorityInfo streamAuthorityInfo = redisCatchStorage.getStreamAuthorityInfo(app, stream); |
| | | if (streamAuthorityInfo != null) { |
| | | calld = streamAuthorityInfo.getCallId(); |
| | | } |
| | | List<StreamInfo> streamInfoList = getMediaList(mediaInfo, app, stream, calld); |
| | | if (streamInfoList.isEmpty()) { |
| | | return null; |
| | | }else { |
| | | return streamInfoList.get(0); |
| | | } |
| | | } |
| | | |
| | | |
| | | |
| | | @Override |
| | | public StreamInfo getStreamInfoByAppAndStreamWithCheck(String app, String stream, String mediaServerId, boolean authority) { |
| | | return getStreamInfoByAppAndStreamWithCheck(app, stream, mediaServerId, null, authority); |
| | | } |
| | | |
| | | @Override |
| | | public StreamInfo getStreamInfoByAppAndStream(MediaServer mediaServer, String app, String stream, MediaInfo mediaInfo, String addr, String callId, boolean isPlay) { |
| | | StreamInfo streamInfoResult = new StreamInfo(); |
| | | streamInfoResult.setStream(stream); |
| | | streamInfoResult.setApp(app); |
| | | if (addr == null) { |
| | | addr = mediaServer.getStreamIp(); |
| | | } |
| | | |
| | | streamInfoResult.setIp(addr); |
| | | streamInfoResult.setMediaServerId(mediaServer.getId()); |
| | | String callIdParam = ObjectUtils.isEmpty(callId)?"":"?callId=" + callId; |
| | | streamInfoResult.setRtmp(addr, mediaServer.getRtmpPort(),mediaServer.getRtmpSSlPort(), app, stream, callIdParam); |
| | | streamInfoResult.setRtsp(addr, mediaServer.getRtspPort(),mediaServer.getRtspSSLPort(), app, stream, callIdParam); |
| | | |
| | | |
| | | if ("abl".equals(mediaServer.getType())) { |
| | | String flvFile = String.format("%s/%s.flv%s", app, stream, callIdParam); |
| | | streamInfoResult.setFlv(addr, mediaServer.getFlvPort(),mediaServer.getFlvSSLPort(), flvFile); |
| | | streamInfoResult.setWsFlv(addr, mediaServer.getWsFlvPort(),mediaServer.getWsFlvSSLPort(), flvFile); |
| | | }else { |
| | | String flvFile = String.format("%s/%s.live.flv%s", app, stream, callIdParam); |
| | | streamInfoResult.setFlv(addr, mediaServer.getFlvPort(),mediaServer.getFlvSSLPort(), flvFile); |
| | | streamInfoResult.setWsFlv(addr, mediaServer.getWsFlvPort(),mediaServer.getWsFlvSSLPort(), flvFile); |
| | | } |
| | | |
| | | streamInfoResult.setFmp4(addr, mediaServer.getHttpPort(),mediaServer.getHttpSSlPort(), app, stream, callIdParam); |
| | | streamInfoResult.setHls(addr, mediaServer.getHttpPort(),mediaServer.getHttpSSlPort(), app, stream, callIdParam); |
| | | streamInfoResult.setTs(addr, mediaServer.getHttpPort(),mediaServer.getHttpSSlPort(), app, stream, callIdParam); |
| | | streamInfoResult.setRtc(addr, mediaServer.getHttpPort(),mediaServer.getHttpSSlPort(), app, stream, callIdParam, isPlay); |
| | | |
| | | streamInfoResult.setMediaInfo(mediaInfo); |
| | | return streamInfoResult; |
| | | } |
| | | |
| | | @Override |
| | | public Boolean isStreamReady(MediaServer mediaServer, String rtp, String streamId) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[isStreamReady] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | return false; |
| | | } |
| | | MediaInfo mediaInfo = mediaNodeServerService.getMediaInfo(mediaServer, rtp, streamId); |
| | | return mediaInfo != null; |
| | | } |
| | | |
| | | @Override |
| | | public void startSendRtpPassive(MediaServer mediaServer, SendRtpItem sendRtpItem, Integer timeout) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[startSendRtpPassive] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到mediaServer对应的实现类"); |
| | | } |
| | | mediaNodeServerService.startSendRtpPassive(mediaServer, sendRtpItem, timeout); |
| | | } |
| | | |
| | | @Override |
| | | public void startSendRtp(MediaServer mediaServer, SendRtpItem sendRtpItem) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[startSendRtpStream] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到mediaServer对应的实现类"); |
| | | } |
| | | logger.info("[开始推流] rtp/{}, 目标={}:{},SSRC={}, RTCP={}", sendRtpItem.getStream(), |
| | | sendRtpItem.getIp(), sendRtpItem.getPort(), sendRtpItem.getSsrc(), sendRtpItem.isRtcp()); |
| | | mediaNodeServerService.startSendRtpStream(mediaServer, sendRtpItem); |
| | | } |
| | | |
| | | @Override |
| | | public SendRtpItem createSendRtpItem(MediaServer mediaServer, String ip, int port, String ssrc, String requesterId, String deviceId, String channelId, boolean isTcp, boolean rtcp) { |
| | | int localPort = sendRtpPortManager.getNextPort(mediaServer); |
| | | if (localPort == 0) { |
| | | return null; |
| | | } |
| | | SendRtpItem sendRtpItem = new SendRtpItem(); |
| | | sendRtpItem.setIp(ip); |
| | | sendRtpItem.setPort(port); |
| | | sendRtpItem.setSsrc(ssrc); |
| | | sendRtpItem.setPlatformId(deviceId); |
| | | sendRtpItem.setDeviceId(deviceId); |
| | | sendRtpItem.setChannelId(channelId); |
| | | sendRtpItem.setTcp(isTcp); |
| | | sendRtpItem.setRtcp(rtcp); |
| | | sendRtpItem.setApp("rtp"); |
| | | sendRtpItem.setLocalPort(localPort); |
| | | sendRtpItem.setServerId(userSetting.getServerId()); |
| | | sendRtpItem.setMediaServerId(mediaServer.getId()); |
| | | return sendRtpItem; |
| | | } |
| | | |
| | | @Override |
| | | public SendRtpItem createSendRtpItem(MediaServer serverItem, String ip, int port, String ssrc, String platformId, |
| | | String app, String stream, String channelId, boolean tcp, boolean rtcp){ |
| | | |
| | | int localPort = sendRtpPortManager.getNextPort(serverItem); |
| | | if (localPort == 0) { |
| | | return null; |
| | | } |
| | | SendRtpItem sendRtpItem = new SendRtpItem(); |
| | | sendRtpItem.setIp(ip); |
| | | sendRtpItem.setPort(port); |
| | | sendRtpItem.setSsrc(ssrc); |
| | | sendRtpItem.setApp(app); |
| | | sendRtpItem.setStream(stream); |
| | | sendRtpItem.setPlatformId(platformId); |
| | | sendRtpItem.setChannelId(channelId); |
| | | sendRtpItem.setTcp(tcp); |
| | | sendRtpItem.setLocalPort(localPort); |
| | | sendRtpItem.setServerId(userSetting.getServerId()); |
| | | sendRtpItem.setMediaServerId(serverItem.getId()); |
| | | sendRtpItem.setRtcp(rtcp); |
| | | return sendRtpItem; |
| | | } |
| | | |
| | | @Override |
| | | public MediaServer getMediaServerByAppAndStream(String app, String stream) { |
| | | List<MediaServer> mediaServerList = getAll(); |
| | | for (MediaServer mediaServer : mediaServerList) { |
| | | MediaInfo mediaInfo = getMediaInfo(mediaServer, app, stream); |
| | | if (mediaInfo != null) { |
| | | return mediaServer; |
| | | } |
| | | } |
| | | return null; |
| | | } |
| | | |
| | | @Override |
| | | public Long updateDownloadProcess(MediaServer mediaServer, String app, String stream) { |
| | | IMediaNodeServerService mediaNodeServerService = nodeServerServiceMap.get(mediaServer.getType()); |
| | | if (mediaNodeServerService == null) { |
| | | logger.info("[updateDownloadProcess] 失败, mediaServer的类型: {},未找到对应的实现类", mediaServer.getType()); |
| | | throw new ControllerException(ErrorCode.ERROR100.getCode(), "未找到mediaServer对应的实现类"); |
| | | } |
| | | return mediaNodeServerService.updateDownloadProcess(mediaServer, app, stream); |
| | | } |
| | | } |