|  |  |  | 
|---|
|  |  |  | package com.genersoft.iot.vmp.media.zlm; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import java.math.BigInteger; | 
|---|
|  |  |  | import java.text.DecimalFormat; | 
|---|
|  |  |  | import java.util.ArrayList; | 
|---|
|  |  |  | import java.util.List; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.alibaba.fastjson.JSON; | 
|---|
|  |  |  | import com.alibaba.fastjson.JSONArray; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.common.StreamInfo; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.conf.MediaServerConfig; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.storager.IVideoManagerStorager; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.utils.IpUtil; | 
|---|
|  |  |  | 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.http.HttpStatus; | 
|---|
|  |  |  | import org.springframework.http.ResponseEntity; | 
|---|
|  |  |  | import org.springframework.web.bind.annotation.PostMapping; | 
|---|
|  |  |  | 
|---|
|  |  |  | import com.alibaba.fastjson.JSONObject; | 
|---|
|  |  |  | import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import javax.servlet.http.HttpServletRequest; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * @Description:针对 ZLMediaServer的hook事件监听 | 
|---|
|  |  |  | * @author: songww | 
|---|
|  |  |  | * @author: swwheihei | 
|---|
|  |  |  | * @date:   2020年5月8日 上午10:46:48 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @RestController | 
|---|
|  |  |  | 
|---|
|  |  |  | public class ZLMHttpHookListener { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private final static Logger logger = LoggerFactory.getLogger(ZLMHttpHookListener.class); | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private SIPCommander cmder; | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private IVideoManagerStorager storager; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ZLMRESTfulUtils zlmresTfulUtils; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Value("${media.ip}") | 
|---|
|  |  |  | private String mediaIp; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Value("${media.port}") | 
|---|
|  |  |  | private int mediaPort; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 流量统计事件,播放器或推流器断开时并且耗用流量超过特定阈值时会触发此事件,阈值通过配置文件general.flowThreshold配置;此事件对回复不敏感。 | 
|---|
|  |  |  | * | 
|---|
|  |  |  | 
|---|
|  |  |  | logger.debug("ZLM HOOK on_flow_report API调用,参数:" + json.toString()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("err", ""); | 
|---|
|  |  |  | json.put("path", ""); | 
|---|
|  |  |  | json.put("second", 600); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("err", ""); | 
|---|
|  |  |  | ret.put("path", ""); | 
|---|
|  |  |  | ret.put("second", 600); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | if (logger.isDebugEnabled()) { | 
|---|
|  |  |  | logger.debug("ZLM HOOK on_publish API调用,参数:" + json.toString()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String app = json.getString("app"); | 
|---|
|  |  |  | String streamId = json.getString("id"); | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //      String ssrc = String.format("%10d", Integer.parseInt(streamId, 16)); // ZLM 要求大写且首位补零 | 
|---|
|  |  |  | String ssrc = new DecimalFormat("0000000000").format(Integer.parseInt(streamId, 16)); | 
|---|
|  |  |  | StreamInfo streamInfo = storager.queryPlayBySSRC(ssrc); | 
|---|
|  |  |  | if ("rtp".equals(app) && streamInfo != null ) { | 
|---|
|  |  |  | MediaServerConfig mediaInfo = storager.getMediaInfo(); | 
|---|
|  |  |  | streamInfo.setFlv(String.format("http://%s:%s/rtp/%s.flv", mediaInfo.getLocalIP(), mediaInfo.getHttpPort(), streamId)); | 
|---|
|  |  |  | streamInfo.setWs_flv(String.format("ws://%s:%s/rtp/%s.flv", mediaInfo.getLocalIP(), mediaInfo.getHttpPort(), streamId)); | 
|---|
|  |  |  | streamInfo.setRtmp(String.format("rtmp://%s:%s/rtp/%s", mediaInfo.getLocalIP(), mediaInfo.getRtmpPort(), streamId)); | 
|---|
|  |  |  | streamInfo.setHls(String.format("http://%s:%s/rtp/%s/hls.m3u8", mediaInfo.getLocalIP(), mediaInfo.getHttpPort(), streamId)); | 
|---|
|  |  |  | streamInfo.setRtsp(String.format("rtsp://%s:%s/rtp/%s", mediaInfo.getLocalIP(), mediaInfo.getRtspPort(), streamId)); | 
|---|
|  |  |  | storager.startPlay(streamInfo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | json.put("enableHls", true); | 
|---|
|  |  |  | json.put("enableMP4", false); | 
|---|
|  |  |  | json.put("enableRtxp", true); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | ret.put("enableHls", true); | 
|---|
|  |  |  | ret.put("enableMP4", false); | 
|---|
|  |  |  | ret.put("enableRtxp", true); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("realm", ""); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("realm", ""); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("encrypted", false); | 
|---|
|  |  |  | json.put("passwd", "test"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("encrypted", false); | 
|---|
|  |  |  | ret.put("passwd", "test"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | if (logger.isDebugEnabled()) { | 
|---|
|  |  |  | logger.debug("ZLM HOOK on_stream_changed API调用,参数:" + json.toString()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 流消失移除redis play | 
|---|
|  |  |  | String app = json.getString("app"); | 
|---|
|  |  |  | String streamId = json.getString("stream"); | 
|---|
|  |  |  | boolean regist = json.getBoolean("regist"); | 
|---|
|  |  |  | //      String ssrc = String.format("%10d", Integer.parseInt(streamId, 16)); // ZLM 要求大写且首位补零 | 
|---|
|  |  |  | String ssrc = new DecimalFormat("0000000000").format(Integer.parseInt(streamId, 16)); | 
|---|
|  |  |  | StreamInfo streamInfo = storager.queryPlayBySSRC(ssrc); | 
|---|
|  |  |  | if ("rtp".equals(app) && !regist ) { | 
|---|
|  |  |  | storager.stopPlay(streamInfo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | cmder.streamByeCmd(ssrc); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("close", true); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("close", true); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @ResponseBody | 
|---|
|  |  |  | @PostMapping(value = "/on_server_started", produces = "application/json;charset=UTF-8") | 
|---|
|  |  |  | public ResponseEntity<String> onServerStarted(@RequestBody JSONObject json){ | 
|---|
|  |  |  | public ResponseEntity<String> onServerStarted(HttpServletRequest request, @RequestBody JSONObject json){ | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (logger.isDebugEnabled()) { | 
|---|
|  |  |  | logger.debug("ZLM HOOK on_server_started API调用,参数:" + json.toString()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //      String data = json.getString("data"); | 
|---|
|  |  |  | //      List<MediaServerConfig> mediaServerConfigs = JSON.parseArray(JSON.toJSONString(json), MediaServerConfig.class); | 
|---|
|  |  |  | //      MediaServerConfig mediaServerConfig = mediaServerConfigs.get(0); | 
|---|
|  |  |  | MediaServerConfig mediaServerConfig = JSON.toJavaObject(json, MediaServerConfig.class); | 
|---|
|  |  |  | mediaServerConfig.setLocalIP(mediaIp); | 
|---|
|  |  |  | storager.updateMediaInfo(mediaServerConfig); | 
|---|
|  |  |  | // TODO Auto-generated method stub | 
|---|
|  |  |  |  | 
|---|
|  |  |  | JSONObject ret = new JSONObject(); | 
|---|
|  |  |  | json.put("code", 0); | 
|---|
|  |  |  | json.put("msg", "success"); | 
|---|
|  |  |  | ret.put("code", 0); | 
|---|
|  |  |  | ret.put("msg", "success"); | 
|---|
|  |  |  | return new ResponseEntity<String>(ret.toString(),HttpStatus.OK); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|