old mode 100644
new mode 100755
|  |  |  | 
|---|
|  |  |  | package com.genersoft.iot.vmp.service.impl; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.alibaba.fastjson.JSONObject; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.MediaServerConfig; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.ZLMRESTfulUtils; | 
|---|
|  |  |  | import com.alibaba.fastjson2.JSONObject; | 
|---|
|  |  |  | import com.baomidou.dynamic.datasource.annotation.DS; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.GeneralCallback; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.StreamInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.DynamicTask; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.UserSetting; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.exception.ControllerException; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.event.subscribe.catalog.CatalogEvent; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.bean.MediaInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.bean.MediaServer; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.hook.Hook; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.hook.HookSubscribe; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.event.hook.HookType; | 
|---|
|  |  |  | 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.media.MediaNotFoundEvent; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.service.IMediaServerService; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.media.zlm.dto.StreamProxyItem; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.IGbStreamService; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.IStreamProxyService; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorager; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorage; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.dao.GbStreamMapper; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.dao.PlatformGbStreamMapper; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.dao.StreamProxyMapper; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.service.IStreamProxyService; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.utils.DateUtil; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.ErrorCode; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.ResourceBaseInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.vmanager.bean.WVPResult; | 
|---|
|  |  |  | import com.github.pagehelper.PageInfo; | 
|---|
|  |  |  | import org.slf4j.Logger; | 
|---|
|  |  |  | import org.slf4j.LoggerFactory; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
|---|
|  |  |  | import org.springframework.context.event.EventListener; | 
|---|
|  |  |  | import org.springframework.jdbc.datasource.DataSourceTransactionManager; | 
|---|
|  |  |  | import org.springframework.scheduling.annotation.Async; | 
|---|
|  |  |  | import org.springframework.scheduling.annotation.Scheduled; | 
|---|
|  |  |  | import org.springframework.stereotype.Service; | 
|---|
|  |  |  | import org.springframework.transaction.TransactionDefinition; | 
|---|
|  |  |  | import org.springframework.transaction.TransactionStatus; | 
|---|
|  |  |  | import org.springframework.util.CollectionUtils; | 
|---|
|  |  |  | import org.springframework.util.ObjectUtils; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import java.util.HashMap; | 
|---|
|  |  |  | import java.util.List; | 
|---|
|  |  |  | import java.util.Map; | 
|---|
|  |  |  | import java.util.UUID; | 
|---|
|  |  |  | import java.util.function.Function; | 
|---|
|  |  |  | import java.util.stream.Collectors; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 视频代理业务 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Service | 
|---|
|  |  |  | @DS("master") | 
|---|
|  |  |  | public class StreamProxyServiceImpl implements IStreamProxyService { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private final static Logger logger = LoggerFactory.getLogger(StreamProxyServiceImpl.class); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IVideoManagerStorager videoManagerStorager; | 
|---|
|  |  |  | private IVideoManagerStorage videoManagerStorager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private StreamProxyMapper streamProxyMapper; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IRedisCatchStorage redisCatchStorage; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ZLMRESTfulUtils zlmresTfulUtils; | 
|---|
|  |  |  | private IVideoManagerStorage storager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private StreamProxyMapper streamProxyMapper; | 
|---|
|  |  |  | private UserSetting userSetting; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private GbStreamMapper gbStreamMapper; | 
|---|
|  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private PlatformGbStreamMapper platformGbStreamMapper; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IGbStreamService gbStreamService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public String save(StreamProxyItem param) { | 
|---|
|  |  |  | MediaServerConfig mediaInfo = redisCatchStorage.getMediaInfo(); | 
|---|
|  |  |  | String dstUrl = String.format("rtmp://%s:%s/%s/%s", "127.0.0.1", mediaInfo.getRtmpPort(), param.getApp(), | 
|---|
|  |  |  | param.getStream() ); | 
|---|
|  |  |  | param.setDst_url(dstUrl); | 
|---|
|  |  |  | StringBuffer result = new StringBuffer(); | 
|---|
|  |  |  | // 更新 | 
|---|
|  |  |  | if (videoManagerStorager.queryStreamProxy(param.getApp(), param.getStream()) != null) { | 
|---|
|  |  |  | if (videoManagerStorager.updateStreamProxy(param)) { | 
|---|
|  |  |  | result.append("保存成功"); | 
|---|
|  |  |  | if (param.isEnable()){ | 
|---|
|  |  |  | JSONObject jsonObject = addStreamProxyToZlm(param); | 
|---|
|  |  |  | if (jsonObject == null) { | 
|---|
|  |  |  | result.append(", 但是启用失败,请检查流地址是否可用"); | 
|---|
|  |  |  | param.setEnable(false); | 
|---|
|  |  |  | videoManagerStorager.updateStreamProxy(param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { // 新增 | 
|---|
|  |  |  | if (videoManagerStorager.addStreamProxy(param)){ | 
|---|
|  |  |  | result.append("保存成功"); | 
|---|
|  |  |  | if (param.isEnable()) { | 
|---|
|  |  |  | JSONObject jsonObject = addStreamProxyToZlm(param); | 
|---|
|  |  |  | if (jsonObject == null) { | 
|---|
|  |  |  | result.append(", 但是启用失败,请检查流地址是否可用"); | 
|---|
|  |  |  | param.setEnable(false); | 
|---|
|  |  |  | videoManagerStorager.updateStreamProxy(param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IMediaServerService mediaServerService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private HookSubscribe hookSubscribe; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private DynamicTask dynamicTask; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | DataSourceTransactionManager dataSourceTransactionManager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | TransactionDefinition transactionDefinition; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流到来的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @org.springframework.context.event.EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaArrivalEvent event) { | 
|---|
|  |  |  | if ("rtsp".equals(event.getSchema())) { | 
|---|
|  |  |  | updateStatus(true, event.getApp(), event.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result.toString(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流离开的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaDepartureEvent event) { | 
|---|
|  |  |  | if ("rtsp".equals(event.getSchema())) { | 
|---|
|  |  |  | updateStatus(false, event.getApp(), event.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流离开的处理 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Async("taskExecutor") | 
|---|
|  |  |  | @EventListener | 
|---|
|  |  |  | public void onApplicationEvent(MediaNotFoundEvent event) { | 
|---|
|  |  |  | if ("rtp".equals(event.getApp())) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 拉流代理 | 
|---|
|  |  |  | StreamProxyItem streamProxyByAppAndStream = getStreamProxyByAppAndStream(event.getApp(), event.getStream()); | 
|---|
|  |  |  | if (streamProxyByAppAndStream != null && streamProxyByAppAndStream.isEnableDisableNoneReader()) { | 
|---|
|  |  |  | start(event.getApp(), event.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public JSONObject addStreamProxyToZlm(StreamProxyItem param) { | 
|---|
|  |  |  | JSONObject result = null; | 
|---|
|  |  |  | if ("default".equals(param.getType())){ | 
|---|
|  |  |  | result = zlmresTfulUtils.addStreamProxy(param.getApp(), param.getStream(), param.getUrl(), | 
|---|
|  |  |  | param.isEnable_hls(), param.isEnable_mp4(), param.getRtp_type()); | 
|---|
|  |  |  | }else if ("ffmpeg".equals(param.getType())) { | 
|---|
|  |  |  | result = zlmresTfulUtils.addFFmpegSource(param.getSrc_url(), param.getDst_url(), | 
|---|
|  |  |  | param.getTimeout_ms() + ""); | 
|---|
|  |  |  | public void save(StreamProxyItem param, GeneralCallback<StreamInfo> callback) { | 
|---|
|  |  |  | MediaServer mediaServer; | 
|---|
|  |  |  | if (ObjectUtils.isEmpty(param.getMediaServerId()) || "auto".equals(param.getMediaServerId())){ | 
|---|
|  |  |  | mediaServer = mediaServerService.getMediaServerForMinimumLoad(null); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | mediaServer = mediaServerService.getOne(param.getMediaServerId()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (mediaServer == null) { | 
|---|
|  |  |  | logger.warn("保存代理未找到在线的ZLM..."); | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "保存代理未找到在线的ZLM"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String dstUrl; | 
|---|
|  |  |  | if ("ffmpeg".equalsIgnoreCase(param.getType())) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | String ffmpegCmd = mediaServerService.getFfmpegCmd(mediaServer, param.getFfmpegCmdKey()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (ffmpegCmd == null) { | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "ffmpeg拉流代理无法获取ffmpeg cmd"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String schema = getSchemaFromFFmpegCmd(ffmpegCmd); | 
|---|
|  |  |  | if (schema == null) { | 
|---|
|  |  |  | throw new ControllerException(ErrorCode.ERROR100.getCode(), "ffmpeg拉流代理无法从ffmpeg cmd中获取到输出格式"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | int port; | 
|---|
|  |  |  | String schemaForUri; | 
|---|
|  |  |  | if (schema.equalsIgnoreCase("rtsp")) { | 
|---|
|  |  |  | port = mediaServer.getRtspPort(); | 
|---|
|  |  |  | schemaForUri = schema; | 
|---|
|  |  |  | }else if (schema.equalsIgnoreCase("flv")) { | 
|---|
|  |  |  | port = mediaServer.getRtmpPort(); | 
|---|
|  |  |  | schemaForUri = schema; | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | port = mediaServer.getRtmpPort(); | 
|---|
|  |  |  | schemaForUri = schema; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | dstUrl = String.format("%s://%s:%s/%s/%s", schemaForUri, "127.0.0.1", port, param.getApp(), | 
|---|
|  |  |  | param.getStream()); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | dstUrl = String.format("rtsp://%s:%s/%s/%s", "127.0.0.1", mediaServer.getRtspPort(), param.getApp(), | 
|---|
|  |  |  | param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | param.setDstUrl(dstUrl); | 
|---|
|  |  |  | logger.info("[拉流代理] 输出地址为:{}", dstUrl); | 
|---|
|  |  |  | param.setMediaServerId(mediaServer.getId()); | 
|---|
|  |  |  | boolean saveResult; | 
|---|
|  |  |  | // 更新 | 
|---|
|  |  |  | if (videoManagerStorager.queryStreamProxy(param.getApp(), param.getStream()) != null) { | 
|---|
|  |  |  | saveResult = updateStreamProxy(param); | 
|---|
|  |  |  | }else { // 新增 | 
|---|
|  |  |  | saveResult = addStreamProxy(param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (!saveResult) { | 
|---|
|  |  |  | callback.run(ErrorCode.ERROR100.getCode(), "保存失败", null); | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Hook hook = Hook.getInstance(HookType.on_media_arrival, param.getApp(), param.getStream(), mediaServer.getId()); | 
|---|
|  |  |  | hookSubscribe.addSubscribe(hook, (hookData) -> { | 
|---|
|  |  |  | StreamInfo streamInfo = mediaServerService.getStreamInfoByAppAndStream( | 
|---|
|  |  |  | mediaServer, param.getApp(), param.getStream(), null, null); | 
|---|
|  |  |  | callback.run(ErrorCode.SUCCESS.getCode(), ErrorCode.SUCCESS.getMsg(), streamInfo); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | if (param.isEnable()) { | 
|---|
|  |  |  | String talkKey = UUID.randomUUID().toString(); | 
|---|
|  |  |  | String delayTalkKey = UUID.randomUUID().toString(); | 
|---|
|  |  |  | dynamicTask.startDelay(delayTalkKey, ()->{ | 
|---|
|  |  |  | StreamInfo streamInfo = mediaServerService.getStreamInfoByAppAndStreamWithCheck(param.getApp(), param.getStream(), mediaServer.getId(), false); | 
|---|
|  |  |  | if (streamInfo != null) { | 
|---|
|  |  |  | callback.run(ErrorCode.SUCCESS.getCode(), ErrorCode.SUCCESS.getMsg(), streamInfo); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | dynamicTask.stop(talkKey); | 
|---|
|  |  |  | callback.run(ErrorCode.ERROR100.getCode(), "超时", null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }, 7000); | 
|---|
|  |  |  | WVPResult<String> result = addStreamProxyToZlm(param); | 
|---|
|  |  |  | if (result != null && result.getCode() == 0) { | 
|---|
|  |  |  | hookSubscribe.removeSubscribe(hook); | 
|---|
|  |  |  | dynamicTask.stop(talkKey); | 
|---|
|  |  |  | StreamInfo streamInfo = mediaServerService.getStreamInfoByAppAndStream( | 
|---|
|  |  |  | mediaServer, param.getApp(), param.getStream(), null, null); | 
|---|
|  |  |  | callback.run(ErrorCode.SUCCESS.getCode(), ErrorCode.SUCCESS.getMsg(), streamInfo); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | param.setEnable(false); | 
|---|
|  |  |  | // 直接移除 | 
|---|
|  |  |  | if (param.isEnableRemoveNoneReader()) { | 
|---|
|  |  |  | del(param.getApp(), param.getStream()); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | updateStreamProxy(param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (result == null){ | 
|---|
|  |  |  | callback.run(ErrorCode.ERROR100.getCode(), "记录已保存,启用失败", null); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | callback.run(ErrorCode.ERROR100.getCode(), result.getMsg(), null); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else{ | 
|---|
|  |  |  | StreamInfo streamInfo = mediaServerService.getStreamInfoByAppAndStream( | 
|---|
|  |  |  | mediaServer, param.getApp(), param.getStream(), null, null); | 
|---|
|  |  |  | callback.run(ErrorCode.SUCCESS.getCode(), ErrorCode.SUCCESS.getMsg(), streamInfo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private String getSchemaFromFFmpegCmd(String ffmpegCmd) { | 
|---|
|  |  |  | ffmpegCmd = ffmpegCmd.replaceAll(" + ", " "); | 
|---|
|  |  |  | String[] paramArray = ffmpegCmd.split(" "); | 
|---|
|  |  |  | if (paramArray.length == 0) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | for (int i = 0; i < paramArray.length; i++) { | 
|---|
|  |  |  | if (paramArray[i].equalsIgnoreCase("-f")) { | 
|---|
|  |  |  | if (i + 1 < paramArray.length - 1) { | 
|---|
|  |  |  | return paramArray[i+1]; | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 新增代理流 | 
|---|
|  |  |  | * @param streamProxyItem | 
|---|
|  |  |  | * @return | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | private boolean addStreamProxy(StreamProxyItem streamProxyItem) { | 
|---|
|  |  |  | TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition); | 
|---|
|  |  |  | boolean result = false; | 
|---|
|  |  |  | streamProxyItem.setStreamType("proxy"); | 
|---|
|  |  |  | streamProxyItem.setStatus(true); | 
|---|
|  |  |  | String now = DateUtil.getNow(); | 
|---|
|  |  |  | streamProxyItem.setCreateTime(now); | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | if (streamProxyMapper.add(streamProxyItem) > 0) { | 
|---|
|  |  |  | if (!ObjectUtils.isEmpty(streamProxyItem.getGbId())) { | 
|---|
|  |  |  | if (gbStreamMapper.add(streamProxyItem) < 0) { | 
|---|
|  |  |  | //事务回滚 | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | //事务回滚 | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | result = true; | 
|---|
|  |  |  | dataSourceTransactionManager.commit(transactionStatus);     //手动提交 | 
|---|
|  |  |  | }catch (Exception e) { | 
|---|
|  |  |  | logger.error("向数据库添加流代理失败:", e); | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 更新代理流 | 
|---|
|  |  |  | * @param streamProxyItem | 
|---|
|  |  |  | * @return | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public boolean updateStreamProxy(StreamProxyItem streamProxyItem) { | 
|---|
|  |  |  | TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition); | 
|---|
|  |  |  | boolean result = false; | 
|---|
|  |  |  | streamProxyItem.setStreamType("proxy"); | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | if (streamProxyMapper.update(streamProxyItem) > 0) { | 
|---|
|  |  |  | if (!ObjectUtils.isEmpty(streamProxyItem.getGbId())) { | 
|---|
|  |  |  | if (gbStreamMapper.updateByAppAndStream(streamProxyItem) == 0) { | 
|---|
|  |  |  | //事务回滚 | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | //事务回滚 | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | dataSourceTransactionManager.commit(transactionStatus);     //手动提交 | 
|---|
|  |  |  | result = true; | 
|---|
|  |  |  | }catch (Exception e) { | 
|---|
|  |  |  | logger.error("未处理的异常 ", e); | 
|---|
|  |  |  | dataSourceTransactionManager.rollback(transactionStatus); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public JSONObject removeStreamProxyFromZlm(StreamProxyItem param) { | 
|---|
|  |  |  | JSONObject result = zlmresTfulUtils.closeStreams(param.getApp(), param.getStream()); | 
|---|
|  |  |  | public WVPResult<String> addStreamProxyToZlm(StreamProxyItem param) { | 
|---|
|  |  |  | WVPResult<String> result = null; | 
|---|
|  |  |  | MediaServer mediaServer = null; | 
|---|
|  |  |  | if (param.getMediaServerId() == null) { | 
|---|
|  |  |  | logger.warn("添加代理时MediaServerId 为null"); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | mediaServer = mediaServerService.getOne(param.getMediaServerId()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (mediaServer == null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (mediaServerService.isStreamReady(mediaServer, param.getApp(), param.getStream())) { | 
|---|
|  |  |  | mediaServerService.closeStreams(mediaServer, param.getApp(), param.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String msgResult; | 
|---|
|  |  |  | if ("ffmpeg".equalsIgnoreCase(param.getType())){ | 
|---|
|  |  |  | result = mediaServerService.addFFmpegSource(mediaServer, param.getSrcUrl().trim(), param.getDstUrl(), | 
|---|
|  |  |  | param.getTimeoutMs(), param.isEnableAudio(), param.isEnableMp4(), | 
|---|
|  |  |  | param.getFfmpegCmdKey()); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | result = mediaServerService.addStreamProxy(mediaServer, param.getApp(), param.getStream(), param.getUrl().trim(), | 
|---|
|  |  |  | param.isEnableAudio(), param.isEnableMp4(), param.getRtpType()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (result != null && result.getCode() == 0) { | 
|---|
|  |  |  | String key = result.getData(); | 
|---|
|  |  |  | if (key == null) { | 
|---|
|  |  |  | logger.warn("[获取拉流代理的结果数据Data] 失败: {}", result ); | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | param.setStreamKey(key); | 
|---|
|  |  |  | streamProxyMapper.update(param); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public Boolean removeStreamProxyFromZlm(StreamProxyItem param) { | 
|---|
|  |  |  | if (param ==null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); | 
|---|
|  |  |  | if (mediaServer == null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | List<StreamInfo> mediaList = mediaServerService.getMediaList(mediaServer, param.getApp(), param.getStream(), null); | 
|---|
|  |  |  | if (mediaList == null || mediaList.isEmpty()) { | 
|---|
|  |  |  | return true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Boolean result = false; | 
|---|
|  |  |  | if ("ffmpeg".equalsIgnoreCase(param.getType())){ | 
|---|
|  |  |  | result = mediaServerService.delFFmpegSource(mediaServer, param.getStreamKey()); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | result = mediaServerService.delStreamProxy(mediaServer, param.getStreamKey()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void del(String app, String stream) { | 
|---|
|  |  |  | StreamProxyItem streamProxyItem = new StreamProxyItem(); | 
|---|
|  |  |  | streamProxyItem.setApp(app); | 
|---|
|  |  |  | streamProxyItem.setStream(stream); | 
|---|
|  |  |  | JSONObject jsonObject = removeStreamProxyFromZlm(streamProxyItem); | 
|---|
|  |  |  | if (jsonObject.getInteger("code") == 0) { | 
|---|
|  |  |  | videoManagerStorager.deleteStreamProxy(app, stream); | 
|---|
|  |  |  | StreamProxyItem streamProxyItem = videoManagerStorager.queryStreamProxy(app, stream); | 
|---|
|  |  |  | if (streamProxyItem != null) { | 
|---|
|  |  |  | gbStreamService.sendCatalogMsg(streamProxyItem, CatalogEvent.DEL); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 如果关联了国标那么移除关联 | 
|---|
|  |  |  | gbStreamMapper.del(app, stream); | 
|---|
|  |  |  | platformGbStreamMapper.delByAppAndStream(app, stream); | 
|---|
|  |  |  | // TODO 如果关联的推流, 那么状态设置为离线 | 
|---|
|  |  |  | gbStreamMapper.del(app, stream); | 
|---|
|  |  |  | videoManagerStorager.deleteStreamProxy(app, stream); | 
|---|
|  |  |  | redisCatchStorage.removeStream(streamProxyItem.getMediaServerId(), "PULL", app, stream); | 
|---|
|  |  |  | Boolean result = removeStreamProxyFromZlm(streamProxyItem); | 
|---|
|  |  |  | if (result != null && result) { | 
|---|
|  |  |  | logger.info("[移除代理]: 代理: {}/{}, 从zlm移除成功", app, stream); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | logger.info("[移除代理]: 代理: {}/{}, 从zlm移除失败", app, stream); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | public boolean start(String app, String stream) { | 
|---|
|  |  |  | boolean result = false; | 
|---|
|  |  |  | StreamProxyItem streamProxy = videoManagerStorager.queryStreamProxy(app, stream); | 
|---|
|  |  |  | if (!streamProxy.isEnable() &&  streamProxy != null) { | 
|---|
|  |  |  | JSONObject jsonObject = addStreamProxyToZlm(streamProxy); | 
|---|
|  |  |  | if (jsonObject == null) return false; | 
|---|
|  |  |  | if (jsonObject.getInteger("code") == 0) { | 
|---|
|  |  |  | if (streamProxy != null && !streamProxy.isEnable() ) { | 
|---|
|  |  |  | WVPResult<String> wvpResult = addStreamProxyToZlm(streamProxy); | 
|---|
|  |  |  | if (wvpResult == null) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (wvpResult.getCode() == 0) { | 
|---|
|  |  |  | result = true; | 
|---|
|  |  |  | streamProxy.setEnable(true); | 
|---|
|  |  |  | videoManagerStorager.updateStreamProxy(streamProxy); | 
|---|
|  |  |  | updateStreamProxy(streamProxy); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | logger.info("启用代理失败: {}/{}->{}({})", app, stream, wvpResult.getMsg(), | 
|---|
|  |  |  | streamProxy.getSrcUrl() == null? streamProxy.getUrl():streamProxy.getSrcUrl()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } else if (streamProxy != null && streamProxy.isEnable()) { | 
|---|
|  |  |  | return true ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | boolean result = false; | 
|---|
|  |  |  | StreamProxyItem streamProxyDto = videoManagerStorager.queryStreamProxy(app, stream); | 
|---|
|  |  |  | if (streamProxyDto != null && streamProxyDto.isEnable()) { | 
|---|
|  |  |  | JSONObject jsonObject = removeStreamProxyFromZlm(streamProxyDto); | 
|---|
|  |  |  | if (jsonObject.getInteger("code") == 0) { | 
|---|
|  |  |  | Boolean removed = removeStreamProxyFromZlm(streamProxyDto); | 
|---|
|  |  |  | if (removed != null && removed) { | 
|---|
|  |  |  | streamProxyDto.setEnable(false); | 
|---|
|  |  |  | result = videoManagerStorager.updateStreamProxy(streamProxyDto); | 
|---|
|  |  |  | result = updateStreamProxy(streamProxyDto); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return result; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public Map<String, String> getFFmpegCMDs(MediaServer mediaServer) { | 
|---|
|  |  |  | return mediaServerService.getFFmpegCMDs(mediaServer); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public StreamProxyItem getStreamProxyByAppAndStream(String app, String streamId) { | 
|---|
|  |  |  | return videoManagerStorager.getStreamProxyByAppAndStream(app, streamId); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void zlmServerOnline(String mediaServerId) { | 
|---|
|  |  |  | // 移除开启了无人观看自动移除的流 | 
|---|
|  |  |  | List<StreamProxyItem> streamProxyItemList = streamProxyMapper.selectAutoRemoveItemByMediaServerId(mediaServerId); | 
|---|
|  |  |  | if (streamProxyItemList.size() > 0) { | 
|---|
|  |  |  | gbStreamMapper.batchDel(streamProxyItemList); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | streamProxyMapper.deleteAutoRemoveItemByMediaServerId(mediaServerId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 移除拉流代理生成的流信息 | 
|---|
|  |  |  | syncPullStream(mediaServerId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 恢复流代理, 只查找这个这个流媒体 | 
|---|
|  |  |  | List<StreamProxyItem> streamProxyListForEnable = storager.getStreamProxyListForEnableInMediaServer( | 
|---|
|  |  |  | mediaServerId, true); | 
|---|
|  |  |  | for (StreamProxyItem streamProxyDto : streamProxyListForEnable) { | 
|---|
|  |  |  | logger.info("恢复流代理," + streamProxyDto.getApp() + "/" + streamProxyDto.getStream()); | 
|---|
|  |  |  | WVPResult<String> wvpResult = addStreamProxyToZlm(streamProxyDto); | 
|---|
|  |  |  | if (wvpResult == null) { | 
|---|
|  |  |  | // 设置为离线 | 
|---|
|  |  |  | logger.info("恢复流代理失败" + streamProxyDto.getApp() + "/" + streamProxyDto.getStream()); | 
|---|
|  |  |  | updateStatus(false, streamProxyDto.getApp(), streamProxyDto.getStream()); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | updateStatus(true, streamProxyDto.getApp(), streamProxyDto.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void zlmServerOffline(String mediaServerId) { | 
|---|
|  |  |  | // 移除开启了无人观看自动移除的流 | 
|---|
|  |  |  | List<StreamProxyItem> streamProxyItemList = streamProxyMapper.selectAutoRemoveItemByMediaServerId(mediaServerId); | 
|---|
|  |  |  | if (streamProxyItemList.size() > 0) { | 
|---|
|  |  |  | gbStreamMapper.batchDel(streamProxyItemList); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | streamProxyMapper.deleteAutoRemoveItemByMediaServerId(mediaServerId); | 
|---|
|  |  |  | // 其他的流设置离线 | 
|---|
|  |  |  | streamProxyMapper.updateStatusByMediaServerId(mediaServerId, false); | 
|---|
|  |  |  | String type = "PULL"; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 发送redis消息 | 
|---|
|  |  |  | List<MediaInfo> mediaInfoList = redisCatchStorage.getStreams(mediaServerId, type); | 
|---|
|  |  |  | if (mediaInfoList.size() > 0) { | 
|---|
|  |  |  | for (MediaInfo mediaInfo : mediaInfoList) { | 
|---|
|  |  |  | JSONObject jsonObject = new JSONObject(); | 
|---|
|  |  |  | jsonObject.put("serverId", userSetting.getServerId()); | 
|---|
|  |  |  | jsonObject.put("app", mediaInfo.getApp()); | 
|---|
|  |  |  | jsonObject.put("stream", mediaInfo.getStream()); | 
|---|
|  |  |  | jsonObject.put("register", false); | 
|---|
|  |  |  | jsonObject.put("mediaServerId", mediaServerId); | 
|---|
|  |  |  | redisCatchStorage.sendStreamChangeMsg(type, jsonObject); | 
|---|
|  |  |  | // 移除redis内流的信息 | 
|---|
|  |  |  | redisCatchStorage.removeStream(mediaServerId, type, mediaInfo.getApp(), mediaInfo.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void clean() { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public int updateStatus(boolean status, String app, String stream) { | 
|---|
|  |  |  | return streamProxyMapper.updateStatus(app, stream, status); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void syncPullStream(String mediaServerId){ | 
|---|
|  |  |  | MediaServer mediaServer = mediaServerService.getOne(mediaServerId); | 
|---|
|  |  |  | if (mediaServer != null) { | 
|---|
|  |  |  | List<MediaInfo> mediaInfoList = redisCatchStorage.getStreams(mediaServerId, "PULL"); | 
|---|
|  |  |  | if (!mediaInfoList.isEmpty()) { | 
|---|
|  |  |  | List<StreamInfo> mediaList = mediaServerService.getMediaList(mediaServer, null, null, null); | 
|---|
|  |  |  | Map<String, StreamInfo> stringStreamInfoMap = new HashMap<>(); | 
|---|
|  |  |  | if (mediaList != null && !mediaList.isEmpty()) { | 
|---|
|  |  |  | for (StreamInfo streamInfo : mediaList) { | 
|---|
|  |  |  | stringStreamInfoMap.put(streamInfo.getApp() + streamInfo.getStream(), streamInfo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (stringStreamInfoMap.isEmpty()) { | 
|---|
|  |  |  | redisCatchStorage.removeStream(mediaServerId, "PULL"); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | for (String key : stringStreamInfoMap.keySet()) { | 
|---|
|  |  |  | StreamInfo streamInfo = stringStreamInfoMap.get(key); | 
|---|
|  |  |  | if (stringStreamInfoMap.get(streamInfo.getApp() + streamInfo.getStream()) == null) { | 
|---|
|  |  |  | redisCatchStorage.removeStream(mediaServerId, "PULL", streamInfo.getApp(), | 
|---|
|  |  |  | streamInfo.getStream()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public ResourceBaseInfo getOverview() { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int total = streamProxyMapper.getAllCount(); | 
|---|
|  |  |  | int online = streamProxyMapper.getOnline(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return new ResourceBaseInfo(total, online); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Scheduled(cron = "* 0/10 * * * ?") | 
|---|
|  |  |  | public void asyncCheckStreamProxyStatus() { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<MediaServer> all = mediaServerService.getAllOnline(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (CollectionUtils.isEmpty(all)){ | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Map<String, MediaServer> serverItemMap = all.stream().collect(Collectors.toMap(MediaServer::getId, Function.identity(), (m1, m2) -> m1)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<StreamProxyItem> list = videoManagerStorager.getStreamProxyListForEnable(true); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (CollectionUtils.isEmpty(list)){ | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (StreamProxyItem streamProxyItem : list) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaServer mediaServerItem = serverItemMap.get(streamProxyItem.getMediaServerId()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MediaInfo mediaInfo = mediaServerService.getMediaInfo(mediaServerItem, streamProxyItem.getApp(), streamProxyItem.getStream()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (mediaInfo == null){ | 
|---|
|  |  |  | streamProxyItem.setStatus(false); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | if (mediaInfo.getOnline() != null && mediaInfo.getOnline()) { | 
|---|
|  |  |  | streamProxyItem.setStatus(true); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | streamProxyItem.setStatus(false); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | updateStreamProxy(streamProxyItem); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|