New file |
| | |
| | | package com.genersoft.iot.vmp.media.abl; |
| | | |
| | | import com.alibaba.fastjson2.JSONObject; |
| | | import com.genersoft.iot.vmp.conf.UserSetting; |
| | | import com.genersoft.iot.vmp.gb28181.event.EventPublisher; |
| | | import com.genersoft.iot.vmp.gb28181.session.AudioBroadcastManager; |
| | | import com.genersoft.iot.vmp.gb28181.session.SSRCFactory; |
| | | import com.genersoft.iot.vmp.gb28181.session.VideoStreamSessionManager; |
| | | import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder; |
| | | import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommanderForPlatform; |
| | | import com.genersoft.iot.vmp.media.abl.bean.hook.*; |
| | | import com.genersoft.iot.vmp.media.abl.event.HookAblServerKeepaliveEvent; |
| | | import com.genersoft.iot.vmp.media.abl.event.HookAblServerStartEvent; |
| | | import com.genersoft.iot.vmp.media.bean.MediaServer; |
| | | import com.genersoft.iot.vmp.media.bean.ResultForOnPublish; |
| | | import com.genersoft.iot.vmp.media.event.hook.HookSubscribe; |
| | | import com.genersoft.iot.vmp.media.event.media.*; |
| | | import com.genersoft.iot.vmp.media.event.mediaServer.MediaSendRtpStoppedEvent; |
| | | import com.genersoft.iot.vmp.media.service.IMediaServerService; |
| | | import com.genersoft.iot.vmp.media.zlm.ZLMMediaListManager; |
| | | import com.genersoft.iot.vmp.media.zlm.dto.hook.*; |
| | | import com.genersoft.iot.vmp.service.*; |
| | | import com.genersoft.iot.vmp.storager.IRedisCatchStorage; |
| | | import com.genersoft.iot.vmp.storager.IVideoManagerStorage; |
| | | import org.slf4j.Logger; |
| | | import org.slf4j.LoggerFactory; |
| | | import org.springframework.beans.factory.annotation.Autowired; |
| | | import org.springframework.beans.factory.annotation.Qualifier; |
| | | import org.springframework.context.ApplicationEventPublisher; |
| | | import org.springframework.data.redis.core.RedisTemplate; |
| | | import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; |
| | | import org.springframework.util.ObjectUtils; |
| | | import org.springframework.web.bind.annotation.*; |
| | | |
| | | import javax.servlet.http.HttpServletRequest; |
| | | import java.util.HashMap; |
| | | import java.util.Map; |
| | | |
| | | /** |
| | | * ABL 的hook事件监听 |
| | | */ |
| | | @RestController |
| | | @RequestMapping("/index/hook/abl") |
| | | public class ABLHttpHookListener { |
| | | |
| | | private final static Logger logger = LoggerFactory.getLogger(ABLHttpHookListener.class); |
| | | |
| | | @Autowired |
| | | private ABLRESTfulUtils ablresTfulUtils; |
| | | |
| | | @Autowired |
| | | private ISIPCommanderForPlatform commanderFroPlatform; |
| | | |
| | | @Autowired |
| | | private AudioBroadcastManager audioBroadcastManager; |
| | | |
| | | @Autowired |
| | | private IPlayService playService; |
| | | |
| | | @Autowired |
| | | private IVideoManagerStorage storager; |
| | | |
| | | @Autowired |
| | | private IRedisCatchStorage redisCatchStorage; |
| | | |
| | | @Autowired |
| | | private IInviteStreamService inviteStreamService; |
| | | |
| | | @Autowired |
| | | private IDeviceService deviceService; |
| | | |
| | | @Autowired |
| | | private IMediaServerService mediaServerService; |
| | | |
| | | @Autowired |
| | | private IStreamProxyService streamProxyService; |
| | | |
| | | @Autowired |
| | | private DeferredResultHolder resultHolder; |
| | | |
| | | @Autowired |
| | | private IMediaService mediaService; |
| | | |
| | | @Autowired |
| | | private EventPublisher eventPublisher; |
| | | |
| | | @Autowired |
| | | private ZLMMediaListManager zlmMediaListManager; |
| | | |
| | | @Autowired |
| | | private HookSubscribe subscribe; |
| | | |
| | | @Autowired |
| | | private UserSetting userSetting; |
| | | |
| | | @Autowired |
| | | private IUserService userService; |
| | | |
| | | @Autowired |
| | | private ICloudRecordService cloudRecordService; |
| | | |
| | | @Autowired |
| | | private VideoStreamSessionManager sessionManager; |
| | | |
| | | @Autowired |
| | | private SSRCFactory ssrcFactory; |
| | | |
| | | @Qualifier("taskExecutor") |
| | | @Autowired |
| | | private ThreadPoolTaskExecutor taskExecutor; |
| | | |
| | | @Autowired |
| | | private RedisTemplate<Object, Object> redisTemplate; |
| | | |
| | | @Autowired |
| | | private ApplicationEventPublisher applicationEventPublisher; |
| | | |
| | | /** |
| | | * 服务器定时上报时间,上报间隔可配置,默认10s上报一次 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_server_keepalive", produces = "application/json;charset=UTF-8") |
| | | public HookResult onServerKeepalive(@RequestBody OnServerKeepaliveABLHookParam param) { |
| | | try { |
| | | HookAblServerKeepaliveEvent event = new HookAblServerKeepaliveEvent(this); |
| | | MediaServer mediaServerItem = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServerItem != null) { |
| | | event.setMediaServerItem(mediaServerItem); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | }catch (Exception e) { |
| | | logger.info("[ZLM-HOOK-心跳] 发送通知失败 ", e); |
| | | } |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 播放器鉴权事件,rtsp/rtmp/http-flv/ws-flv/hls的播放都将触发此鉴权事件。 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_play", produces = "application/json;charset=UTF-8") |
| | | public HookResult onPlay(@RequestBody OnPlayABLHookParam param) { |
| | | |
| | | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServer == null) { |
| | | return new HookResultForOnPublish(0, "success"); |
| | | } |
| | | |
| | | Map<String, String> paramMap = urlParamToMap(param.getParams()); |
| | | // 对于播放流进行鉴权 |
| | | boolean authenticateResult = mediaService.authenticatePlay(param.getApp(), param.getStream(), paramMap.get("callId")); |
| | | if (!authenticateResult) { |
| | | logger.info("[ABL HOOK] 播放鉴权 失败:{}->{}", param.getMediaServerId(), param); |
| | | ablresTfulUtils.closeStreams(mediaServer, param.getApp(), param.getStream()); |
| | | |
| | | } |
| | | logger.info("[ABL HOOK] 播放鉴权成功:{}->{}", param.getMediaServerId(), param); |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * rtsp/rtmp/rtp推流鉴权事件。 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_publish", produces = "application/json;charset=UTF-8") |
| | | public HookResult onPublish(@RequestBody OnPublishABLHookParam param) { |
| | | |
| | | |
| | | logger.info("[ABL HOOK]推流鉴权:{}->{}", param.getMediaServerId(), param); |
| | | // TODO 加快处理速度 |
| | | |
| | | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServer == null) { |
| | | return new HookResultForOnPublish(0, "success"); |
| | | } |
| | | |
| | | ResultForOnPublish resultForOnPublish = mediaService.authenticatePublish(mediaServer, param.getApp(), param.getStream(), param.getParams()); |
| | | if (resultForOnPublish == null) { |
| | | logger.info("[ABL HOOK]推流鉴权 拒绝 响应:{}->{}", param.getMediaServerId(), param); |
| | | ablresTfulUtils.closeStreams(mediaServer, param.getApp(), param.getStream()); |
| | | } |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 如果某一个码流进行MP4录像(enable_mp4=1),会触发录像进度通知事件 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_record_progress", produces = "application/json;charset=UTF-8") |
| | | public HookResult onRecordProgress(@RequestBody OnRecordProgressABLHookParam param) { |
| | | |
| | | |
| | | logger.info("[ABL HOOK] 录像进度通知:{}->{}/{}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream(), param.getCurrentFileDuration(), param.getTotalVideoDuration()); |
| | | |
| | | // TODO 这里用来做录像进度 |
| | | // MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | // if (mediaServer == null) { |
| | | // return new HookResultForOnPublish(0, "success"); |
| | | // } |
| | | // |
| | | // ResultForOnPublish resultForOnPublish = mediaService.authenticatePublish(mediaServer, param.getApp(), param.getStream(), param.getParams()); |
| | | // if (resultForOnPublish == null) { |
| | | // logger.info("[ABL HOOK]推流鉴权 拒绝 响应:{}->{}", param.getMediaServerId(), param); |
| | | // ablresTfulUtils.closeStreams(mediaServer, param.getApp(), param.getStream()); |
| | | // } |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 当代理拉流、国标接入等等 码流不到达时会发出 码流不到达的事件通知 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_stream_not_arrive", produces = "application/json;charset=UTF-8") |
| | | public HookResult onStreamNotArrive(@RequestBody ABLHookParam param) { |
| | | |
| | | |
| | | logger.info("[ABL HOOK] 码流不到达通知:{}->{}/{}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream(), param.getCurrentFileDuration(), param.getTotalVideoDuration()); |
| | | try { |
| | | if ("rtp".equals(param.getApp())) { |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | MediaRtpServerTimeoutEvent event = new MediaRtpServerTimeoutEvent(this); |
| | | MediaServer mediaServerItem = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServerItem != null) { |
| | | event.setMediaServer(mediaServerItem); |
| | | event.setApp("rtp"); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | }catch (Exception e) { |
| | | logger.info("[ABL-HOOK-码流不到达通知] 发送通知失败 ", e); |
| | | } |
| | | |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 如果某一个码流进行MP4录像(enable_mp4=1),当某个MP4文件被删除会触发该事件通知 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_delete_record_mp4", produces = "application/json;charset=UTF-8") |
| | | public HookResult onDeleteRecordMp4(@RequestBody OnRecordMp4ABLHookParam param) { |
| | | |
| | | |
| | | logger.info("[ABL HOOK] MP4文件被删除通知:{}->{}/{}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream(), param.getCurrentFileDuration(), param.getTotalVideoDuration()); |
| | | |
| | | |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | |
| | | /** |
| | | * rtsp/rtmp流注册或注销时触发此事件;此事件对回复不敏感。 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_stream_arrive", produces = "application/json;charset=UTF-8") |
| | | public HookResult onStreamArrive(@RequestBody OnStreamArriveABLHookParam param) { |
| | | |
| | | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServer == null) { |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | logger.info("[ABL HOOK] 码流到达, {}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream()); |
| | | MediaArrivalEvent mediaArrivalEvent = MediaArrivalEvent.getInstance(this, param, mediaServer); |
| | | applicationEventPublisher.publishEvent(mediaArrivalEvent); |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 流无人观看时事件,用户可以通过此事件选择是否关闭无人看的流。 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_stream_none_reader", produces = "application/json;charset=UTF-8") |
| | | public JSONObject onStreamNoneReader(@RequestBody ABLHookParam param) { |
| | | |
| | | logger.info("[ZLM HOOK]流无人观看:{}->{}/{}", param.getMediaServerId(), |
| | | param.getApp(), param.getStream()); |
| | | JSONObject ret = new JSONObject(); |
| | | |
| | | boolean close = mediaService.closeStreamOnNoneReader(param.getMediaServerId(), param.getApp(), param.getStream(), null); |
| | | ret.put("code", close); |
| | | return ret; |
| | | } |
| | | |
| | | /** |
| | | * 当播放一个url,如果不存在时,会发出一个消息通知 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_stream_not_found", produces = "application/json;charset=UTF-8") |
| | | public HookResult onStreamNotFound(@RequestBody ABLHookParam param) { |
| | | logger.info("[ABL HOOK] 流未找到:{}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream()); |
| | | |
| | | |
| | | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (!userSetting.isAutoApplyPlay() || mediaServer == null) { |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | MediaNotFoundEvent mediaNotFoundEvent = MediaNotFoundEvent.getInstance(this, param, mediaServer); |
| | | applicationEventPublisher.publishEvent(mediaNotFoundEvent); |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * ABLMediaServer启动时会发送上线通知 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_server_started", produces = "application/json;charset=UTF-8") |
| | | public HookResult onServerStarted(HttpServletRequest request, @RequestBody OnServerStaredABLHookParam param) { |
| | | |
| | | logger.info("[ABL HOOK] 启动 " + param.getMediaServerId()); |
| | | try { |
| | | HookAblServerStartEvent event = new HookAblServerStartEvent(this); |
| | | MediaServer mediaServerItem = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServerItem != null) { |
| | | event.setMediaServerItem(mediaServerItem); |
| | | applicationEventPublisher.publishEvent(event); |
| | | } |
| | | }catch (Exception e) { |
| | | logger.info("[ABL-HOOK-启动] 发送通知失败 ", e); |
| | | } |
| | | |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * TODO 发送rtp(startSendRtp)被动关闭时回调 |
| | | */ |
| | | // @ResponseBody |
| | | // @PostMapping(value = "/on_send_rtp_stopped", produces = "application/json;charset=UTF-8") |
| | | // public HookResult onSendRtpStopped(HttpServletRequest request, @RequestBody OnSendRtpStoppedHookParam param) { |
| | | // |
| | | // logger.info("[ZLM HOOK] rtp发送关闭:{}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream()); |
| | | // |
| | | // // 查找对应的上级推流,发送停止 |
| | | // if (!"rtp".equals(param.getApp())) { |
| | | // return HookResult.SUCCESS(); |
| | | // } |
| | | // try { |
| | | // MediaSendRtpStoppedEvent event = new MediaSendRtpStoppedEvent(this); |
| | | // MediaServer mediaServerItem = mediaServerService.getOne(param.getMediaServerId()); |
| | | // if (mediaServerItem != null) { |
| | | // event.setMediaServer(mediaServerItem); |
| | | // applicationEventPublisher.publishEvent(event); |
| | | // } |
| | | // }catch (Exception e) { |
| | | // logger.info("[ZLM-HOOK-rtp发送关闭] 发送通知失败 ", e); |
| | | // } |
| | | // |
| | | // return HookResult.SUCCESS(); |
| | | // } |
| | | |
| | | /** |
| | | * TODO 录像完成事件 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_record_mp4", produces = "application/json;charset=UTF-8") |
| | | public HookResult onRecordMp4(HttpServletRequest request, @RequestBody OnRecordMp4ABLHookParam param) { |
| | | logger.info("[ABL HOOK] 录像完成事件:{}->{}", param.getMediaServerId(), param.getFileName()); |
| | | |
| | | // try { |
| | | // MediaServer mediaServerItem = mediaServerService.getOne(param.getMediaServerId()); |
| | | // if (mediaServerItem != null) { |
| | | // MediaRecordMp4Event event = MediaRecordMp4Event.getInstance(this, param, mediaServerItem); |
| | | // event.setMediaServer(mediaServerItem); |
| | | // applicationEventPublisher.publishEvent(event); |
| | | // } |
| | | // }catch (Exception e) { |
| | | // logger.info("[ZLM-HOOK-rtpServer收流超时] 发送通知失败 ", e); |
| | | // } |
| | | |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | /** |
| | | * 当某一路码流断开时会发送通知 |
| | | */ |
| | | @ResponseBody |
| | | @PostMapping(value = "/on_stream_disconnect", produces = "application/json;charset=UTF-8") |
| | | public HookResult onRecordMp4(HttpServletRequest request, @RequestBody ABLHookParam param) { |
| | | logger.info("[ABL HOOK] 码流断开事件, {}->{}/{}", param.getMediaServerId(), param.getApp(), param.getStream()); |
| | | |
| | | MediaServer mediaServer = mediaServerService.getOne(param.getMediaServerId()); |
| | | if (mediaServer == null) { |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | MediaDepartureEvent mediaDepartureEvent = MediaDepartureEvent.getInstance(this, param, mediaServer); |
| | | applicationEventPublisher.publishEvent(mediaDepartureEvent); |
| | | |
| | | return HookResult.SUCCESS(); |
| | | } |
| | | |
| | | private Map<String, String> urlParamToMap(String params) { |
| | | HashMap<String, String> map = new HashMap<>(); |
| | | if (ObjectUtils.isEmpty(params)) { |
| | | return map; |
| | | } |
| | | String[] paramsArray = params.split("&"); |
| | | if (paramsArray.length == 0) { |
| | | return map; |
| | | } |
| | | for (String param : paramsArray) { |
| | | String[] paramArray = param.split("="); |
| | | if (paramArray.length == 2) { |
| | | map.put(paramArray[0], paramArray[1]); |
| | | } |
| | | } |
| | | return map; |
| | | } |
| | | } |