Merge branch '648540858:wvp-28181-2.0' into wvp-28181-2.0
commit
22e1d92a9d
|
@ -0,0 +1,91 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
######################################################
|
||||||
|
# Copyright 2019 Pham Ngoc Hoai
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
#
|
||||||
|
# Repo: https://github.com/tyrion9/spring-boot-startup-script
|
||||||
|
#
|
||||||
|
######### PARAM ######################################
|
||||||
|
|
||||||
|
JAVA_OPT=-Xmx1024m
|
||||||
|
JARFILE=`ls -1r *.jar 2>/dev/null | head -n 1`
|
||||||
|
PID_FILE=pid.file
|
||||||
|
RUNNING=N
|
||||||
|
PWD=`pwd`
|
||||||
|
|
||||||
|
######### DO NOT MODIFY ########
|
||||||
|
|
||||||
|
if [ -f $PID_FILE ]; then
|
||||||
|
PID=`cat $PID_FILE`
|
||||||
|
if [ ! -z "$PID" ] && kill -0 $PID 2>/dev/null; then
|
||||||
|
RUNNING=Y
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
start()
|
||||||
|
{
|
||||||
|
if [ $RUNNING == "Y" ]; then
|
||||||
|
echo "Application already started"
|
||||||
|
else
|
||||||
|
if [ -z "$JARFILE" ]
|
||||||
|
then
|
||||||
|
echo "ERROR: jar file not found"
|
||||||
|
else
|
||||||
|
nohup java $JAVA_OPT -Djava.security.egd=file:/dev/./urandom -jar $PWD/$JARFILE > nohup.out 2>&1 &
|
||||||
|
echo $! > $PID_FILE
|
||||||
|
echo "Application $JARFILE starting..."
|
||||||
|
tail -f nohup.out
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
stop()
|
||||||
|
{
|
||||||
|
if [ $RUNNING == "Y" ]; then
|
||||||
|
kill -9 $PID
|
||||||
|
rm -f $PID_FILE
|
||||||
|
echo "Application stopped"
|
||||||
|
else
|
||||||
|
echo "Application not running"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
restart()
|
||||||
|
{
|
||||||
|
stop
|
||||||
|
start
|
||||||
|
}
|
||||||
|
|
||||||
|
case "$1" in
|
||||||
|
|
||||||
|
'start')
|
||||||
|
start
|
||||||
|
;;
|
||||||
|
|
||||||
|
'stop')
|
||||||
|
stop
|
||||||
|
;;
|
||||||
|
|
||||||
|
'restart')
|
||||||
|
restart
|
||||||
|
;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
echo "Usage: $0 { start | stop | restart }"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
exit 0
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
package com.genersoft.iot.vmp.conf;
|
package com.genersoft.iot.vmp.conf;
|
||||||
|
|
||||||
import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.transmit.event.request.impl.message.response.cmd.CatalogResponseMessageHandler;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
|
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
|
||||||
|
@ -18,6 +21,8 @@ import java.util.concurrent.ScheduledFuture;
|
||||||
@Component
|
@Component
|
||||||
public class DynamicTask {
|
public class DynamicTask {
|
||||||
|
|
||||||
|
private Logger logger = LoggerFactory.getLogger(DynamicTask.class);
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
|
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
|
||||||
|
|
||||||
|
@ -26,7 +31,12 @@ public class DynamicTask {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
|
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
|
||||||
return new ThreadPoolTaskScheduler();
|
ThreadPoolTaskScheduler schedulerPool = new ThreadPoolTaskScheduler();
|
||||||
|
schedulerPool.setPoolSize(300);
|
||||||
|
schedulerPool.setWaitForTasksToCompleteOnShutdown(true);
|
||||||
|
schedulerPool.setAwaitTerminationSeconds(10);
|
||||||
|
return schedulerPool;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,11 +47,24 @@ public class DynamicTask {
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
public void startCron(String key, Runnable task, int cycleForCatalog) {
|
public void startCron(String key, Runnable task, int cycleForCatalog) {
|
||||||
stop(key);
|
ScheduledFuture future = futureMap.get(key);
|
||||||
|
if (future != null) {
|
||||||
|
if (future.isCancelled()) {
|
||||||
|
logger.info("任务【{}】已存在但是关闭状态!!!", key);
|
||||||
|
} else {
|
||||||
|
logger.info("任务【{}】已存在且已启动!!!", key);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
// scheduleWithFixedDelay 必须等待上一个任务结束才开始计时period, cycleForCatalog表示执行的间隔
|
// scheduleWithFixedDelay 必须等待上一个任务结束才开始计时period, cycleForCatalog表示执行的间隔
|
||||||
ScheduledFuture future = threadPoolTaskScheduler.scheduleWithFixedDelay(task, cycleForCatalog * 1000L);
|
future = threadPoolTaskScheduler.scheduleAtFixedRate(task, cycleForCatalog * 1000L);
|
||||||
futureMap.put(key, future);
|
if (future != null){
|
||||||
runnableMap.put(key, task);
|
futureMap.put(key, future);
|
||||||
|
runnableMap.put(key, task);
|
||||||
|
logger.info("任务【{}】启动成功!!!", key);
|
||||||
|
}else {
|
||||||
|
logger.info("任务【{}】启动失败!!!", key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -54,9 +77,25 @@ public class DynamicTask {
|
||||||
public void startDelay(String key, Runnable task, int delay) {
|
public void startDelay(String key, Runnable task, int delay) {
|
||||||
stop(key);
|
stop(key);
|
||||||
Date starTime = new Date(System.currentTimeMillis() + delay);
|
Date starTime = new Date(System.currentTimeMillis() + delay);
|
||||||
|
|
||||||
|
ScheduledFuture future = futureMap.get(key);
|
||||||
|
if (future != null) {
|
||||||
|
if (future.isCancelled()) {
|
||||||
|
logger.info("任务【{}】已存在但是关闭状态!!!", key);
|
||||||
|
} else {
|
||||||
|
logger.info("任务【{}】已存在且已启动!!!", key);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
// scheduleWithFixedDelay 必须等待上一个任务结束才开始计时period, cycleForCatalog表示执行的间隔
|
// scheduleWithFixedDelay 必须等待上一个任务结束才开始计时period, cycleForCatalog表示执行的间隔
|
||||||
ScheduledFuture future = threadPoolTaskScheduler.schedule(task, starTime);
|
future = threadPoolTaskScheduler.schedule(task, starTime);
|
||||||
futureMap.put(key, future);
|
if (future != null){
|
||||||
|
futureMap.put(key, future);
|
||||||
|
runnableMap.put(key, task);
|
||||||
|
logger.info("任务【{}】启动成功!!!", key);
|
||||||
|
}else {
|
||||||
|
logger.info("任务【{}】启动失败!!!", key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void stop(String key) {
|
public void stop(String key) {
|
||||||
|
@ -78,4 +117,7 @@ public class DynamicTask {
|
||||||
return futureMap.keySet();
|
return futureMap.keySet();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Runnable get(String key) {
|
||||||
|
return runnableMap.get(key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,6 +48,7 @@ public class SipLayer{
|
||||||
properties.setProperty("javax.sip.STACK_NAME", "GB28181_SIP");
|
properties.setProperty("javax.sip.STACK_NAME", "GB28181_SIP");
|
||||||
properties.setProperty("javax.sip.IP_ADDRESS", sipConfig.getMonitorIp());
|
properties.setProperty("javax.sip.IP_ADDRESS", sipConfig.getMonitorIp());
|
||||||
properties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT", "true");
|
properties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT", "true");
|
||||||
|
properties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true"); // 接收所有notify请求,即使没有订阅
|
||||||
/**
|
/**
|
||||||
* sip_server_log.log 和 sip_debug_log.log public static final int TRACE_NONE =
|
* sip_server_log.log 和 sip_debug_log.log public static final int TRACE_NONE =
|
||||||
* 0; public static final int TRACE_MESSAGES = 16; public static final int
|
* 0; public static final int TRACE_MESSAGES = 16; public static final int
|
||||||
|
|
|
@ -4,6 +4,7 @@ import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class CatalogData {
|
public class CatalogData {
|
||||||
|
private int sn; // 命令序列号
|
||||||
private int total;
|
private int total;
|
||||||
private List<DeviceChannel> channelList;
|
private List<DeviceChannel> channelList;
|
||||||
private Date lastTime;
|
private Date lastTime;
|
||||||
|
@ -15,6 +16,15 @@ public class CatalogData {
|
||||||
}
|
}
|
||||||
private CatalogDataStatus status;
|
private CatalogDataStatus status;
|
||||||
|
|
||||||
|
|
||||||
|
public int getSn() {
|
||||||
|
return sn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSn(int sn) {
|
||||||
|
this.sn = sn;
|
||||||
|
}
|
||||||
|
|
||||||
public int getTotal() {
|
public int getTotal() {
|
||||||
return total;
|
return total;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,12 @@
|
||||||
package com.genersoft.iot.vmp.gb28181.bean;
|
package com.genersoft.iot.vmp.gb28181.bean;
|
||||||
|
|
||||||
|
import com.genersoft.iot.vmp.common.VideoManagerConstants;
|
||||||
|
import com.genersoft.iot.vmp.conf.DynamicTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.task.impl.MobilePositionSubscribeHandlerTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommanderForPlatform;
|
||||||
|
import com.genersoft.iot.vmp.storager.IRedisCatchStorage;
|
||||||
|
import com.genersoft.iot.vmp.storager.IVideoManagerStorage;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Component;
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
@ -9,12 +16,32 @@ import java.util.concurrent.ConcurrentHashMap;
|
||||||
@Component
|
@Component
|
||||||
public class SubscribeHolder {
|
public class SubscribeHolder {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private DynamicTask dynamicTask;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private IRedisCatchStorage redisCatchStorage;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private ISIPCommanderForPlatform sipCommanderForPlatform;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private IVideoManagerStorage storager;
|
||||||
|
|
||||||
|
private final String taskOverduePrefix = "subscribe_overdue_";
|
||||||
|
|
||||||
private static ConcurrentHashMap<String, SubscribeInfo> catalogMap = new ConcurrentHashMap<>();
|
private static ConcurrentHashMap<String, SubscribeInfo> catalogMap = new ConcurrentHashMap<>();
|
||||||
private static ConcurrentHashMap<String, SubscribeInfo> mobilePositionMap = new ConcurrentHashMap<>();
|
private static ConcurrentHashMap<String, SubscribeInfo> mobilePositionMap = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
|
|
||||||
public void putCatalogSubscribe(String platformId, SubscribeInfo subscribeInfo) {
|
public void putCatalogSubscribe(String platformId, SubscribeInfo subscribeInfo) {
|
||||||
catalogMap.put(platformId, subscribeInfo);
|
catalogMap.put(platformId, subscribeInfo);
|
||||||
|
// 添加订阅到期
|
||||||
|
String taskOverdueKey = taskOverduePrefix + "catalog_" + platformId;
|
||||||
|
dynamicTask.stop(taskOverdueKey);
|
||||||
|
// 添加任务处理订阅过期
|
||||||
|
dynamicTask.startDelay(taskOverdueKey, () -> removeCatalogSubscribe(subscribeInfo.getId()),
|
||||||
|
subscribeInfo.getExpires() * 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SubscribeInfo getCatalogSubscribe(String platformId) {
|
public SubscribeInfo getCatalogSubscribe(String platformId) {
|
||||||
|
@ -23,10 +50,24 @@ public class SubscribeHolder {
|
||||||
|
|
||||||
public void removeCatalogSubscribe(String platformId) {
|
public void removeCatalogSubscribe(String platformId) {
|
||||||
catalogMap.remove(platformId);
|
catalogMap.remove(platformId);
|
||||||
|
String taskOverdueKey = taskOverduePrefix + "catalog_" + platformId;
|
||||||
|
// 添加任务处理订阅过期
|
||||||
|
dynamicTask.stop(taskOverdueKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void putMobilePositionSubscribe(String platformId, SubscribeInfo subscribeInfo) {
|
public void putMobilePositionSubscribe(String platformId, SubscribeInfo subscribeInfo) {
|
||||||
mobilePositionMap.put(platformId, subscribeInfo);
|
mobilePositionMap.put(platformId, subscribeInfo);
|
||||||
|
String key = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + "MobilePosition_" + platformId;
|
||||||
|
// 添加任务处理GPS定时推送
|
||||||
|
dynamicTask.startCron(key, new MobilePositionSubscribeHandlerTask(redisCatchStorage, sipCommanderForPlatform, storager, platformId, subscribeInfo.getSn(), key, this), subscribeInfo.getGpsInterval());
|
||||||
|
String taskOverdueKey = taskOverduePrefix + "MobilePosition_" + platformId;
|
||||||
|
dynamicTask.stop(taskOverdueKey);
|
||||||
|
// 添加任务处理订阅过期
|
||||||
|
dynamicTask.startDelay(taskOverdueKey, () -> {
|
||||||
|
System.out.println("订阅过期");
|
||||||
|
removeMobilePositionSubscribe(subscribeInfo.getId());
|
||||||
|
},
|
||||||
|
subscribeInfo.getExpires() * 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SubscribeInfo getMobilePositionSubscribe(String platformId) {
|
public SubscribeInfo getMobilePositionSubscribe(String platformId) {
|
||||||
|
@ -35,6 +76,12 @@ public class SubscribeHolder {
|
||||||
|
|
||||||
public void removeMobilePositionSubscribe(String platformId) {
|
public void removeMobilePositionSubscribe(String platformId) {
|
||||||
mobilePositionMap.remove(platformId);
|
mobilePositionMap.remove(platformId);
|
||||||
|
String key = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + "MobilePosition_" + platformId;
|
||||||
|
// 结束任务处理GPS定时推送
|
||||||
|
dynamicTask.stop(key);
|
||||||
|
String taskOverdueKey = taskOverduePrefix + "MobilePosition_" + platformId;
|
||||||
|
// 添加任务处理订阅过期
|
||||||
|
dynamicTask.stop(taskOverdueKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<String> getAllCatalogSubscribePlatform() {
|
public List<String> getAllCatalogSubscribePlatform() {
|
||||||
|
@ -48,7 +95,7 @@ public class SubscribeHolder {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void removeAllSubscribe(String platformId) {
|
public void removeAllSubscribe(String platformId) {
|
||||||
mobilePositionMap.remove(platformId);
|
removeMobilePositionSubscribe(platformId);
|
||||||
catalogMap.remove(platformId);
|
removeCatalogSubscribe(platformId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,14 @@ public class SubscribeInfo {
|
||||||
private ServerTransaction transaction;
|
private ServerTransaction transaction;
|
||||||
private Dialog dialog;
|
private Dialog dialog;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 以下为可选字段
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
private String sn;
|
||||||
|
private int gpsInterval;
|
||||||
|
|
||||||
|
|
||||||
public String getId() {
|
public String getId() {
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
@ -88,4 +96,20 @@ public class SubscribeInfo {
|
||||||
public void setDialog(Dialog dialog) {
|
public void setDialog(Dialog dialog) {
|
||||||
this.dialog = dialog;
|
this.dialog = dialog;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getSn() {
|
||||||
|
return sn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSn(String sn) {
|
||||||
|
this.sn = sn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getGpsInterval() {
|
||||||
|
return gpsInterval;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGpsInterval(int gpsInterval) {
|
||||||
|
this.gpsInterval = gpsInterval;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,6 +54,7 @@ public class OnlineEventListener implements ApplicationListener<OnlineEvent> {
|
||||||
@Autowired
|
@Autowired
|
||||||
private SIPCommander cmder;
|
private SIPCommander cmder;
|
||||||
|
|
||||||
|
|
||||||
private SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
|
private SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -76,7 +77,7 @@ public class OnlineEventListener implements ApplicationListener<OnlineEvent> {
|
||||||
if (deviceInStore == null) { //第一次上线
|
if (deviceInStore == null) { //第一次上线
|
||||||
logger.info("[{}] 首次注册,查询设备信息以及通道信息", device.getDeviceId());
|
logger.info("[{}] 首次注册,查询设备信息以及通道信息", device.getDeviceId());
|
||||||
cmder.deviceInfoQuery(device);
|
cmder.deviceInfoQuery(device);
|
||||||
cmder.catalogQuery(device, null);
|
deviceService.sync(device);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
// 设备主动发送心跳触发的在线事件
|
// 设备主动发送心跳触发的在线事件
|
||||||
|
@ -99,7 +100,10 @@ public class OnlineEventListener implements ApplicationListener<OnlineEvent> {
|
||||||
storager.updateDevice(device);
|
storager.updateDevice(device);
|
||||||
// 上线添加订阅
|
// 上线添加订阅
|
||||||
if (device.getSubscribeCycleForCatalog() > 0) {
|
if (device.getSubscribeCycleForCatalog() > 0) {
|
||||||
|
// 查询在线设备那些开启了订阅,为设备开启定时的目录订阅
|
||||||
deviceService.addCatalogSubscribe(device);
|
deviceService.addCatalogSubscribe(device);
|
||||||
|
}
|
||||||
|
if (device.getSubscribeCycleForMobilePosition() > 0) {
|
||||||
deviceService.addMobilePositionSubscribe(device);
|
deviceService.addMobilePositionSubscribe(device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,50 +0,0 @@
|
||||||
package com.genersoft.iot.vmp.gb28181.event.subscribe;
|
|
||||||
|
|
||||||
import com.genersoft.iot.vmp.common.VideoManagerConstants;
|
|
||||||
import com.genersoft.iot.vmp.conf.DynamicTask;
|
|
||||||
import com.genersoft.iot.vmp.conf.RedisKeyExpirationEventMessageListener;
|
|
||||||
import com.genersoft.iot.vmp.conf.UserSetting;
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
|
||||||
import org.springframework.data.redis.connection.Message;
|
|
||||||
import org.springframework.data.redis.listener.RedisMessageListenerContainer;
|
|
||||||
import org.springframework.stereotype.Component;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 平台订阅到期事件
|
|
||||||
*/
|
|
||||||
@Component
|
|
||||||
public class SubscribeListenerForPlatform extends RedisKeyExpirationEventMessageListener {
|
|
||||||
|
|
||||||
private Logger logger = LoggerFactory.getLogger(SubscribeListenerForPlatform.class);
|
|
||||||
|
|
||||||
@Autowired
|
|
||||||
private UserSetting userSetting;
|
|
||||||
|
|
||||||
@Autowired
|
|
||||||
private DynamicTask dynamicTask;
|
|
||||||
|
|
||||||
public SubscribeListenerForPlatform(RedisMessageListenerContainer listenerContainer, UserSetting userSetting) {
|
|
||||||
super(listenerContainer, userSetting);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 监听失效的key
|
|
||||||
* @param message
|
|
||||||
* @param pattern
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public void onMessage(Message message, byte[] pattern) {
|
|
||||||
// 获取失效的key
|
|
||||||
String expiredKey = message.toString();
|
|
||||||
logger.debug(expiredKey);
|
|
||||||
// 订阅到期
|
|
||||||
String PLATFORM_KEEPLIVEKEY_PREFIX = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + userSetting.getServerId() + "_";
|
|
||||||
if (expiredKey.startsWith(PLATFORM_KEEPLIVEKEY_PREFIX)) {
|
|
||||||
// 取消定时任务
|
|
||||||
dynamicTask.stop(expiredKey);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -61,8 +61,6 @@ public class CatalogEventLister implements ApplicationListener<CatalogEvent> {
|
||||||
if (event.getPlatformId() != null) {
|
if (event.getPlatformId() != null) {
|
||||||
parentPlatform = storager.queryParentPlatByServerGBId(event.getPlatformId());
|
parentPlatform = storager.queryParentPlatByServerGBId(event.getPlatformId());
|
||||||
if (parentPlatform != null && !parentPlatform.isStatus())return;
|
if (parentPlatform != null && !parentPlatform.isStatus())return;
|
||||||
String key = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + userSetting.getServerId() + "_Catalog_" + event.getPlatformId();
|
|
||||||
// subscribe = redisCatchStorage.getSubscribe(key);
|
|
||||||
subscribe = subscribeHolder.getCatalogSubscribe(event.getPlatformId());
|
subscribe = subscribeHolder.getCatalogSubscribe(event.getPlatformId());
|
||||||
|
|
||||||
if (subscribe == null) {
|
if (subscribe == null) {
|
||||||
|
|
|
@ -26,28 +26,35 @@ public class CatalogDataCatch {
|
||||||
@Autowired
|
@Autowired
|
||||||
private IVideoManagerStorage storager;
|
private IVideoManagerStorage storager;
|
||||||
|
|
||||||
public void addReady(String key) {
|
public void addReady(Device device, int sn ) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(device.getDeviceId());
|
||||||
if (catalogData == null || catalogData.getStatus().equals(CatalogData.CatalogDataStatus.end)) {
|
if (catalogData == null || catalogData.getStatus().equals(CatalogData.CatalogDataStatus.end)) {
|
||||||
catalogData = new CatalogData();
|
catalogData = new CatalogData();
|
||||||
catalogData.setChannelList(new ArrayList<>());
|
catalogData.setChannelList(new ArrayList<>());
|
||||||
|
catalogData.setDevice(device);
|
||||||
|
catalogData.setSn(sn);
|
||||||
catalogData.setStatus(CatalogData.CatalogDataStatus.ready);
|
catalogData.setStatus(CatalogData.CatalogDataStatus.ready);
|
||||||
catalogData.setLastTime(new Date(System.currentTimeMillis()));
|
catalogData.setLastTime(new Date(System.currentTimeMillis()));
|
||||||
data.put(key, catalogData);
|
data.put(device.getDeviceId(), catalogData);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void put(String key, int total, Device device, List<DeviceChannel> deviceChannelList) {
|
public void put(String deviceId, int sn, int total, Device device, List<DeviceChannel> deviceChannelList) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData == null) {
|
if (catalogData == null) {
|
||||||
catalogData = new CatalogData();
|
catalogData = new CatalogData();
|
||||||
|
catalogData.setSn(sn);
|
||||||
catalogData.setTotal(total);
|
catalogData.setTotal(total);
|
||||||
catalogData.setDevice(device);
|
catalogData.setDevice(device);
|
||||||
catalogData.setChannelList(new ArrayList<>());
|
catalogData.setChannelList(new ArrayList<>());
|
||||||
catalogData.setStatus(CatalogData.CatalogDataStatus.runIng);
|
catalogData.setStatus(CatalogData.CatalogDataStatus.runIng);
|
||||||
catalogData.setLastTime(new Date(System.currentTimeMillis()));
|
catalogData.setLastTime(new Date(System.currentTimeMillis()));
|
||||||
data.put(key, catalogData);
|
data.put(deviceId, catalogData);
|
||||||
}else {
|
}else {
|
||||||
|
// 同一个设备的通道同步请求只考虑一个,其他的直接忽略
|
||||||
|
if (catalogData.getSn() != sn) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
catalogData.setTotal(total);
|
catalogData.setTotal(total);
|
||||||
catalogData.setDevice(device);
|
catalogData.setDevice(device);
|
||||||
catalogData.setStatus(CatalogData.CatalogDataStatus.runIng);
|
catalogData.setStatus(CatalogData.CatalogDataStatus.runIng);
|
||||||
|
@ -56,20 +63,20 @@ public class CatalogDataCatch {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<DeviceChannel> get(String key) {
|
public List<DeviceChannel> get(String deviceId) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData == null) return null;
|
if (catalogData == null) return null;
|
||||||
return catalogData.getChannelList();
|
return catalogData.getChannelList();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getTotal(String key) {
|
public int getTotal(String deviceId) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData == null) return 0;
|
if (catalogData == null) return 0;
|
||||||
return catalogData.getTotal();
|
return catalogData.getTotal();
|
||||||
}
|
}
|
||||||
|
|
||||||
public SyncStatus getSyncStatus(String key) {
|
public SyncStatus getSyncStatus(String deviceId) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData == null) return null;
|
if (catalogData == null) return null;
|
||||||
SyncStatus syncStatus = new SyncStatus();
|
SyncStatus syncStatus = new SyncStatus();
|
||||||
syncStatus.setCurrent(catalogData.getChannelList().size());
|
syncStatus.setCurrent(catalogData.getChannelList().size());
|
||||||
|
@ -78,10 +85,6 @@ public class CatalogDataCatch {
|
||||||
return syncStatus;
|
return syncStatus;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void del(String key) {
|
|
||||||
data.remove(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Scheduled(fixedRate = 5 * 1000) //每5秒执行一次, 发现数据5秒未更新则移除数据并认为数据接收超时
|
@Scheduled(fixedRate = 5 * 1000) //每5秒执行一次, 发现数据5秒未更新则移除数据并认为数据接收超时
|
||||||
private void timerTask(){
|
private void timerTask(){
|
||||||
Set<String> keys = data.keySet();
|
Set<String> keys = data.keySet();
|
||||||
|
@ -92,23 +95,30 @@ public class CatalogDataCatch {
|
||||||
Calendar calendarBefore30S = Calendar.getInstance();
|
Calendar calendarBefore30S = Calendar.getInstance();
|
||||||
calendarBefore30S.setTime(new Date());
|
calendarBefore30S.setTime(new Date());
|
||||||
calendarBefore30S.set(Calendar.SECOND, calendarBefore30S.get(Calendar.SECOND) - 30);
|
calendarBefore30S.set(Calendar.SECOND, calendarBefore30S.get(Calendar.SECOND) - 30);
|
||||||
for (String key : keys) {
|
for (String deviceId : keys) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData.getLastTime().before(calendarBefore5S.getTime())) { // 超过五秒收不到消息任务超时, 只更新这一部分数据
|
if ( catalogData.getLastTime().before(calendarBefore5S.getTime())) { // 超过五秒收不到消息任务超时, 只更新这一部分数据
|
||||||
storager.resetChannels(catalogData.getDevice().getDeviceId(), catalogData.getChannelList());
|
if (catalogData.getStatus().equals(CatalogData.CatalogDataStatus.runIng)) {
|
||||||
String errorMsg = "更新成功,共" + catalogData.getTotal() + "条,已更新" + catalogData.getChannelList().size() + "条";
|
storager.resetChannels(catalogData.getDevice().getDeviceId(), catalogData.getChannelList());
|
||||||
|
if (catalogData.getTotal() != catalogData.getChannelList().size()) {
|
||||||
|
String errorMsg = "更新成功,共" + catalogData.getTotal() + "条,已更新" + catalogData.getChannelList().size() + "条";
|
||||||
|
catalogData.setErrorMsg(errorMsg);
|
||||||
|
}
|
||||||
|
}else if (catalogData.getStatus().equals(CatalogData.CatalogDataStatus.ready)) {
|
||||||
|
String errorMsg = "同步失败,等待回复超时";
|
||||||
|
catalogData.setErrorMsg(errorMsg);
|
||||||
|
}
|
||||||
catalogData.setStatus(CatalogData.CatalogDataStatus.end);
|
catalogData.setStatus(CatalogData.CatalogDataStatus.end);
|
||||||
catalogData.setErrorMsg(errorMsg);
|
|
||||||
}
|
}
|
||||||
if (catalogData.getLastTime().before(calendarBefore30S.getTime())) { // 超过三十秒,如果标记为end则删除
|
if (catalogData.getStatus().equals(CatalogData.CatalogDataStatus.end) && catalogData.getLastTime().before(calendarBefore30S.getTime())) { // 超过三十秒,如果标记为end则删除
|
||||||
data.remove(key);
|
data.remove(deviceId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void setChannelSyncEnd(String key, String errorMsg) {
|
public void setChannelSyncEnd(String deviceId, String errorMsg) {
|
||||||
CatalogData catalogData = data.get(key);
|
CatalogData catalogData = data.get(deviceId);
|
||||||
if (catalogData == null)return;
|
if (catalogData == null)return;
|
||||||
catalogData.setStatus(CatalogData.CatalogDataStatus.end);
|
catalogData.setStatus(CatalogData.CatalogDataStatus.end);
|
||||||
catalogData.setErrorMsg(errorMsg);
|
catalogData.setErrorMsg(errorMsg);
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
package com.genersoft.iot.vmp.gb28181.task;
|
package com.genersoft.iot.vmp.gb28181.task;
|
||||||
|
|
||||||
|
import javax.sip.DialogState;
|
||||||
|
|
||||||
public interface ISubscribeTask extends Runnable{
|
public interface ISubscribeTask extends Runnable{
|
||||||
void stop();
|
void stop();
|
||||||
|
|
||||||
|
DialogState getDialogState();
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommander;
|
import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommander;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.scheduling.annotation.Async;
|
||||||
|
|
||||||
import javax.sip.Dialog;
|
import javax.sip.Dialog;
|
||||||
import javax.sip.DialogState;
|
import javax.sip.DialogState;
|
||||||
|
@ -72,4 +73,10 @@ public class CatalogSubscribeTask implements ISubscribeTask {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DialogState getDialogState() {
|
||||||
|
if (dialog == null) return null;
|
||||||
|
return dialog.getState();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
package com.genersoft.iot.vmp.gb28181.task.impl;
|
package com.genersoft.iot.vmp.gb28181.task.impl;
|
||||||
|
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.GbStream;
|
import com.genersoft.iot.vmp.gb28181.bean.*;
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.ParentPlatform;
|
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.SubscribeHolder;
|
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.SubscribeInfo;
|
|
||||||
import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommanderForPlatform;
|
import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommanderForPlatform;
|
||||||
import com.genersoft.iot.vmp.service.bean.GPSMsgInfo;
|
import com.genersoft.iot.vmp.service.bean.GPSMsgInfo;
|
||||||
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 org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.scheduling.annotation.Async;
|
||||||
|
|
||||||
import java.text.SimpleDateFormat;
|
import javax.sip.DialogState;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -18,20 +18,21 @@ import java.util.List;
|
||||||
*/
|
*/
|
||||||
public class MobilePositionSubscribeHandlerTask implements ISubscribeTask {
|
public class MobilePositionSubscribeHandlerTask implements ISubscribeTask {
|
||||||
|
|
||||||
|
private Logger logger = LoggerFactory.getLogger(MobilePositionSubscribeHandlerTask.class);
|
||||||
|
|
||||||
private IRedisCatchStorage redisCatchStorage;
|
private IRedisCatchStorage redisCatchStorage;
|
||||||
private IVideoManagerStorage storager;
|
private IVideoManagerStorage storager;
|
||||||
private ISIPCommanderForPlatform sipCommanderForPlatform;
|
private ISIPCommanderForPlatform sipCommanderForPlatform;
|
||||||
private SubscribeHolder subscribeHolder;
|
private SubscribeHolder subscribeHolder;
|
||||||
private String platformId;
|
private ParentPlatform platform;
|
||||||
private String sn;
|
private String sn;
|
||||||
private String key;
|
private String key;
|
||||||
|
|
||||||
private final SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
|
|
||||||
|
|
||||||
public MobilePositionSubscribeHandlerTask(IRedisCatchStorage redisCatchStorage, ISIPCommanderForPlatform sipCommanderForPlatform, IVideoManagerStorage storager, String platformId, String sn, String key, SubscribeHolder subscribeInfo) {
|
public MobilePositionSubscribeHandlerTask(IRedisCatchStorage redisCatchStorage, ISIPCommanderForPlatform sipCommanderForPlatform, IVideoManagerStorage storager, String platformId, String sn, String key, SubscribeHolder subscribeInfo) {
|
||||||
|
System.out.println("MobilePositionSubscribeHandlerTask 初始化");
|
||||||
this.redisCatchStorage = redisCatchStorage;
|
this.redisCatchStorage = redisCatchStorage;
|
||||||
this.storager = storager;
|
this.storager = storager;
|
||||||
this.platformId = platformId;
|
this.platform = storager.queryParentPlatByServerGBId(platformId);
|
||||||
this.sn = sn;
|
this.sn = sn;
|
||||||
this.key = key;
|
this.key = key;
|
||||||
this.sipCommanderForPlatform = sipCommanderForPlatform;
|
this.sipCommanderForPlatform = sipCommanderForPlatform;
|
||||||
|
@ -41,30 +42,31 @@ public class MobilePositionSubscribeHandlerTask implements ISubscribeTask {
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
|
|
||||||
SubscribeInfo subscribe = subscribeHolder.getMobilePositionSubscribe(platformId);
|
if (platform == null) return;
|
||||||
|
SubscribeInfo subscribe = subscribeHolder.getMobilePositionSubscribe(platform.getServerGBId());
|
||||||
if (subscribe != null) {
|
if (subscribe != null) {
|
||||||
ParentPlatform parentPlatform = storager.queryParentPlatByServerGBId(platformId);
|
|
||||||
if (parentPlatform == null || parentPlatform.isStatus()) {
|
// if (!parentPlatform.isStatus()) {
|
||||||
// TODO 暂时只处理视频流的回复,后续增加对国标设备的支持
|
// logger.info("发送订阅时发现平台已经离线:{}", platformId);
|
||||||
List<GbStream> gbStreams = storager.queryGbStreamListInPlatform(platformId);
|
// return;
|
||||||
if (gbStreams.size() > 0) {
|
// }
|
||||||
for (GbStream gbStream : gbStreams) {
|
// TODO 暂时只处理视频流的回复,后续增加对国标设备的支持
|
||||||
String gbId = gbStream.getGbId();
|
List<GbStream> gbStreams = storager.queryGbStreamListInPlatform(platform.getServerGBId());
|
||||||
GPSMsgInfo gpsMsgInfo = redisCatchStorage.getGpsMsgInfo(gbId);
|
if (gbStreams.size() == 0) {
|
||||||
if (gpsMsgInfo != null) {
|
logger.info("发送订阅时发现平台已经没有关联的直播流:{}", platform.getServerGBId());
|
||||||
// 发送GPS消息
|
return;
|
||||||
sipCommanderForPlatform.sendNotifyMobilePosition(parentPlatform, gpsMsgInfo, subscribe);
|
}
|
||||||
}else {
|
for (GbStream gbStream : gbStreams) {
|
||||||
// 没有在redis找到新的消息就使用数据库的消息
|
String gbId = gbStream.getGbId();
|
||||||
gpsMsgInfo = new GPSMsgInfo();
|
GPSMsgInfo gpsMsgInfo = redisCatchStorage.getGpsMsgInfo(gbId);
|
||||||
gpsMsgInfo.setId(gbId);
|
if (gpsMsgInfo != null) { // 无最新位置不发送
|
||||||
gpsMsgInfo.setLat(gbStream.getLongitude());
|
logger.info("无最新位置不发送");
|
||||||
gpsMsgInfo.setLng(gbStream.getLongitude());
|
// 经纬度都为0不发送
|
||||||
// 发送GPS消息
|
if (gpsMsgInfo.getLng() == 0 && gpsMsgInfo.getLat() == 0) {
|
||||||
sipCommanderForPlatform.sendNotifyMobilePosition(parentPlatform, gpsMsgInfo, subscribe);
|
continue;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
// 发送GPS消息
|
||||||
|
sipCommanderForPlatform.sendNotifyMobilePosition(platform, gpsMsgInfo, subscribe);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -74,4 +76,9 @@ public class MobilePositionSubscribeHandlerTask implements ISubscribeTask {
|
||||||
public void stop() {
|
public void stop() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DialogState getDialogState() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,10 +6,13 @@ import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommander;
|
||||||
import org.dom4j.Element;
|
import org.dom4j.Element;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.scheduling.annotation.Async;
|
||||||
|
|
||||||
import javax.sip.Dialog;
|
import javax.sip.Dialog;
|
||||||
import javax.sip.DialogState;
|
import javax.sip.DialogState;
|
||||||
import javax.sip.ResponseEvent;
|
import javax.sip.ResponseEvent;
|
||||||
|
import java.util.Timer;
|
||||||
|
import java.util.TimerTask;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 移动位置订阅的定时更新
|
* 移动位置订阅的定时更新
|
||||||
|
@ -20,6 +23,8 @@ public class MobilePositionSubscribeTask implements ISubscribeTask {
|
||||||
private ISIPCommander sipCommander;
|
private ISIPCommander sipCommander;
|
||||||
private Dialog dialog;
|
private Dialog dialog;
|
||||||
|
|
||||||
|
private Timer timer ;
|
||||||
|
|
||||||
public MobilePositionSubscribeTask(Device device, ISIPCommander sipCommander) {
|
public MobilePositionSubscribeTask(Device device, ISIPCommander sipCommander) {
|
||||||
this.device = device;
|
this.device = device;
|
||||||
this.sipCommander = sipCommander;
|
this.sipCommander = sipCommander;
|
||||||
|
@ -27,10 +32,14 @@ public class MobilePositionSubscribeTask implements ISubscribeTask {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
|
if (timer != null ) {
|
||||||
|
timer.cancel();
|
||||||
|
timer = null;
|
||||||
|
}
|
||||||
sipCommander.mobilePositionSubscribe(device, dialog, eventResult -> {
|
sipCommander.mobilePositionSubscribe(device, dialog, eventResult -> {
|
||||||
if (eventResult.dialog != null || eventResult.dialog.getState().equals(DialogState.CONFIRMED)) {
|
// if (eventResult.dialog != null || eventResult.dialog.getState().equals(DialogState.CONFIRMED)) {
|
||||||
dialog = eventResult.dialog;
|
// dialog = eventResult.dialog;
|
||||||
}
|
// }
|
||||||
ResponseEvent event = (ResponseEvent) eventResult.event;
|
ResponseEvent event = (ResponseEvent) eventResult.event;
|
||||||
if (event.getResponse().getRawContent() != null) {
|
if (event.getResponse().getRawContent() != null) {
|
||||||
// 成功
|
// 成功
|
||||||
|
@ -43,6 +52,13 @@ public class MobilePositionSubscribeTask implements ISubscribeTask {
|
||||||
dialog = null;
|
dialog = null;
|
||||||
// 失败
|
// 失败
|
||||||
logger.warn("[移动位置订阅]失败,信令发送失败: {}-{} ", device.getDeviceId(), eventResult.msg);
|
logger.warn("[移动位置订阅]失败,信令发送失败: {}-{} ", device.getDeviceId(), eventResult.msg);
|
||||||
|
timer = new Timer();
|
||||||
|
timer.schedule(new TimerTask() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
MobilePositionSubscribeTask.this.run();
|
||||||
|
}
|
||||||
|
}, 2000);
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -56,6 +72,10 @@ public class MobilePositionSubscribeTask implements ISubscribeTask {
|
||||||
* COMPLETED-> Completed Dialog状态-已完成
|
* COMPLETED-> Completed Dialog状态-已完成
|
||||||
* TERMINATED-> Terminated Dialog状态-终止
|
* TERMINATED-> Terminated Dialog状态-终止
|
||||||
*/
|
*/
|
||||||
|
if (timer != null ) {
|
||||||
|
timer.cancel();
|
||||||
|
timer = null;
|
||||||
|
}
|
||||||
if (dialog != null && dialog.getState().equals(DialogState.CONFIRMED)) {
|
if (dialog != null && dialog.getState().equals(DialogState.CONFIRMED)) {
|
||||||
logger.info("取消移动订阅时dialog状态为{}", dialog.getState());
|
logger.info("取消移动订阅时dialog状态为{}", dialog.getState());
|
||||||
device.setSubscribeCycleForMobilePosition(0);
|
device.setSubscribeCycleForMobilePosition(0);
|
||||||
|
@ -74,4 +94,9 @@ public class MobilePositionSubscribeTask implements ISubscribeTask {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@Override
|
||||||
|
public DialogState getDialogState() {
|
||||||
|
if (dialog == null) return null;
|
||||||
|
return dialog.getState();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -250,7 +250,7 @@ public interface ISIPCommander {
|
||||||
*
|
*
|
||||||
* @param device 视频设备
|
* @param device 视频设备
|
||||||
*/
|
*/
|
||||||
boolean catalogQuery(Device device, SipSubscribe.Event errorEvent);
|
boolean catalogQuery(Device device, int sn, SipSubscribe.Event errorEvent);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 查询录像信息
|
* 查询录像信息
|
||||||
|
|
|
@ -1208,14 +1208,14 @@ public class SIPCommander implements ISIPCommander {
|
||||||
* @param device 视频设备
|
* @param device 视频设备
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean catalogQuery(Device device, SipSubscribe.Event errorEvent) {
|
public boolean catalogQuery(Device device, int sn, SipSubscribe.Event errorEvent) {
|
||||||
try {
|
try {
|
||||||
StringBuffer catalogXml = new StringBuffer(200);
|
StringBuffer catalogXml = new StringBuffer(200);
|
||||||
String charset = device.getCharset();
|
String charset = device.getCharset();
|
||||||
catalogXml.append("<?xml version=\"1.0\" encoding=\"" + charset + "\"?>\r\n");
|
catalogXml.append("<?xml version=\"1.0\" encoding=\"" + charset + "\"?>\r\n");
|
||||||
catalogXml.append("<Query>\r\n");
|
catalogXml.append("<Query>\r\n");
|
||||||
catalogXml.append("<CmdType>Catalog</CmdType>\r\n");
|
catalogXml.append("<CmdType>Catalog</CmdType>\r\n");
|
||||||
catalogXml.append("<SN>" + (int)((Math.random()*9+1)*100000) + "</SN>\r\n");
|
catalogXml.append("<SN>" + sn + "</SN>\r\n");
|
||||||
catalogXml.append("<DeviceID>" + device.getDeviceId() + "</DeviceID>\r\n");
|
catalogXml.append("<DeviceID>" + device.getDeviceId() + "</DeviceID>\r\n");
|
||||||
catalogXml.append("</Query>\r\n");
|
catalogXml.append("</Query>\r\n");
|
||||||
|
|
||||||
|
@ -1566,17 +1566,28 @@ public class SIPCommander implements ISIPCommander {
|
||||||
cmdXml.append("<DeviceID>" + device.getDeviceId() + "</DeviceID>\r\n");
|
cmdXml.append("<DeviceID>" + device.getDeviceId() + "</DeviceID>\r\n");
|
||||||
cmdXml.append("</Query>\r\n");
|
cmdXml.append("</Query>\r\n");
|
||||||
|
|
||||||
String tm = Long.toString(System.currentTimeMillis());
|
|
||||||
|
|
||||||
CallIdHeader callIdHeader = device.getTransport().equals("TCP") ? tcpSipProvider.getNewCallId()
|
Request request;
|
||||||
: udpSipProvider.getNewCallId();
|
if (dialog != null) {
|
||||||
|
logger.info("发送目录订阅消息时 dialog的状态为: {}", dialog.getState());
|
||||||
|
request = dialog.createRequest(Request.SUBSCRIBE);
|
||||||
|
ContentTypeHeader contentTypeHeader = sipFactory.createHeaderFactory().createContentTypeHeader("Application", "MANSCDP+xml");
|
||||||
|
request.setContent(cmdXml.toString(), contentTypeHeader);
|
||||||
|
ExpiresHeader expireHeader = sipFactory.createHeaderFactory().createExpiresHeader(device.getSubscribeCycleForMobilePosition());
|
||||||
|
request.addHeader(expireHeader);
|
||||||
|
}else {
|
||||||
|
String tm = Long.toString(System.currentTimeMillis());
|
||||||
|
|
||||||
// 有效时间默认为60秒以上
|
CallIdHeader callIdHeader = device.getTransport().equals("TCP") ? tcpSipProvider.getNewCallId()
|
||||||
Request request = headerProvider.createSubscribeRequest(device, cmdXml.toString(), "z9hG4bK-viaPos-" + tm,
|
: udpSipProvider.getNewCallId();
|
||||||
"fromTagPos" + tm, null, device.getSubscribeCycleForCatalog(), "Catalog" ,
|
|
||||||
callIdHeader);
|
// 有效时间默认为60秒以上
|
||||||
|
request = headerProvider.createSubscribeRequest(device, cmdXml.toString(), "z9hG4bK-viaPos-" + tm,
|
||||||
|
"fromTagPos" + tm, null, device.getSubscribeCycleForCatalog(), "Catalog" ,
|
||||||
|
callIdHeader);
|
||||||
|
|
||||||
|
}
|
||||||
transmitRequest(device, request, errorEvent, okEvent);
|
transmitRequest(device, request, errorEvent, okEvent);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
} catch ( NumberFormatException | ParseException | InvalidArgumentException | SipException e) {
|
} catch ( NumberFormatException | ParseException | InvalidArgumentException | SipException e) {
|
||||||
|
|
|
@ -385,7 +385,7 @@ public class SIPCommanderFroPlatform implements ISIPCommanderForPlatform {
|
||||||
if (parentPlatform == null) {
|
if (parentPlatform == null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
logger.info("[发送 移动位置订阅] {}/{}->{},{}", parentPlatform.getServerGBId(), gpsMsgInfo.getId(), gpsMsgInfo.getLng(), gpsMsgInfo.getLat());
|
||||||
try {
|
try {
|
||||||
String characterSet = parentPlatform.getCharacterSet();
|
String characterSet = parentPlatform.getCharacterSet();
|
||||||
StringBuffer deviceStatusXml = new StringBuffer(600);
|
StringBuffer deviceStatusXml = new StringBuffer(600);
|
||||||
|
@ -405,7 +405,7 @@ public class SIPCommanderFroPlatform implements ISIPCommanderForPlatform {
|
||||||
CallIdHeader callIdHeader = parentPlatform.getTransport().equals("TCP") ? tcpSipProvider.getNewCallId()
|
CallIdHeader callIdHeader = parentPlatform.getTransport().equals("TCP") ? tcpSipProvider.getNewCallId()
|
||||||
: udpSipProvider.getNewCallId();
|
: udpSipProvider.getNewCallId();
|
||||||
callIdHeader.setCallId(subscribeInfo.getCallId());
|
callIdHeader.setCallId(subscribeInfo.getCallId());
|
||||||
logger.info("[发送Notify-MobilePosition] {}/{}->{},{}", parentPlatform.getServerGBId(), gpsMsgInfo.getId(), gpsMsgInfo.getLng(), gpsMsgInfo.getLat());
|
|
||||||
sendNotify(parentPlatform, deviceStatusXml.toString(), subscribeInfo, eventResult -> {
|
sendNotify(parentPlatform, deviceStatusXml.toString(), subscribeInfo, eventResult -> {
|
||||||
logger.error("发送NOTIFY通知消息失败。错误:{} {}", eventResult.statusCode, eventResult.msg);
|
logger.error("发送NOTIFY通知消息失败。错误:{} {}", eventResult.statusCode, eventResult.msg);
|
||||||
}, null);
|
}, null);
|
||||||
|
@ -459,7 +459,7 @@ public class SIPCommanderFroPlatform implements ISIPCommanderForPlatform {
|
||||||
// 设置编码, 防止中文乱码
|
// 设置编码, 防止中文乱码
|
||||||
messageFactory.setDefaultContentEncodingCharset(characterSet);
|
messageFactory.setDefaultContentEncodingCharset(characterSet);
|
||||||
Dialog dialog = subscribeInfo.getDialog();
|
Dialog dialog = subscribeInfo.getDialog();
|
||||||
if (dialog == null) return;
|
if (dialog == null || !dialog.getState().equals(DialogState.CONFIRMED)) return;
|
||||||
SIPRequest notifyRequest = (SIPRequest)dialog.createRequest(Request.NOTIFY);
|
SIPRequest notifyRequest = (SIPRequest)dialog.createRequest(Request.NOTIFY);
|
||||||
ContentTypeHeader contentTypeHeader = sipFactory.createHeaderFactory().createContentTypeHeader("Application", "MANSCDP+xml");
|
ContentTypeHeader contentTypeHeader = sipFactory.createHeaderFactory().createContentTypeHeader("Application", "MANSCDP+xml");
|
||||||
notifyRequest.setContent(catalogXmlContent, contentTypeHeader);
|
notifyRequest.setContent(catalogXmlContent, contentTypeHeader);
|
||||||
|
|
|
@ -146,7 +146,7 @@ public class NotifyRequestProcessor extends SIPRequestProcessorParent implements
|
||||||
} else {
|
} else {
|
||||||
mobilePosition.setAltitude(0.0);
|
mobilePosition.setAltitude(0.0);
|
||||||
}
|
}
|
||||||
logger.info("[收到Notify-MobilePosition]:{}/{}->{}.{}", mobilePosition.getDeviceId(), mobilePosition.getChannelId(),
|
logger.info("[收到 移动位置订阅]:{}/{}->{}.{}", mobilePosition.getDeviceId(), mobilePosition.getChannelId(),
|
||||||
mobilePosition.getLongitude(), mobilePosition.getLatitude());
|
mobilePosition.getLongitude(), mobilePosition.getLatitude());
|
||||||
mobilePosition.setReportSource("Mobile Position");
|
mobilePosition.setReportSource("Mobile Position");
|
||||||
BaiduPoint bp = GpsUtil.Wgs84ToBd09(String.valueOf(mobilePosition.getLongitude()), String.valueOf(mobilePosition.getLatitude()));
|
BaiduPoint bp = GpsUtil.Wgs84ToBd09(String.valueOf(mobilePosition.getLongitude()), String.valueOf(mobilePosition.getLatitude()));
|
||||||
|
@ -281,7 +281,7 @@ public class NotifyRequestProcessor extends SIPRequestProcessorParent implements
|
||||||
Element eventElement = itemDevice.element("Event");
|
Element eventElement = itemDevice.element("Event");
|
||||||
DeviceChannel channel = XmlUtil.channelContentHander(itemDevice);
|
DeviceChannel channel = XmlUtil.channelContentHander(itemDevice);
|
||||||
channel.setDeviceId(device.getDeviceId());
|
channel.setDeviceId(device.getDeviceId());
|
||||||
logger.info("[收到Notify-Catalog]:{}/{}", device.getDeviceId(), channel.getChannelId());
|
logger.info("[收到 目录订阅]:{}/{}", device.getDeviceId(), channel.getChannelId());
|
||||||
switch (eventElement.getText().toUpperCase()) {
|
switch (eventElement.getText().toUpperCase()) {
|
||||||
case CatalogEvent.ON: // 上线
|
case CatalogEvent.ON: // 上线
|
||||||
logger.info("收到来自设备【{}】的通道【{}】上线通知", device.getDeviceId(), channel.getChannelId());
|
logger.info("收到来自设备【{}】的通道【{}】上线通知", device.getDeviceId(), channel.getChannelId());
|
||||||
|
|
|
@ -149,8 +149,7 @@ public class SubscribeRequestProcessor extends SIPRequestProcessorParent impleme
|
||||||
subscribeInfo.setDialog(dialog);
|
subscribeInfo.setDialog(dialog);
|
||||||
}
|
}
|
||||||
String sn = XmlUtil.getText(rootElement, "SN");
|
String sn = XmlUtil.getText(rootElement, "SN");
|
||||||
String key = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + userSetting.getServerId() + "_MobilePosition_" + platformId;
|
logger.info("[回复 移动位置订阅]: {}", platformId);
|
||||||
logger.info("[notify-MobilePosition]: {}", platformId);
|
|
||||||
StringBuilder resultXml = new StringBuilder(200);
|
StringBuilder resultXml = new StringBuilder(200);
|
||||||
resultXml.append("<?xml version=\"1.0\" ?>\r\n")
|
resultXml.append("<?xml version=\"1.0\" ?>\r\n")
|
||||||
.append("<Response>\r\n")
|
.append("<Response>\r\n")
|
||||||
|
@ -161,14 +160,25 @@ public class SubscribeRequestProcessor extends SIPRequestProcessorParent impleme
|
||||||
.append("</Response>\r\n");
|
.append("</Response>\r\n");
|
||||||
|
|
||||||
if (subscribeInfo.getExpires() > 0) {
|
if (subscribeInfo.getExpires() > 0) {
|
||||||
if (subscribeHolder.getMobilePositionSubscribe(platformId) != null) {
|
|
||||||
dynamicTask.stop(key);
|
|
||||||
}
|
|
||||||
String interval = XmlUtil.getText(rootElement, "Interval"); // GPS上报时间间隔
|
String interval = XmlUtil.getText(rootElement, "Interval"); // GPS上报时间间隔
|
||||||
dynamicTask.startCron(key, new MobilePositionSubscribeHandlerTask(redisCatchStorage, sipCommanderForPlatform, storager, platformId, sn, key, subscribeHolder), Integer.parseInt(interval) -1 );
|
if (interval == null) {
|
||||||
|
subscribeInfo.setGpsInterval(5);
|
||||||
|
}else {
|
||||||
|
subscribeInfo.setGpsInterval(Integer.parseInt(interval));
|
||||||
|
}
|
||||||
|
|
||||||
|
subscribeInfo.setSn(sn);
|
||||||
subscribeHolder.putMobilePositionSubscribe(platformId, subscribeInfo);
|
subscribeHolder.putMobilePositionSubscribe(platformId, subscribeInfo);
|
||||||
|
// if (subscribeHolder.getMobilePositionSubscribe(platformId) == null ) {
|
||||||
|
// subscribeHolder.putMobilePositionSubscribe(platformId, subscribeInfo);
|
||||||
|
// }else {
|
||||||
|
// if (subscribeHolder.getMobilePositionSubscribe(platformId).getDialog() != null
|
||||||
|
// && subscribeHolder.getMobilePositionSubscribe(platformId).getDialog().getState() != null
|
||||||
|
// && !subscribeHolder.getMobilePositionSubscribe(platformId).getDialog().getState().equals(DialogState.CONFIRMED)) {
|
||||||
|
// subscribeHolder.putMobilePositionSubscribe(platformId, subscribeInfo);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
}else if (subscribeInfo.getExpires() == 0) {
|
}else if (subscribeInfo.getExpires() == 0) {
|
||||||
dynamicTask.stop(key);
|
|
||||||
subscribeHolder.removeMobilePositionSubscribe(platformId);
|
subscribeHolder.removeMobilePositionSubscribe(platformId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -202,8 +212,7 @@ public class SubscribeRequestProcessor extends SIPRequestProcessorParent impleme
|
||||||
subscribeInfo.setDialog(dialog);
|
subscribeInfo.setDialog(dialog);
|
||||||
}
|
}
|
||||||
String sn = XmlUtil.getText(rootElement, "SN");
|
String sn = XmlUtil.getText(rootElement, "SN");
|
||||||
String key = VideoManagerConstants.SIP_SUBSCRIBE_PREFIX + userSetting.getServerId() + "_Catalog_" + platformId;
|
logger.info("[回复 目录订阅]: {}/{}", platformId, deviceID);
|
||||||
logger.info("[notify-Catalog]: {}", platformId);
|
|
||||||
StringBuilder resultXml = new StringBuilder(200);
|
StringBuilder resultXml = new StringBuilder(200);
|
||||||
resultXml.append("<?xml version=\"1.0\" ?>\r\n")
|
resultXml.append("<?xml version=\"1.0\" ?>\r\n")
|
||||||
.append("<Response>\r\n")
|
.append("<Response>\r\n")
|
||||||
|
|
|
@ -86,23 +86,17 @@ public class CatalogResponseMessageHandler extends SIPRequestProcessorParent imp
|
||||||
rootElement = getRootElement(evt, device.getCharset());
|
rootElement = getRootElement(evt, device.getCharset());
|
||||||
Element deviceListElement = rootElement.element("DeviceList");
|
Element deviceListElement = rootElement.element("DeviceList");
|
||||||
Element sumNumElement = rootElement.element("SumNum");
|
Element sumNumElement = rootElement.element("SumNum");
|
||||||
if (sumNumElement == null || deviceListElement == null) {
|
Element snElement = rootElement.element("SN");
|
||||||
|
if (snElement == null || sumNumElement == null || deviceListElement == null) {
|
||||||
responseAck(evt, Response.BAD_REQUEST, "xml error");
|
responseAck(evt, Response.BAD_REQUEST, "xml error");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
int sumNum = Integer.parseInt(sumNumElement.getText());
|
int sumNum = Integer.parseInt(sumNumElement.getText());
|
||||||
|
|
||||||
if (sumNum == 0) {
|
if (sumNum == 0) {
|
||||||
// 数据已经完整接收
|
// 数据已经完整接收
|
||||||
storager.cleanChannelsForDevice(device.getDeviceId());
|
storager.cleanChannelsForDevice(device.getDeviceId());
|
||||||
RequestMessage msg = new RequestMessage();
|
catalogDataCatch.setChannelSyncEnd(device.getDeviceId(), null);
|
||||||
msg.setKey(key);
|
|
||||||
WVPResult<Object> result = new WVPResult<>();
|
|
||||||
result.setCode(0);
|
|
||||||
result.setData(device);
|
|
||||||
msg.setData(result);
|
|
||||||
result.setMsg("更新成功,共0条");
|
|
||||||
deferredResultHolder.invokeAllResult(msg);
|
|
||||||
catalogDataCatch.del(key);
|
|
||||||
}else {
|
}else {
|
||||||
Iterator<Element> deviceListIterator = deviceListElement.elementIterator();
|
Iterator<Element> deviceListIterator = deviceListElement.elementIterator();
|
||||||
if (deviceListIterator != null) {
|
if (deviceListIterator != null) {
|
||||||
|
@ -123,31 +117,22 @@ public class CatalogResponseMessageHandler extends SIPRequestProcessorParent imp
|
||||||
|
|
||||||
channelList.add(deviceChannel);
|
channelList.add(deviceChannel);
|
||||||
}
|
}
|
||||||
|
int sn = Integer.parseInt(snElement.getText());
|
||||||
logger.info("收到来自设备【{}】的通道: {}个,{}/{}", device.getDeviceId(), channelList.size(), catalogDataCatch.get(key) == null ? 0 :catalogDataCatch.get(key).size(), sumNum);
|
logger.info("收到来自设备【{}】的通道: {}个,{}/{}", device.getDeviceId(), channelList.size(), catalogDataCatch.get(key) == null ? 0 :catalogDataCatch.get(key).size(), sumNum);
|
||||||
catalogDataCatch.put(key, sumNum, device, channelList);
|
catalogDataCatch.put(device.getDeviceId(), sn, sumNum, device, channelList);
|
||||||
if (catalogDataCatch.get(key).size() == sumNum) {
|
if (catalogDataCatch.get(device.getDeviceId()).size() == sumNum) {
|
||||||
// 数据已经完整接收
|
// 数据已经完整接收
|
||||||
boolean resetChannelsResult = storager.resetChannels(device.getDeviceId(), catalogDataCatch.get(key));
|
boolean resetChannelsResult = storager.resetChannels(device.getDeviceId(), catalogDataCatch.get(device.getDeviceId()));
|
||||||
RequestMessage msg = new RequestMessage();
|
if (!resetChannelsResult) {
|
||||||
msg.setKey(key);
|
String errorMsg = "接收成功,写入失败,共" + sumNum + "条,已接收" + catalogDataCatch.get(device.getDeviceId()).size() + "条";
|
||||||
WVPResult<Object> result = new WVPResult<>();
|
catalogDataCatch.setChannelSyncEnd(device.getDeviceId(), errorMsg);
|
||||||
result.setCode(0);
|
|
||||||
result.setData(device);
|
|
||||||
if (resetChannelsResult || sumNum ==0) {
|
|
||||||
result.setMsg("更新成功,共" + sumNum + "条,已更新" + catalogDataCatch.get(key).size() + "条");
|
|
||||||
}else {
|
}else {
|
||||||
result.setMsg("接收成功,写入失败,共" + sumNum + "条,已接收" + catalogDataCatch.get(key).size() + "条");
|
catalogDataCatch.setChannelSyncEnd(device.getDeviceId(), null);
|
||||||
}
|
}
|
||||||
msg.setData(result);
|
|
||||||
deferredResultHolder.invokeAllResult(msg);
|
|
||||||
catalogDataCatch.del(key);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 回复200 OK
|
// 回复200 OK
|
||||||
responseAck(evt, Response.OK);
|
responseAck(evt, Response.OK);
|
||||||
if (offLineDetector.isOnline(device.getDeviceId())) {
|
|
||||||
publisher.onlineEventPublish(device, VideoManagerConstants.EVENT_ONLINE_MESSAGE);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} catch (DocumentException e) {
|
} catch (DocumentException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
|
@ -231,21 +216,18 @@ public class CatalogResponseMessageHandler extends SIPRequestProcessorParent imp
|
||||||
}
|
}
|
||||||
|
|
||||||
public SyncStatus getChannelSyncProgress(String deviceId) {
|
public SyncStatus getChannelSyncProgress(String deviceId) {
|
||||||
String key = DeferredResultHolder.CALLBACK_CMD_CATALOG + deviceId;
|
if (catalogDataCatch.get(deviceId) == null) {
|
||||||
if (catalogDataCatch.get(key) == null) {
|
|
||||||
return null;
|
return null;
|
||||||
}else {
|
}else {
|
||||||
return catalogDataCatch.getSyncStatus(key);
|
return catalogDataCatch.getSyncStatus(deviceId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setChannelSyncReady(String deviceId) {
|
public void setChannelSyncReady(Device device, int sn) {
|
||||||
String key = DeferredResultHolder.CALLBACK_CMD_CATALOG + deviceId;
|
catalogDataCatch.addReady(device, sn);
|
||||||
catalogDataCatch.addReady(key);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setChannelSyncEnd(String deviceId, String errorMsg) {
|
public void setChannelSyncEnd(String deviceId, String errorMsg) {
|
||||||
String key = DeferredResultHolder.CALLBACK_CMD_CATALOG + deviceId;
|
catalogDataCatch.setChannelSyncEnd(deviceId, errorMsg);
|
||||||
catalogDataCatch.setChannelSyncEnd(key, errorMsg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,15 +44,8 @@ public interface IDeviceService {
|
||||||
SyncStatus getChannelSyncStatus(String deviceId);
|
SyncStatus getChannelSyncStatus(String deviceId);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 设置通道同步状态
|
* 通道同步
|
||||||
* @param deviceId 设备ID
|
* @param device
|
||||||
*/
|
*/
|
||||||
void setChannelSyncReady(String deviceId);
|
void sync(Device device);
|
||||||
|
|
||||||
/**
|
|
||||||
* 设置同步结束
|
|
||||||
* @param deviceId 设备ID
|
|
||||||
* @param errorMsg 错误信息
|
|
||||||
*/
|
|
||||||
void setChannelSyncEnd(String deviceId, String errorMsg);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,6 +14,8 @@ import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
import javax.sip.DialogState;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 设备业务(目录订阅)
|
* 设备业务(目录订阅)
|
||||||
*/
|
*/
|
||||||
|
@ -39,19 +41,17 @@ public class DeviceServiceImpl implements IDeviceService {
|
||||||
if (device == null || device.getSubscribeCycleForCatalog() < 0) {
|
if (device == null || device.getSubscribeCycleForCatalog() < 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (dynamicTask.contains(device.getDeviceId() + "catalog")) {
|
CatalogSubscribeTask task = (CatalogSubscribeTask)dynamicTask.get(device.getDeviceId() + "catalog");
|
||||||
// 存在则停止现有的,开启新的
|
if (task != null && task.getDialogState() != null && task.getDialogState().equals(DialogState.CONFIRMED)) { // 已存在不需要再次添加
|
||||||
dynamicTask.stop(device.getDeviceId() + "catalog");
|
return true;
|
||||||
}
|
}
|
||||||
logger.info("[添加目录订阅] 设备{}", device.getDeviceId());
|
logger.info("[添加目录订阅] 设备{}", device.getDeviceId());
|
||||||
// 添加目录订阅
|
// 添加目录订阅
|
||||||
CatalogSubscribeTask catalogSubscribeTask = new CatalogSubscribeTask(device, sipCommander);
|
CatalogSubscribeTask catalogSubscribeTask = new CatalogSubscribeTask(device, sipCommander);
|
||||||
catalogSubscribeTask.run();
|
|
||||||
// 提前开始刷新订阅
|
// 提前开始刷新订阅
|
||||||
int subscribeCycleForCatalog = device.getSubscribeCycleForCatalog();
|
int subscribeCycleForCatalog = Math.max(device.getSubscribeCycleForCatalog(),30);
|
||||||
// 设置最小值为30
|
// 设置最小值为30
|
||||||
subscribeCycleForCatalog = Math.max(subscribeCycleForCatalog, 30);
|
dynamicTask.startCron(device.getDeviceId() + "catalog", catalogSubscribeTask, subscribeCycleForCatalog -1);
|
||||||
dynamicTask.startCron(device.getDeviceId() + "catalog", catalogSubscribeTask, subscribeCycleForCatalog);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,18 +70,16 @@ public class DeviceServiceImpl implements IDeviceService {
|
||||||
if (device == null || device.getSubscribeCycleForMobilePosition() < 0) {
|
if (device == null || device.getSubscribeCycleForMobilePosition() < 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (dynamicTask.contains(device.getDeviceId() + "mobile_position")) {
|
|
||||||
// 存在则停止现有的,开启新的
|
|
||||||
dynamicTask.stop(device.getDeviceId() + "mobile_position");
|
|
||||||
}
|
|
||||||
logger.info("[添加移动位置订阅] 设备{}", device.getDeviceId());
|
logger.info("[添加移动位置订阅] 设备{}", device.getDeviceId());
|
||||||
|
MobilePositionSubscribeTask task = (MobilePositionSubscribeTask)dynamicTask.get(device.getDeviceId() + "mobile_position");
|
||||||
|
if (task != null && task.getDialogState() != null && task.getDialogState().equals(DialogState.CONFIRMED)) { // 已存在不需要再次添加
|
||||||
|
return true;
|
||||||
|
}
|
||||||
// 添加目录订阅
|
// 添加目录订阅
|
||||||
MobilePositionSubscribeTask mobilePositionSubscribeTask = new MobilePositionSubscribeTask(device, sipCommander);
|
MobilePositionSubscribeTask mobilePositionSubscribeTask = new MobilePositionSubscribeTask(device, sipCommander);
|
||||||
mobilePositionSubscribeTask.run();
|
|
||||||
// 提前开始刷新订阅
|
// 提前开始刷新订阅
|
||||||
int subscribeCycleForCatalog = device.getSubscribeCycleForCatalog();
|
|
||||||
// 设置最小值为30
|
// 设置最小值为30
|
||||||
subscribeCycleForCatalog = Math.max(subscribeCycleForCatalog, 30);
|
int subscribeCycleForCatalog = Math.max(device.getSubscribeCycleForMobilePosition(),30);
|
||||||
dynamicTask.startCron(device.getDeviceId() + "mobile_position" , mobilePositionSubscribeTask, subscribeCycleForCatalog -1 );
|
dynamicTask.startCron(device.getDeviceId() + "mobile_position" , mobilePositionSubscribeTask, subscribeCycleForCatalog -1 );
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -102,12 +100,16 @@ public class DeviceServiceImpl implements IDeviceService {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setChannelSyncReady(String deviceId) {
|
public void sync(Device device) {
|
||||||
catalogResponseMessageHandler.setChannelSyncReady(deviceId);
|
if (catalogResponseMessageHandler.getChannelSyncProgress(device.getDeviceId()) != null) {
|
||||||
}
|
logger.info("开启同步时发现同步已经存在");
|
||||||
|
return;
|
||||||
@Override
|
}
|
||||||
public void setChannelSyncEnd(String deviceId, String errorMsg) {
|
int sn = (int)((Math.random()*9+1)*100000);
|
||||||
catalogResponseMessageHandler.setChannelSyncEnd(deviceId, errorMsg);
|
catalogResponseMessageHandler.setChannelSyncReady(device, sn);
|
||||||
|
sipCommander.catalogQuery(device, sn, event -> {
|
||||||
|
String errorMsg = String.format("同步通道失败,错误码: %s, %s", event.statusCode, event.msg);
|
||||||
|
catalogResponseMessageHandler.setChannelSyncEnd(device.getDeviceId(), errorMsg);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -238,12 +238,15 @@ public class VideoManagerStorageImpl implements IVideoManagerStorage {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean resetChannels(String deviceId, List<DeviceChannel> deviceChannelList) {
|
public boolean resetChannels(String deviceId, List<DeviceChannel> deviceChannelList) {
|
||||||
|
if (deviceChannelList == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);
|
TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);
|
||||||
// 数据去重
|
// 数据去重
|
||||||
List<DeviceChannel> channels = new ArrayList<>();
|
List<DeviceChannel> channels = new ArrayList<>();
|
||||||
StringBuilder stringBuilder = new StringBuilder();
|
StringBuilder stringBuilder = new StringBuilder();
|
||||||
Map<String, Integer> subContMap = new HashMap<>();
|
Map<String, Integer> subContMap = new HashMap<>();
|
||||||
if (deviceChannelList.size() > 1) {
|
if (deviceChannelList != null && deviceChannelList.size() > 1) {
|
||||||
// 数据去重
|
// 数据去重
|
||||||
Set<String> gbIdSet = new HashSet<>();
|
Set<String> gbIdSet = new HashSet<>();
|
||||||
for (DeviceChannel deviceChannel : deviceChannelList) {
|
for (DeviceChannel deviceChannel : deviceChannelList) {
|
||||||
|
@ -300,6 +303,7 @@ public class VideoManagerStorageImpl implements IVideoManagerStorage {
|
||||||
dataSourceTransactionManager.commit(transactionStatus); //手动提交
|
dataSourceTransactionManager.commit(transactionStatus); //手动提交
|
||||||
return true;
|
return true;
|
||||||
}catch (Exception e) {
|
}catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
dataSourceTransactionManager.rollback(transactionStatus);
|
dataSourceTransactionManager.rollback(transactionStatus);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -415,10 +419,9 @@ public class VideoManagerStorageImpl implements IVideoManagerStorage {
|
||||||
TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);
|
TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);
|
||||||
boolean result = false;
|
boolean result = false;
|
||||||
try {
|
try {
|
||||||
if (platformChannelMapper.delChannelForDeviceId(deviceId) <0 // 删除与国标平台的关联
|
platformChannelMapper.delChannelForDeviceId(deviceId);
|
||||||
|| deviceChannelMapper.cleanChannelsByDeviceId(deviceId) < 0 // 删除他的通道
|
deviceChannelMapper.cleanChannelsByDeviceId(deviceId);
|
||||||
|| deviceMapper.del(deviceId) < 0 // 移除设备信息
|
if ( deviceMapper.del(deviceId) < 0 ) {
|
||||||
) {
|
|
||||||
//事务回滚
|
//事务回滚
|
||||||
dataSourceTransactionManager.rollback(transactionStatus);
|
dataSourceTransactionManager.rollback(transactionStatus);
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,8 +4,13 @@ import com.alibaba.fastjson.JSONObject;
|
||||||
import com.genersoft.iot.vmp.conf.DynamicTask;
|
import com.genersoft.iot.vmp.conf.DynamicTask;
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.Device;
|
import com.genersoft.iot.vmp.gb28181.bean.Device;
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
|
import com.genersoft.iot.vmp.gb28181.bean.DeviceChannel;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.bean.SubscribeHolder;
|
||||||
import com.genersoft.iot.vmp.gb28181.bean.SyncStatus;
|
import com.genersoft.iot.vmp.gb28181.bean.SyncStatus;
|
||||||
import com.genersoft.iot.vmp.gb28181.event.DeviceOffLineDetector;
|
import com.genersoft.iot.vmp.gb28181.event.DeviceOffLineDetector;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.task.ISubscribeTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.task.impl.CatalogSubscribeTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.task.impl.MobilePositionSubscribeHandlerTask;
|
||||||
|
import com.genersoft.iot.vmp.gb28181.task.impl.MobilePositionSubscribeTask;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder;
|
import com.genersoft.iot.vmp.gb28181.transmit.callback.DeferredResultHolder;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.callback.RequestMessage;
|
import com.genersoft.iot.vmp.gb28181.transmit.callback.RequestMessage;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander;
|
import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander;
|
||||||
|
@ -29,9 +34,8 @@ import org.springframework.util.StringUtils;
|
||||||
import org.springframework.web.bind.annotation.*;
|
import org.springframework.web.bind.annotation.*;
|
||||||
import org.springframework.web.context.request.async.DeferredResult;
|
import org.springframework.web.context.request.async.DeferredResult;
|
||||||
|
|
||||||
import java.util.List;
|
import javax.sip.DialogState;
|
||||||
import java.util.Set;
|
import java.util.*;
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
@Api(tags = "国标设备查询", value = "国标设备查询")
|
@Api(tags = "国标设备查询", value = "国标设备查询")
|
||||||
@SuppressWarnings("rawtypes")
|
@SuppressWarnings("rawtypes")
|
||||||
|
@ -63,6 +67,9 @@ public class DeviceQuery {
|
||||||
@Autowired
|
@Autowired
|
||||||
private DynamicTask dynamicTask;
|
private DynamicTask dynamicTask;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private SubscribeHolder subscribeHolder;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 使用ID查询国标设备
|
* 使用ID查询国标设备
|
||||||
* @param deviceId 国标ID
|
* @param deviceId 国标ID
|
||||||
|
@ -165,12 +172,8 @@ public class DeviceQuery {
|
||||||
wvpResult.setData(syncStatus);
|
wvpResult.setData(syncStatus);
|
||||||
return wvpResult;
|
return wvpResult;
|
||||||
}
|
}
|
||||||
SyncStatus syncStatusReady = new SyncStatus();
|
deviceService.sync(device);
|
||||||
deviceService.setChannelSyncReady(deviceId);
|
|
||||||
cmder.catalogQuery(device, event -> {
|
|
||||||
String errorMsg = String.format("同步通道失败,错误码: %s, %s", event.statusCode, event.msg);
|
|
||||||
deviceService.setChannelSyncEnd(deviceId, errorMsg);
|
|
||||||
});
|
|
||||||
WVPResult<SyncStatus> wvpResult = new WVPResult<>();
|
WVPResult<SyncStatus> wvpResult = new WVPResult<>();
|
||||||
wvpResult.setCode(0);
|
wvpResult.setCode(0);
|
||||||
wvpResult.setMsg("开始同步");
|
wvpResult.setMsg("开始同步");
|
||||||
|
@ -469,4 +472,29 @@ public class DeviceQuery {
|
||||||
}
|
}
|
||||||
return wvpResult;
|
return wvpResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GetMapping("/{deviceId}/subscribe_info")
|
||||||
|
@ApiOperation(value = "获取设备的订阅状态", notes = "获取设备的订阅状态")
|
||||||
|
public WVPResult<Map<String, String>> getSubscribeInfo(@PathVariable String deviceId) {
|
||||||
|
Set<String> allKeys = dynamicTask.getAllKeys();
|
||||||
|
Map<String, String> dialogStateMap = new HashMap<>();
|
||||||
|
for (String key : allKeys) {
|
||||||
|
if (key.startsWith(deviceId)) {
|
||||||
|
ISubscribeTask subscribeTask = (ISubscribeTask)dynamicTask.get(key);
|
||||||
|
DialogState dialogState = subscribeTask.getDialogState();
|
||||||
|
if (dialogState == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (subscribeTask instanceof CatalogSubscribeTask) {
|
||||||
|
dialogStateMap.put("catalog", dialogState.toString());
|
||||||
|
}else if (subscribeTask instanceof MobilePositionSubscribeTask) {
|
||||||
|
dialogStateMap.put("mobilePosition", dialogState.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
WVPResult<Map<String, String>> wvpResult = new WVPResult<>();
|
||||||
|
wvpResult.setCode(0);
|
||||||
|
wvpResult.setData(dialogStateMap);
|
||||||
|
return wvpResult;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import com.alibaba.fastjson.JSON;
|
||||||
import com.alibaba.fastjson.JSONObject;
|
import com.alibaba.fastjson.JSONObject;
|
||||||
import com.genersoft.iot.vmp.VManageBootstrap;
|
import com.genersoft.iot.vmp.VManageBootstrap;
|
||||||
import com.genersoft.iot.vmp.common.VersionPo;
|
import com.genersoft.iot.vmp.common.VersionPo;
|
||||||
|
import com.genersoft.iot.vmp.conf.DynamicTask;
|
||||||
import com.genersoft.iot.vmp.conf.SipConfig;
|
import com.genersoft.iot.vmp.conf.SipConfig;
|
||||||
import com.genersoft.iot.vmp.conf.UserSetting;
|
import com.genersoft.iot.vmp.conf.UserSetting;
|
||||||
import com.genersoft.iot.vmp.conf.VersionInfo;
|
import com.genersoft.iot.vmp.conf.VersionInfo;
|
||||||
|
@ -27,6 +28,7 @@ import javax.sip.ObjectInUseException;
|
||||||
import javax.sip.SipProvider;
|
import javax.sip.SipProvider;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
@SuppressWarnings("rawtypes")
|
@SuppressWarnings("rawtypes")
|
||||||
@Api(tags = "服务控制")
|
@Api(tags = "服务控制")
|
||||||
|
@ -42,13 +44,16 @@ public class ServerController {
|
||||||
private IMediaServerService mediaServerService;
|
private IMediaServerService mediaServerService;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
VersionInfo versionInfo;
|
private VersionInfo versionInfo;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
SipConfig sipConfig;
|
private SipConfig sipConfig;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
UserSetting userSetting;
|
private UserSetting userSetting;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private DynamicTask dynamicTask;
|
||||||
|
|
||||||
@Value("${server.port}")
|
@Value("${server.port}")
|
||||||
private int serverPort;
|
private int serverPort;
|
||||||
|
@ -248,4 +253,35 @@ public class ServerController {
|
||||||
result.setData(jsonObject);
|
result.setData(jsonObject);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// @ApiOperation("当前进行中的动态任务")
|
||||||
|
// @GetMapping(value = "/dynamicTask")
|
||||||
|
// @ResponseBody
|
||||||
|
// public WVPResult<JSONObject> getDynamicTask(){
|
||||||
|
// WVPResult<JSONObject> result = new WVPResult<>();
|
||||||
|
// result.setCode(0);
|
||||||
|
// result.setMsg("success");
|
||||||
|
//
|
||||||
|
// JSONObject jsonObject = new JSONObject();
|
||||||
|
//
|
||||||
|
// Set<String> allKeys = dynamicTask.getAllKeys();
|
||||||
|
// jsonObject.put("server.port", serverPort);
|
||||||
|
// if (StringUtils.isEmpty(type)) {
|
||||||
|
// jsonObject.put("sip", JSON.toJSON(sipConfig));
|
||||||
|
// jsonObject.put("base", JSON.toJSON(userSetting));
|
||||||
|
// }else {
|
||||||
|
// switch (type){
|
||||||
|
// case "sip":
|
||||||
|
// jsonObject.put("sip", sipConfig);
|
||||||
|
// break;
|
||||||
|
// case "base":
|
||||||
|
// jsonObject.put("base", userSetting);
|
||||||
|
// break;
|
||||||
|
// default:
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// result.setData(jsonObject);
|
||||||
|
// return result;
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,23 +61,36 @@ export default {
|
||||||
if (!this.syncFlag) {
|
if (!this.syncFlag) {
|
||||||
this.syncFlag = true;
|
this.syncFlag = true;
|
||||||
}
|
}
|
||||||
if (res.data.data == null) {
|
|
||||||
this.syncStatus = "success"
|
if (res.data.data != null) {
|
||||||
this.percentage = 100;
|
if (res.data.data.total == 0) {
|
||||||
this.msg = '同步成功';
|
if (res.data.data.errorMsg !== null ){
|
||||||
}else if (res.data.data.total == 0){
|
this.msg = res.data.data.errorMsg;
|
||||||
this.msg = `等待同步中`;
|
this.syncStatus = "exception"
|
||||||
this.timmer = setTimeout(this.getProgress, 300)
|
}else {
|
||||||
}else if (res.data.data.errorMsg !== null ){
|
this.msg = `等待同步中`;
|
||||||
this.msg = res.data.data.errorMsg;
|
this.timmer = setTimeout(this.getProgress, 300)
|
||||||
this.syncStatus = "exception"
|
}
|
||||||
}else {
|
}else {
|
||||||
this.total = res.data.data.total;
|
if (res.data.data.total == res.data.data.current) {
|
||||||
this.current = res.data.data.current;
|
this.syncStatus = "success"
|
||||||
this.percentage = Math.floor(Number(res.data.data.current)/Number(res.data.data.total)* 10000)/100;
|
this.percentage = 100;
|
||||||
this.msg = `同步中...[${res.data.data.current}/${res.data.data.total}]`;
|
this.msg = '同步成功';
|
||||||
this.timmer = setTimeout(this.getProgress, 300)
|
}else {
|
||||||
|
if (res.data.data.errorMsg !== null ){
|
||||||
|
this.msg = res.data.data.errorMsg;
|
||||||
|
this.syncStatus = "exception"
|
||||||
|
}else {
|
||||||
|
this.total = res.data.data.total;
|
||||||
|
this.current = res.data.data.current;
|
||||||
|
this.percentage = Math.floor(Number(res.data.data.current)/Number(res.data.data.total)* 10000)/100;
|
||||||
|
this.msg = `同步中...[${res.data.data.current}/${res.data.data.total}]`;
|
||||||
|
this.timmer = setTimeout(this.getProgress, 300)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}else {
|
}else {
|
||||||
if (this.syncFlag) {
|
if (this.syncFlag) {
|
||||||
this.syncStatus = "success"
|
this.syncStatus = "success"
|
||||||
|
|
Loading…
Reference in New Issue