支持通用通道国标类型的播放

结构优化
648540858 2023-11-24 10:12:36 +08:00
parent cbc0ca91b1
commit d72f57c772
14 changed files with 474 additions and 163 deletions

View File

@ -1,7 +1,10 @@
package com.genersoft.iot.vmp.common; package com.genersoft.iot.vmp.common;
import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
import io.swagger.v3.oas.annotations.media.Schema; import io.swagger.v3.oas.annotations.media.Schema;
import java.util.List;
public class CommonGbChannel { public class CommonGbChannel {
@ -597,4 +600,9 @@ public class CommonGbChannel {
public void setCreateTime(String createTime) { public void setCreateTime(String createTime) {
this.createTime = createTime; this.createTime = createTime;
} }
public static CommonGbChannel getInstance(List<String> syncKeys, DeviceChannel channel){
}
} }

View File

@ -0,0 +1,123 @@
package com.genersoft.iot.vmp.gb28181;
import com.genersoft.iot.vmp.common.CommonGbChannel;
import com.genersoft.iot.vmp.common.StreamInfo;
import com.genersoft.iot.vmp.conf.exception.ControllerException;
import com.genersoft.iot.vmp.gb28181.bean.Device;
import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
import com.genersoft.iot.vmp.media.zlm.dto.MediaServerItem;
import com.genersoft.iot.vmp.service.IDeviceChannelService;
import com.genersoft.iot.vmp.service.IPlayService;
import com.genersoft.iot.vmp.service.IResourcePlayCallback;
import com.genersoft.iot.vmp.service.IResourceService;
import com.genersoft.iot.vmp.service.bean.InviteErrorCode;
import com.genersoft.iot.vmp.storager.dao.DeviceChannelMapper;
import com.genersoft.iot.vmp.storager.dao.DeviceMapper;
import com.genersoft.iot.vmp.storager.impl.RedisCatchStorageImpl;
import com.genersoft.iot.vmp.vmanager.bean.ErrorCode;
import com.genersoft.iot.vmp.vmanager.bean.StreamContent;
import com.genersoft.iot.vmp.vmanager.bean.WVPResult;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.net.MalformedURLException;
import java.net.URL;
/**
*
*/
@Service("28181")
public class GB28181ResourceServiceImpl implements IResourceService {
private final Logger logger = LoggerFactory.getLogger(GB28181ResourceServiceImpl.class);
public static final String resourceType = "28181";
@Autowired
private DeviceMapper deviceMapper;
@Autowired
private DeviceChannelMapper deviceChannelMapper;
@Autowired
private IPlayService playService;
@Override
public boolean deleteChannel(CommonGbChannel commonGbChannel) {
if (!GB28181ResourceServiceImpl.resourceType.equals(commonGbChannel.getType())) {
logger.warn("[资源类-国标28181] 收到移除通道: {} 时发现类型不为28181", commonGbChannel.getCommonGbId());
return false;
}
return deviceChannelMapper.removeCommonChannelId(commonGbChannel.getCommonGbId()) > 0;
}
@Override
public void startPlay(CommonGbChannel commonGbChannel, IResourcePlayCallback callback) {
assert callback != null;
if (!GB28181ResourceServiceImpl.resourceType.equals(commonGbChannel.getType())) {
logger.warn("[资源类-国标28181] 收到播放通道: {} 时发现类型不为28181", commonGbChannel.getCommonGbId());
callback.call(commonGbChannel, ErrorCode.ERROR500.getCode(), ErrorCode.ERROR500.getMsg(), null);
return;
}
DeviceChannel channel = deviceChannelMapper.getChannelByCommonChannelId(commonGbChannel.getCommonGbId());
if (channel == null) {
logger.warn("[资源类-国标28181] 收到播放通道: {} 时未找到国标通道", commonGbChannel.getCommonGbId());
callback.call(commonGbChannel, ErrorCode.ERROR500.getCode(), ErrorCode.ERROR500.getMsg(), null);
return;
}
Device device = deviceMapper.getDeviceByDeviceId(channel.getDeviceId());
if (device == null) {
logger.warn("[资源类-国标28181] 收到播放通道: {} 时未找到通道 {} 所属的国标设备",
commonGbChannel.getCommonGbId(), channel.getDeviceId());
callback.call(commonGbChannel, ErrorCode.ERROR500.getCode(), ErrorCode.ERROR500.getMsg(), null);
return;
}
MediaServerItem mediaServerItem = playService.getNewMediaServerItem(device);
playService.play(mediaServerItem, channel.getDeviceId(), channel.getChannelId(), null, (code, msg, data) -> {
if (code == InviteErrorCode.SUCCESS.getCode()) {
if (data != null) {
StreamInfo streamInfo = (StreamInfo)data;
callback.call(commonGbChannel, ErrorCode.SUCCESS.getCode(), ErrorCode.SUCCESS.getMsg(), streamInfo);
}
}else {
callback.call(commonGbChannel, code, msg, null);
}
});
}
@Override
public void stopPlay(CommonGbChannel commonGbChannel, IResourcePlayCallback callback) {
if (!GB28181ResourceServiceImpl.resourceType.equals(commonGbChannel.getType())) {
logger.warn("[资源类-国标28181] 收到停止播放通道: {} 时发现类型不为28181", commonGbChannel.getCommonGbId());
if (callback != null) {
callback.call(commonGbChannel, ErrorCode.ERROR500.getCode(), ErrorCode.ERROR500.getMsg(), null);
}
return;
}
DeviceChannel channel = deviceChannelMapper.getChannelByCommonChannelId(commonGbChannel.getCommonGbId());
if (channel == null) {
logger.warn("[资源类-国标28181] 收到停止播放通道: {} 时未找到国标通道", commonGbChannel.getCommonGbId());
if (callback != null) {
callback.call(commonGbChannel, ErrorCode.ERROR500.getCode(), ErrorCode.ERROR500.getMsg(), null);
}
return;
}
try {
playService.stop(channel.getDeviceId(), channel.getChannelId());
} catch (ControllerException exception) {
if (callback != null) {
callback.call(commonGbChannel, exception.getCode(), exception.getMsg(), null);
}
}
}
@Override
public boolean ptzControl(CommonGbChannel commonGbChannel,
String command, Integer horizonSpeed,
Integer verticalSpeed, Integer zoomSpeed) {
return false;
}
}

View File

@ -55,6 +55,9 @@ public class DeferredResultHolder {
public static final String CALLBACK_CMD_SNAP= "CALLBACK_SNAP"; public static final String CALLBACK_CMD_SNAP= "CALLBACK_SNAP";
public static final String CALLBACK_CHANNEL_PLAY = "CALLBACK_CHANNEL_PLAY";
private Map<String, Map<String, DeferredResultEx>> map = new ConcurrentHashMap<>(); private Map<String, Map<String, DeferredResultEx>> map = new ConcurrentHashMap<>();

View File

@ -12,6 +12,10 @@ import com.github.pagehelper.PageInfo;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
/**
*
*
*/
public interface ICommonGbChannelService { public interface ICommonGbChannelService {
CommonGbChannel getChannel(String channelId); CommonGbChannel getChannel(String channelId);
@ -56,8 +60,6 @@ public interface ICommonGbChannelService {
PageInfo<CommonGbChannel> queryChannelList(String query, int page, int count); PageInfo<CommonGbChannel> queryChannelList(String query, int page, int count);
String getRandomCode(Gb28181CodeType type);
List<IndustryCodeType> getIndustryCodeList(); List<IndustryCodeType> getIndustryCodeList();
List<DeviceType> getDeviceTypeList(); List<DeviceType> getDeviceTypeList();
@ -71,4 +73,8 @@ public interface ICommonGbChannelService {
void removeFromRegion(UpdateCommonChannelToRegion params); void removeFromRegion(UpdateCommonChannelToRegion params);
void updateChannelToRegion(UpdateCommonChannelToRegion params); void updateChannelToRegion(UpdateCommonChannelToRegion params);
void startPlay(CommonGbChannel channel, IResourcePlayCallback callback);
void stopPlay(CommonGbChannel channel, IResourcePlayCallback callback);
} }

View File

@ -92,4 +92,5 @@ public interface IDeviceChannelService {
* *
*/ */
boolean resetChannels(Device device, List<DeviceChannel> deviceChannelList); boolean resetChannels(Device device, List<DeviceChannel> deviceChannelList);
} }

View File

@ -45,4 +45,5 @@ public interface IPlayService {
void getSnap(String deviceId, String channelId, String fileName, ErrorCallback errorCallback); void getSnap(String deviceId, String channelId, String fileName, ErrorCallback errorCallback);
void stop(String deviceId, String channelId);
} }

View File

@ -0,0 +1,19 @@
package com.genersoft.iot.vmp.service;
import com.genersoft.iot.vmp.common.CommonGbChannel;
import com.genersoft.iot.vmp.common.StreamInfo;
/**
*
*/
public interface IResourcePlayCallback {
/**
*
* @param commonGbChannel
* @param code
* @param message
* @param streamInfo
*/
void call(CommonGbChannel commonGbChannel, int code, String message, StreamInfo streamInfo);
}

View File

@ -0,0 +1,40 @@
package com.genersoft.iot.vmp.service;
import com.genersoft.iot.vmp.common.CommonGbChannel;
import com.genersoft.iot.vmp.common.StreamInfo;
/**
*
* GIS
*/
public interface IResourceService {
/**
*
*/
boolean deleteChannel(CommonGbChannel commonGbChannel);
/**
*
*/
void startPlay(CommonGbChannel commonGbChannel, IResourcePlayCallback callback);
/**
*
*/
void stopPlay(CommonGbChannel commonGbChannel, IResourcePlayCallback callback);
/**
*
* @param commonGbChannel
* @param command ,: left, right, up, down, upleft, upright, downleft, downright, zoomin, zoomout, stop
* @param horizonSpeed 0-255
* @param verticalSpeed 0-255
* @param zoomSpeed
* @return
*/
boolean ptzControl(CommonGbChannel commonGbChannel, String command,
Integer horizonSpeed, Integer verticalSpeed, Integer zoomSpeed);
}

View File

@ -8,6 +8,8 @@ import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
import com.genersoft.iot.vmp.gb28181.bean.Gb28181CodeType; import com.genersoft.iot.vmp.gb28181.bean.Gb28181CodeType;
import com.genersoft.iot.vmp.gb28181.utils.SipUtils; import com.genersoft.iot.vmp.gb28181.utils.SipUtils;
import com.genersoft.iot.vmp.service.ICommonGbChannelService; import com.genersoft.iot.vmp.service.ICommonGbChannelService;
import com.genersoft.iot.vmp.service.IResourcePlayCallback;
import com.genersoft.iot.vmp.service.IResourceService;
import com.genersoft.iot.vmp.service.bean.*; import com.genersoft.iot.vmp.service.bean.*;
import com.genersoft.iot.vmp.storager.dao.CommonGbChannelMapper; import com.genersoft.iot.vmp.storager.dao.CommonGbChannelMapper;
import com.genersoft.iot.vmp.storager.dao.DeviceChannelMapper; import com.genersoft.iot.vmp.storager.dao.DeviceChannelMapper;
@ -57,6 +59,9 @@ public class CommonGbChannelServiceImpl implements ICommonGbChannelService {
@Autowired @Autowired
private CivilCodeFileConf civilCodeFileConf; private CivilCodeFileConf civilCodeFileConf;
@Autowired
private Map<String, IResourceService> resourceServiceMap;
@Override @Override
public CommonGbChannel getChannel(String channelId) { public CommonGbChannel getChannel(String channelId) {
@ -634,12 +639,6 @@ public class CommonGbChannelServiceImpl implements ICommonGbChannelService {
return new PageInfo<>(all); return new PageInfo<>(all);
} }
@Override
public String getRandomCode(Gb28181CodeType type) {
return "";
}
@Override @Override
public List<IndustryCodeType> getIndustryCodeList() { public List<IndustryCodeType> getIndustryCodeList() {
IndustryCodeTypeEnum[] values = IndustryCodeTypeEnum.values(); IndustryCodeTypeEnum[] values = IndustryCodeTypeEnum.values();
@ -702,4 +701,20 @@ public class CommonGbChannelServiceImpl implements ICommonGbChannelService {
public void updateChannelToRegion(UpdateCommonChannelToRegion params) { public void updateChannelToRegion(UpdateCommonChannelToRegion params) {
commonGbChannelMapper.updateChannelToRegion(params); commonGbChannelMapper.updateChannelToRegion(params);
} }
@Override
public void startPlay(CommonGbChannel channel, IResourcePlayCallback callback) {
IResourceService resourceService = resourceServiceMap.get(channel.getType());
assert resourceService != null;
resourceService.startPlay(channel, callback);
}
@Override
public void stopPlay(CommonGbChannel channel, IResourcePlayCallback callback) {
IResourceService resourceService = resourceServiceMap.get(channel.getType());
assert resourceService != null;
resourceService.stopPlay(channel,callback);
}
} }

View File

@ -59,6 +59,7 @@ public class DeviceChannelServiceImpl implements IDeviceChannelService {
@Autowired @Autowired
ICommonGbChannelService commonGbChannelService; ICommonGbChannelService commonGbChannelService;
@Override @Override
public DeviceChannel updateGps(DeviceChannel deviceChannel, Device device) { public DeviceChannel updateGps(DeviceChannel deviceChannel, Device device) {
if (deviceChannel.getLongitude()*deviceChannel.getLatitude() > 0) { if (deviceChannel.getLongitude()*deviceChannel.getLatitude() > 0) {
@ -97,7 +98,6 @@ public class DeviceChannelServiceImpl implements IDeviceChannelService {
public void updateChannel(String deviceId, DeviceChannel channel) { public void updateChannel(String deviceId, DeviceChannel channel) {
String channelId = channel.getChannelId(); String channelId = channel.getChannelId();
channel.setDeviceId(deviceId); channel.setDeviceId(deviceId);
// StreamInfo streamInfo = redisCatchStorage.queryPlayByDevice(deviceId, channelId);
InviteInfo inviteInfo = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId); InviteInfo inviteInfo = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId);
if (inviteInfo != null && inviteInfo.getStreamInfo() != null) { if (inviteInfo != null && inviteInfo.getStreamInfo() != null) {
channel.setStreamId(inviteInfo.getStreamInfo().getStream()); channel.setStreamId(inviteInfo.getStreamInfo().getStream());
@ -283,25 +283,24 @@ public class DeviceChannelServiceImpl implements IDeviceChannelService {
if (CollectionUtils.isEmpty(deviceChannelList)) { if (CollectionUtils.isEmpty(deviceChannelList)) {
return false; return false;
} }
List<DeviceChannel> allChannels = channelMapper.queryAllChannels(device.getDeviceId()); Map<String, DeviceChannel> allChannelMap = channelMapper.queryAllChannelsForMap(device.getDeviceId());
Map<String,DeviceChannel> allChannelMap = new ConcurrentHashMap<>();
if (!allChannels.isEmpty()) { // 存储数据,方便对数据去重
allChannels.stream().forEach(deviceChannel -> {
allChannelMap.put(deviceChannel.getChannelId(), deviceChannel);
});
}
// 数据去重
List<DeviceChannel> channels = new ArrayList<>(); List<DeviceChannel> channels = new ArrayList<>();
List<DeviceChannel> updateChannels = new ArrayList<>(); // 存储需要更新的数据
List<DeviceChannel> updateChannelsForInfo = new ArrayList<>();
List<DeviceChannel> updateChannelsForStatus = new ArrayList<>();
// 存储需要需要新增的数据库
List<DeviceChannel> addChannels = new ArrayList<>(); List<DeviceChannel> addChannels = new ArrayList<>();
StringBuilder stringBuilder = new StringBuilder(); StringBuilder stringBuilder = new StringBuilder();
Map<String, Integer> subContMap = new HashMap<>(); Map<String, Integer> subContMap = new HashMap<>();
// 数据去重
Set<String> gbIdSet = new HashSet<>(); Set<String> gbIdSet = new HashSet<>();
for (DeviceChannel deviceChannel : deviceChannelList) { for (DeviceChannel deviceChannel : deviceChannelList) {
// 数据去重
if (gbIdSet.contains(deviceChannel.getChannelId())) { if (gbIdSet.contains(deviceChannel.getChannelId())) {
stringBuilder.append(deviceChannel.getChannelId()).append(","); stringBuilder.append(deviceChannel.getChannelId()).append(",");
continue; continue;
@ -311,17 +310,21 @@ public class DeviceChannelServiceImpl implements IDeviceChannelService {
deviceChannel.setStreamId(allChannelMap.get(deviceChannel.getChannelId()).getStreamId()); deviceChannel.setStreamId(allChannelMap.get(deviceChannel.getChannelId()).getStreamId());
deviceChannel.setHasAudio(allChannelMap.get(deviceChannel.getChannelId()).isHasAudio()); deviceChannel.setHasAudio(allChannelMap.get(deviceChannel.getChannelId()).isHasAudio());
deviceChannel.setCommonGbChannelId(allChannelMap.get(deviceChannel.getChannelId()).getCommonGbChannelId()); deviceChannel.setCommonGbChannelId(allChannelMap.get(deviceChannel.getChannelId()).getCommonGbChannelId());
if (allChannelMap.get(deviceChannel.getChannelId()).isStatus() !=deviceChannel.isStatus()){ deviceChannel.setCommonGbChannelId(allChannelMap.get(deviceChannel.getChannelId()).getCommonGbChannelId());
List<String> strings = platformChannelMapper.queryParentPlatformByChannelId(deviceChannel.getChannelId());
if (!CollectionUtils.isEmpty(strings)){
strings.forEach(platformId->{
eventPublisher.catalogEventPublish(platformId, deviceChannel, deviceChannel.isStatus()? CatalogEvent.ON:CatalogEvent.OFF);
});
}
}
deviceChannel.setUpdateTime(DateUtil.getNow()); deviceChannel.setUpdateTime(DateUtil.getNow());
updateChannels.add(deviceChannel); // 同步时发现状态变化
if (allChannelMap.get(deviceChannel.getChannelId()).isStatus() !=deviceChannel.isStatus()){
// TODO 应该通知给commonChannel
updateChannelsForStatus.add(deviceChannel);
// List<String> strings = platformChannelMapper.queryParentPlatformByChannelId(deviceChannel.getChannelId());
// if (!CollectionUtils.isEmpty(strings)){
// strings.forEach(platformId->{
// eventPublisher.catalogEventPublish(platformId, deviceChannel, deviceChannel.isStatus()? CatalogEvent.ON:CatalogEvent.OFF);
// });
// }
}else {
updateChannelsForInfo.add(deviceChannel);
}
}else { }else {
deviceChannel.setCreateTime(DateUtil.getNow()); deviceChannel.setCreateTime(DateUtil.getNow());
deviceChannel.setUpdateTime(DateUtil.getNow()); deviceChannel.setUpdateTime(DateUtil.getNow());
@ -385,17 +388,17 @@ public class DeviceChannelServiceImpl implements IDeviceChannelService {
result = channelMapper.batchAdd(addChannels) < 0; result = channelMapper.batchAdd(addChannels) < 0;
} }
} }
if (!updateChannels.isEmpty()) { if (!updateChannelsForInfo.isEmpty()) {
if (updateChannels.size() > limitCount) { if (updateChannelsForInfo.size() > limitCount) {
for (int i = 0; i < updateChannels.size(); i += limitCount) { for (int i = 0; i < updateChannelsForInfo.size(); i += limitCount) {
int toIndex = i + limitCount; int toIndex = i + limitCount;
if (i + limitCount > updateChannels.size()) { if (i + limitCount > updateChannelsForInfo.size()) {
toIndex = updateChannels.size(); toIndex = updateChannelsForInfo.size();
} }
result = result || channelMapper.batchUpdate(updateChannels.subList(i, toIndex)) < 0; result = result || channelMapper.batchUpdate(updateChannelsForInfo.subList(i, toIndex)) < 0;
} }
}else { }else {
result = result || channelMapper.batchUpdate(updateChannels) < 0; result = result || channelMapper.batchUpdate(updateChannelsForInfo) < 0;
} }
} }
} }

View File

@ -32,6 +32,7 @@ import com.genersoft.iot.vmp.service.bean.InviteErrorCode;
import com.genersoft.iot.vmp.service.bean.SSRCInfo; import com.genersoft.iot.vmp.service.bean.SSRCInfo;
import com.genersoft.iot.vmp.storager.IRedisCatchStorage; import com.genersoft.iot.vmp.storager.IRedisCatchStorage;
import com.genersoft.iot.vmp.storager.IVideoManagerStorage; import com.genersoft.iot.vmp.storager.IVideoManagerStorage;
import com.genersoft.iot.vmp.storager.dao.DeviceMapper;
import com.genersoft.iot.vmp.utils.DateUtil; import com.genersoft.iot.vmp.utils.DateUtil;
import com.genersoft.iot.vmp.vmanager.bean.ErrorCode; import com.genersoft.iot.vmp.vmanager.bean.ErrorCode;
import gov.nist.javax.sip.message.SIPResponse; import gov.nist.javax.sip.message.SIPResponse;
@ -74,9 +75,6 @@ public class PlayServiceImpl implements IPlayService {
@Autowired @Autowired
private IInviteStreamService inviteStreamService; private IInviteStreamService inviteStreamService;
@Autowired
private DeferredResultHolder resultHolder;
@Autowired @Autowired
private ZLMRESTfulUtils zlmresTfulUtils; private ZLMRESTfulUtils zlmresTfulUtils;
@ -96,7 +94,7 @@ public class PlayServiceImpl implements IPlayService {
private VideoStreamSessionManager streamSession; private VideoStreamSessionManager streamSession;
@Autowired @Autowired
private IDeviceService deviceService; private DeviceMapper deviceMapper;
@Autowired @Autowired
private UserSetting userSetting; private UserSetting userSetting;
@ -298,6 +296,29 @@ public class PlayServiceImpl implements IPlayService {
} }
} }
@Override
public void stop(String deviceId, String channelId) {
Device device = storager.queryVideoDevice(deviceId);
if (device == null) {
throw new ControllerException(ErrorCode.ERROR100.getCode(), "设备[" + deviceId + "]不存在");
}
InviteInfo inviteInfo = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId);
if (inviteInfo == null) {
throw new ControllerException(ErrorCode.ERROR100.getCode(), "点播未找到");
}
if (InviteSessionStatus.ok == inviteInfo.getStatus()) {
try {
logger.info("[停止点播] {}/{}", deviceId, channelId);
cmder.streamByeCmd(device, channelId, inviteInfo.getStream(), null, null);
} catch (InvalidArgumentException | SipException | ParseException | SsrcTransactionNotFoundException e) {
logger.error("[命令发送失败] 停止点播, 发送BYE: {}", e.getMessage());
throw new ControllerException(ErrorCode.ERROR100.getCode(), "命令发送失败: " + e.getMessage());
}
}
inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId);
storager.stopPlay(deviceId, channelId);
}
private void tcpActiveHandler(Device device, String channelId, String contentString, private void tcpActiveHandler(Device device, String channelId, String contentString,
MediaServerItem mediaServerItem, MediaServerItem mediaServerItem,
String timeOutTaskKey, SSRCInfo ssrcInfo, ErrorCallback<Object> callback){ String timeOutTaskKey, SSRCInfo ssrcInfo, ErrorCallback<Object> callback){
@ -828,7 +849,7 @@ public class PlayServiceImpl implements IPlayService {
if (allSsrc.size() > 0) { if (allSsrc.size() > 0) {
for (SsrcTransaction ssrcTransaction : allSsrc) { for (SsrcTransaction ssrcTransaction : allSsrc) {
if (ssrcTransaction.getMediaServerId().equals(mediaServerId)) { if (ssrcTransaction.getMediaServerId().equals(mediaServerId)) {
Device device = deviceService.getDevice(ssrcTransaction.getDeviceId()); Device device = deviceMapper.getDeviceByDeviceId(ssrcTransaction.getDeviceId());
if (device == null) { if (device == null) {
continue; continue;
} }
@ -944,7 +965,7 @@ public class PlayServiceImpl implements IPlayService {
@Override @Override
public void getSnap(String deviceId, String channelId, String fileName, ErrorCallback errorCallback) { public void getSnap(String deviceId, String channelId, String fileName, ErrorCallback errorCallback) {
Device device = deviceService.getDevice(deviceId); Device device = deviceMapper.getDeviceByDeviceId(deviceId);
if (device == null) { if (device == null) {
errorCallback.run(InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getCode(), InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getMsg(), null); errorCallback.run(InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getCode(), InviteErrorCode.ERROR_FOR_PARAMETER_ERROR.getMsg(), null);
return; return;

View File

@ -10,6 +10,7 @@ import org.apache.ibatis.annotations.Param;
import org.springframework.stereotype.Repository; import org.springframework.stereotype.Repository;
import java.util.List; import java.util.List;
import java.util.Map;
/** /**
* *
@ -405,6 +406,9 @@ public interface DeviceChannelMapper {
@Select("select * from wvp_device_channel where device_id = #{deviceId}") @Select("select * from wvp_device_channel where device_id = #{deviceId}")
List<DeviceChannel> queryAllChannels(String deviceId); List<DeviceChannel> queryAllChannels(String deviceId);
@MapKey("channelId")
@Select("select * from wvp_device_channel where device_id = #{deviceId}")
Map<String, DeviceChannel> queryAllChannelsForMap(String deviceId);
@Select("select channelId" + @Select("select channelId" +
", device_id" + ", device_id" +
@ -473,4 +477,17 @@ public interface DeviceChannelMapper {
" where wdc.device_id = #{deviceId}") " where wdc.device_id = #{deviceId}")
int updateCommonChannelId(@Param("deviceId") String deviceId); int updateCommonChannelId(@Param("deviceId") String deviceId);
@Select(value = {" <script>" +
"select * " +
"from wvp_device_channel " +
"where common_gb_channel_id=#{commonGbId}" +
" </script>"})
DeviceChannel getChannelByCommonChannelId(@Param("commonGbId") int commonGbId);
@Update(" update wvp_device_channel " +
" set common_gb_channel_id = null " +
" where common_gb_channel_id = #{commonGbId}")
int removeCommonChannelId(@Param("commonGbId") int commonGbId);
} }

View File

@ -1,11 +1,19 @@
package com.genersoft.iot.vmp.vmanager.channel; package com.genersoft.iot.vmp.vmanager.channel;
import com.alibaba.fastjson2.JSONObject;
import com.genersoft.iot.vmp.common.CommonGbChannel; import com.genersoft.iot.vmp.common.CommonGbChannel;
import com.genersoft.iot.vmp.common.InviteSessionType;
import com.genersoft.iot.vmp.common.StreamInfo;
import com.genersoft.iot.vmp.conf.UserSetting;
import com.genersoft.iot.vmp.conf.exception.ControllerException;
import com.genersoft.iot.vmp.gb28181.bean.Device;
import com.genersoft.iot.vmp.gb28181.bean.Gb28181CodeType; import com.genersoft.iot.vmp.gb28181.bean.Gb28181CodeType;
import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder;
import com.genersoft.iot.vmp.gb28181.transmit.callback.RequestMessage;
import com.genersoft.iot.vmp.media.zlm.dto.MediaServerItem;
import com.genersoft.iot.vmp.service.ICommonGbChannelService; import com.genersoft.iot.vmp.service.ICommonGbChannelService;
import com.genersoft.iot.vmp.service.bean.*; import com.genersoft.iot.vmp.service.bean.*;
import com.genersoft.iot.vmp.vmanager.bean.UpdateCommonChannelToGroup; import com.genersoft.iot.vmp.vmanager.bean.*;
import com.genersoft.iot.vmp.vmanager.bean.UpdateCommonChannelToRegion;
import com.github.pagehelper.PageInfo; import com.github.pagehelper.PageInfo;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.Operation;
@ -16,9 +24,14 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*; import org.springframework.web.bind.annotation.*;
import org.springframework.web.context.request.async.DeferredResult;
import javax.servlet.http.HttpServletRequest;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.Arrays; import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.UUID;
@Tag(name = "通用国标通道") @Tag(name = "通用国标通道")
@ -31,23 +44,70 @@ public class CommonChannelController {
@Autowired @Autowired
private ICommonGbChannelService commonGbChannelService; private ICommonGbChannelService commonGbChannelService;
@Autowired
private DeferredResultHolder resultHolder;
@Operation(summary = "查询区域下的通道") @Autowired
@Parameter(name = "regionDeviceId", description = "区域的编号", required = true) private UserSetting userSetting;
@Parameter(name = "query", description = "要搜索的内容", required = false)
@Parameter(name = "page", description = "当前页", required = true)
@Parameter(name = "count", description = "每页查询数量", required = true) /**
@GetMapping("/region/list") * TODO
public PageInfo<CommonGbChannel> getChannelsInRegion( *
@RequestParam(required = true) String regionDeviceId, * @param deviceId
@RequestParam(required = false) String query, */
@RequestParam(required = true) int page, @GetMapping("/sync/device")
@RequestParam(required = true) int count @Operation(summary = "从下级设备中同步通道")
) { @Parameter(name = "deviceId", description = "设备编号")
return commonGbChannelService.getChannelsInRegion(regionDeviceId, query, page, count); @Parameter(name = "syncKeys", description = "选择性同步的字段")
public boolean syncFromDevice(String deviceId, String[] syncKeys,
@RequestParam(required = false) Boolean syncGroup,
@RequestParam(required = false) Boolean syncRegion) {
return commonGbChannelService.syncChannelFromGb28181Device(deviceId, Lists.newArrayList(syncKeys), syncGroup, syncRegion);
} }
@Operation(summary = "查询分组下的通道")
@Operation(summary = "更新通道信息")
@Parameter(name = "CommonGbChannel", description = "commonGbChannel", required = true)
@ResponseBody
@GetMapping("/update")
public void update(
@RequestParam(required = false) CommonGbChannel commonGbChannel
){
commonGbChannelService.update(commonGbChannel);
}
/**
* TODO controller
*/
@Operation(summary = "获取行业编码列表")
@ResponseBody
@GetMapping("/industry/list")
public List<IndustryCodeType> getIndustryCodeList(){
return commonGbChannelService.getIndustryCodeList();
}
/**
* TODO controller
*/
@Operation(summary = "获取编码列表")
@ResponseBody
@GetMapping("/type/list")
public List<DeviceType> getDeviceTypeList(){
return commonGbChannelService.getDeviceTypeList();
}
/**
* TODO controller
*/
@Operation(summary = "获取编码列表")
@ResponseBody
@GetMapping("/network/identification/list")
public List<NetworkIdentificationType> getNetworkIdentificationTypeList(){
return commonGbChannelService.getNetworkIdentificationTypeList();
}
@Operation(summary = "查询分组或区域下的通道")
@Parameter(name = "groupDeviceId", description = "分组的编号", required = false) @Parameter(name = "groupDeviceId", description = "分组的编号", required = false)
@Parameter(name = "regionDeviceId", description = "区域的编号", required = false) @Parameter(name = "regionDeviceId", description = "区域的编号", required = false)
@Parameter(name = "query", description = "要搜索的内容", required = false) @Parameter(name = "query", description = "要搜索的内容", required = false)
@ -86,81 +146,21 @@ public class CommonChannelController {
inGroup, inRegion, type); inGroup, inRegion, type);
} }
/** @Operation(summary = "为区域添加分组")
* @ResponseBody
* @PostMapping("/region/update")
* @param deviceId public void updateChannelToRegion(@RequestBody UpdateCommonChannelToRegion params){
*/ assert params.getCommonGbCivilCode() != null;
@GetMapping("/sync/device") assert !params.getCommonGbIds().isEmpty();
@Operation(summary = "从下级设备中同步通道") commonGbChannelService.updateChannelToRegion(params);
@Parameter(name = "deviceId", description = "设备编号")
@Parameter(name = "syncKeys", description = "选择性同步的字段")
public boolean syncFromDevice(String deviceId, String[] syncKeys,
@RequestParam(required = false) Boolean syncGroup,
@RequestParam(required = false) Boolean syncRegion) {
return commonGbChannelService.syncChannelFromGb28181Device(deviceId, Lists.newArrayList(syncKeys), syncGroup, syncRegion);
} }
@Operation(summary = "从区域中移除通道")
// @Operation(summary = "分页查询通道")
// @Parameter(name = "query", description = "要搜索的内容", required = false)
// @Parameter(name = "page", description = "当前页", required = true)
// @Parameter(name = "count", description = "每页查询数量", required = true)
// @ResponseBody
// @GetMapping("/list")
// public PageInfo<CommonGbChannel> queryChannelList(
// @RequestParam(required = false) String query,
// @RequestParam(required = true) int page,
// @RequestParam(required = true) int count ){
//
// return commonGbChannelService.queryChannelList(query, page, count);
// }
@Operation(summary = "更新通道")
@Parameter(name = "CommonGbChannel", description = "commonGbChannel", required = true)
@ResponseBody @ResponseBody
@GetMapping("/update") @PostMapping("/region/remove")
public void update( public void removeFromRegion(@RequestBody UpdateCommonChannelToRegion params){
@RequestParam(required = false) CommonGbChannel commonGbChannel assert params.getCommonGbCivilCode() != null || !params.getCommonGbIds().isEmpty();
){ commonGbChannelService.removeFromRegion(params);
commonGbChannelService.update(commonGbChannel);
}
@Operation(summary = "获取一个随机的可用国标编号")
@Parameter(name = "type", description = "类型: " +
"CIVIL_CODE_PROVINCE 省级编号 " +
"CIVIL_CODE_CIT 市级编号" +
"CIVIL_CODE_GRASS_ROOTS 区级编号" +
"CIVIL_CODE_GRASS_ROOTS 基层接入单位编号 " +
"BUSINESS_GROUP 业务分组 " +
"VIRTUAL_ORGANIZATION 虚拟组织 ", required = true)
@ResponseBody
@GetMapping("/code/random")
public String getRandomCode(
@RequestParam(required = true) Gb28181CodeType type
){
return commonGbChannelService.getRandomCode(type);
}
@Operation(summary = "获取行业编码列表")
@ResponseBody
@GetMapping("/industry/list")
public List<IndustryCodeType> getIndustryCodeList(){
return commonGbChannelService.getIndustryCodeList();
}
@Operation(summary = "获取编码列表")
@ResponseBody
@GetMapping("/type/list")
public List<DeviceType> getDeviceTypeList(){
return commonGbChannelService.getDeviceTypeList();
}
@Operation(summary = "获取编码列表")
@ResponseBody
@GetMapping("/network/identification/list")
public List<NetworkIdentificationType> getNetworkIdentificationTypeList(){
return commonGbChannelService.getNetworkIdentificationTypeList();
} }
@Operation(summary = "为通道添加分组") @Operation(summary = "为通道添加分组")
@ -180,22 +180,96 @@ public class CommonChannelController {
commonGbChannelService.removeFromGroup(params); commonGbChannelService.removeFromGroup(params);
} }
@Operation(summary = "从区域中移除通道")
@Operation(summary = "播放通道")
@Parameter(name = "channelDeviceId", description = "通道国标编号", required = true)
@ResponseBody @ResponseBody
@PostMapping("/region/remove") @PostMapping("/play")
public void removeFromRegion(@RequestBody UpdateCommonChannelToRegion params){ public DeferredResult<WVPResult<StreamContent>> play(HttpServletRequest request, String channelDeviceId){
assert params.getCommonGbCivilCode() != null || !params.getCommonGbIds().isEmpty(); logger.info("[播放通道] channelDeviceId{} ", channelDeviceId);
commonGbChannelService.removeFromRegion(params); assert !ObjectUtils.isEmpty(channelDeviceId);
CommonGbChannel channel = commonGbChannelService.getChannel(channelDeviceId);
assert channel != null;
DeferredResult<WVPResult<StreamContent>> result = new DeferredResult<>(userSetting.getPlayTimeout().longValue());
result.onTimeout(()->{
logger.info("[播放通道] 超时 channelDeviceId{} ", channelDeviceId);
// 释放rtpserver
WVPResult<StreamContent> wvpResult = new WVPResult<>();
wvpResult.setCode(ErrorCode.ERROR100.getCode());
wvpResult.setMsg("播放通道超时");
result.setResult(wvpResult);
commonGbChannelService.stopPlay(channel, null);
});
commonGbChannelService.startPlay(channel, (callbackChannel, code, message, streamInfo) -> {
if (code == ErrorCode.SUCCESS.getCode()) {
WVPResult<StreamContent> wvpResult = new WVPResult<>();
wvpResult.setCode(ErrorCode.SUCCESS.getCode());
wvpResult.setMsg(ErrorCode.SUCCESS.getMsg());
if (streamInfo != null) {
if (userSetting.getUseSourceIpAsStreamIp()) {
streamInfo = streamInfo.clone();//深拷贝
String host;
try {
URL url = new URL(request.getRequestURL().toString());
host = url.getHost();
} catch (MalformedURLException e) {
host = request.getLocalAddr();
}
streamInfo.channgeStreamIp(host);
}
wvpResult.setData(new StreamContent(streamInfo));
}
}else {
WVPResult<StreamContent> wvpResult = new WVPResult<>();
wvpResult.setCode(code);
wvpResult.setMsg(message);
result.setResult(wvpResult);
commonGbChannelService.stopPlay(channel, null);
}
});
return result;
} }
@Operation(summary = "为通道添加分组")
@ResponseBody @Operation(summary = "停止播放通道")
@PostMapping("/region/update") @Parameter(name = "channelDeviceId", description = "通道国标编号", required = true)
public void updateChannelToRegion(@RequestBody UpdateCommonChannelToRegion params){ @GetMapping("/stopPlay")
assert params.getCommonGbCivilCode() != null; public void playStop(String channelDeviceId) {
assert !params.getCommonGbIds().isEmpty();
commonGbChannelService.updateChannelToRegion(params); logger.info("[停止播放通道] channelDeviceId{} ", channelDeviceId);
assert !ObjectUtils.isEmpty(channelDeviceId);
CommonGbChannel channel = commonGbChannelService.getChannel(channelDeviceId);
assert channel != null;
DeferredResult<WVPResult<StreamContent>> result = new DeferredResult<>();
result.onTimeout(()->{
logger.info("[停止播放通道] 超时 channelDeviceId{} ", channelDeviceId);
// 释放rtpserver
WVPResult wvpResult = new WVPResult<>();
wvpResult.setCode(ErrorCode.ERROR100.getCode());
wvpResult.setMsg("停止播放通道超时");
result.setResult(wvpResult);
commonGbChannelService.stopPlay(channel, null);
});
commonGbChannelService.stopPlay(channel, (commonGbChannel, code, message, streamInfo) -> {
WVPResult wvpResult = new WVPResult();
wvpResult.setCode(code);
wvpResult.setMsg(message);
result.setResult(wvpResult);
});
} }
// 将通道共享到上级平台
// 从上级平台共享中移除通道
} }

View File

@ -164,27 +164,7 @@ public class PlayController {
if (deviceId == null || channelId == null) { if (deviceId == null || channelId == null) {
throw new ControllerException(ErrorCode.ERROR400); throw new ControllerException(ErrorCode.ERROR400);
} }
playService.stop(deviceId, channelId);
Device device = storager.queryVideoDevice(deviceId);
if (device == null) {
throw new ControllerException(ErrorCode.ERROR100.getCode(), "设备[" + deviceId + "]不存在");
}
InviteInfo inviteInfo = inviteStreamService.getInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId);
if (inviteInfo == null) {
throw new ControllerException(ErrorCode.ERROR100.getCode(), "点播未找到");
}
if (InviteSessionStatus.ok == inviteInfo.getStatus()) {
try {
logger.info("[停止点播] {}/{}", device.getDeviceId(), channelId);
cmder.streamByeCmd(device, channelId, inviteInfo.getStream(), null, null);
} catch (InvalidArgumentException | SipException | ParseException | SsrcTransactionNotFoundException e) {
logger.error("[命令发送失败] 停止点播, 发送BYE: {}", e.getMessage());
throw new ControllerException(ErrorCode.ERROR100.getCode(), "命令发送失败: " + e.getMessage());
}
}
inviteStreamService.removeInviteInfoByDeviceAndChannel(InviteSessionType.PLAY, deviceId, channelId);
storager.stopPlay(deviceId, channelId);
JSONObject json = new JSONObject(); JSONObject json = new JSONObject();
json.put("deviceId", deviceId); json.put("deviceId", deviceId);