Merge branch 'master' of https://github.com/swwheihei/wvp.git
Conflicts: src/main/java/com/genersoft/iot/vmp/gb28181/transmit/response/impl/InviteResponseProcessor.javapull/1/head
commit
29710b7cc1
|
@ -0,0 +1,21 @@
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2020 swwhaha
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
4
pom.xml
4
pom.xml
|
@ -39,7 +39,7 @@
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-undertow</artifactId>
|
<artifactId>spring-boot-starter-tomcat</artifactId>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
@ -127,7 +127,7 @@
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.dom4j</groupId>
|
<groupId>org.dom4j</groupId>
|
||||||
<artifactId>dom4j</artifactId>
|
<artifactId>dom4j</artifactId>
|
||||||
<version>2.1.1</version>
|
<version>2.1.3</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.google.code.gson</groupId>
|
<groupId>com.google.code.gson</groupId>
|
||||||
|
|
|
@ -2,7 +2,7 @@ package com.genersoft.iot.vmp.common;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2019年5月30日 下午3:04:04
|
* @date: 2019年5月30日 下午3:04:04
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -13,7 +13,7 @@ import com.genersoft.iot.vmp.utils.redis.FastJsonRedisSerializer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:Redis中间件配置类,使用spring-data-redis集成,自动从application.yml中加载redis配置
|
* @Description:Redis中间件配置类,使用spring-data-redis集成,自动从application.yml中加载redis配置
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2019年5月30日 上午10:58:25
|
* @date: 2019年5月30日 上午10:58:25
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -5,7 +5,7 @@ import org.springframework.context.annotation.Configuration;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午2:46:00
|
* @date: 2020年5月6日 下午2:46:00
|
||||||
*/
|
*/
|
||||||
@Configuration("vmConfig")
|
@Configuration("vmConfig")
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
package com.genersoft.iot.vmp.gb28181;
|
package com.genersoft.iot.vmp.gb28181;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
import javax.annotation.PostConstruct;
|
import javax.annotation.PostConstruct;
|
||||||
|
@ -84,7 +85,7 @@ public class SipLayer implements SipListener, Runnable {
|
||||||
* 0; public static final int TRACE_MESSAGES = 16; public static final int
|
* 0; public static final int TRACE_MESSAGES = 16; public static final int
|
||||||
* TRACE_EXCEPTION = 17; public static final int TRACE_DEBUG = 32;
|
* TRACE_EXCEPTION = 17; public static final int TRACE_DEBUG = 32;
|
||||||
*/
|
*/
|
||||||
properties.setProperty("gov.nist.javax.sip.TRACE_LEVEL", "0");
|
properties.setProperty("gov.nist.javax.sip.TRACE_LEVEL", "32");
|
||||||
properties.setProperty("gov.nist.javax.sip.SERVER_LOG", "sip_server_log");
|
properties.setProperty("gov.nist.javax.sip.SERVER_LOG", "sip_server_log");
|
||||||
properties.setProperty("gov.nist.javax.sip.DEBUG_LOG", "sip_debug_log");
|
properties.setProperty("gov.nist.javax.sip.DEBUG_LOG", "sip_debug_log");
|
||||||
sipStack = (SipStackImpl) sipFactory.createSipStack(properties);
|
sipStack = (SipStackImpl) sipFactory.createSipStack(properties);
|
||||||
|
@ -99,13 +100,15 @@ public class SipLayer implements SipListener, Runnable {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void startTcpListener() throws Exception {
|
private void startTcpListener() throws Exception {
|
||||||
ListeningPoint tcpListeningPoint = sipStack.createListeningPoint(sipConfig.getSipIp(), sipConfig.getSipPort(), "TCP");
|
ListeningPoint tcpListeningPoint = sipStack.createListeningPoint(sipConfig.getSipIp(), sipConfig.getSipPort(),
|
||||||
|
"TCP");
|
||||||
tcpSipProvider = sipStack.createSipProvider(tcpListeningPoint);
|
tcpSipProvider = sipStack.createSipProvider(tcpListeningPoint);
|
||||||
tcpSipProvider.addSipListener(this);
|
tcpSipProvider.addSipListener(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void startUdpListener() throws Exception {
|
private void startUdpListener() throws Exception {
|
||||||
ListeningPoint udpListeningPoint = sipStack.createListeningPoint(sipConfig.getSipIp(), sipConfig.getSipPort(), "UDP");
|
ListeningPoint udpListeningPoint = sipStack.createListeningPoint(sipConfig.getSipIp(), sipConfig.getSipPort(),
|
||||||
|
"UDP");
|
||||||
udpSipProvider = sipStack.createSipProvider(udpListeningPoint);
|
udpSipProvider = sipStack.createSipProvider(udpListeningPoint);
|
||||||
udpSipProvider.addSipListener(this);
|
udpSipProvider.addSipListener(this);
|
||||||
}
|
}
|
||||||
|
@ -126,14 +129,23 @@ public class SipLayer implements SipListener, Runnable {
|
||||||
int status = response.getStatusCode();
|
int status = response.getStatusCode();
|
||||||
if ((status >= 200) && (status < 300)) { // Success!
|
if ((status >= 200) && (status < 300)) { // Success!
|
||||||
ISIPResponseProcessor processor = processorFactory.createResponseProcessor(evt);
|
ISIPResponseProcessor processor = processorFactory.createResponseProcessor(evt);
|
||||||
|
try {
|
||||||
processor.process(evt, this, sipConfig);
|
processor.process(evt, this, sipConfig);
|
||||||
|
} catch (ParseException e) {
|
||||||
|
// TODO Auto-generated catch block
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
// } else if (status == Response.TRYING) {
|
||||||
|
// trying不会回复
|
||||||
|
} else if ((status >= 100) && (status < 200)) {
|
||||||
|
// 增加其它无需回复的响应,如101、180等
|
||||||
} else {
|
} else {
|
||||||
logger.warn("接收到失败的response响应!status:" + status + ",message:" + response.getContent().toString());
|
logger.warn("接收到失败的response响应!status:" + status + ",message:" + response.getReasonPhrase()/* .getContent().toString()*/);
|
||||||
}
|
}
|
||||||
// trying不会回复
|
// trying不会回复
|
||||||
if (status == Response.TRYING) {
|
// if (status == Response.TRYING) {
|
||||||
|
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -8,7 +8,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:注册逻辑处理,当设备注册后触发逻辑。
|
* @Description:注册逻辑处理,当设备注册后触发逻辑。
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午9:41:46
|
* @date: 2020年5月8日 下午9:41:46
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -4,7 +4,7 @@ import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备录像信息bean
|
* @Description:设备录像信息bean
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午2:05:56
|
* @date: 2020年5月8日 下午2:05:56
|
||||||
*/
|
*/
|
||||||
public class RecordInfo {
|
public class RecordInfo {
|
||||||
|
|
|
@ -2,7 +2,7 @@ package com.genersoft.iot.vmp.gb28181.bean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备录像bean
|
* @Description:设备录像bean
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午2:06:54
|
* @date: 2020年5月8日 下午2:06:54
|
||||||
*/
|
*/
|
||||||
public class RecordItem {
|
public class RecordItem {
|
||||||
|
|
|
@ -8,7 +8,7 @@ import com.genersoft.iot.vmp.utils.redis.RedisUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备离在线状态检测器,用于检测设备状态
|
* @Description:设备离在线状态检测器,用于检测设备状态
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月13日 下午2:40:29
|
* @date: 2020年5月13日 下午2:40:29
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -9,7 +9,7 @@ import com.genersoft.iot.vmp.gb28181.event.online.OnlineEvent;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:Event事件通知推送器,支持推送在线事件、离线事件
|
* @Description:Event事件通知推送器,支持推送在线事件、离线事件
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 上午11:30:50
|
* @date: 2020年5月6日 上午11:30:50
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -11,7 +11,7 @@ import com.genersoft.iot.vmp.gb28181.event.EventPublisher;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备心跳超时监听,借助redis过期特性,进行监听,监听到说明设备心跳超时,发送离线事件
|
* @Description:设备心跳超时监听,借助redis过期特性,进行监听,监听到说明设备心跳超时,发送离线事件
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 上午11:35:46
|
* @date: 2020年5月6日 上午11:35:46
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -4,7 +4,7 @@ import org.springframework.context.ApplicationEvent;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 上午11:33:13
|
* @date: 2020年5月6日 上午11:33:13
|
||||||
*/
|
*/
|
||||||
public class OfflineEvent extends ApplicationEvent {
|
public class OfflineEvent extends ApplicationEvent {
|
||||||
|
|
|
@ -14,7 +14,7 @@ import com.genersoft.iot.vmp.utils.redis.RedisUtil;
|
||||||
* @Description: 离线事件监听器,监听到离线后,修改设备离在线状态。 设备离线有两个来源:
|
* @Description: 离线事件监听器,监听到离线后,修改设备离在线状态。 设备离线有两个来源:
|
||||||
* 1、设备主动注销,发送注销指令,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.RegisterRequestProcessor}
|
* 1、设备主动注销,发送注销指令,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.RegisterRequestProcessor}
|
||||||
* 2、设备未知原因离线,心跳超时,{@link com.genersoft.iot.vmp.gb28181.event.offline.OfflineEventListener}
|
* 2、设备未知原因离线,心跳超时,{@link com.genersoft.iot.vmp.gb28181.event.offline.OfflineEventListener}
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午1:51:23
|
* @date: 2020年5月6日 下午1:51:23
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -4,7 +4,7 @@ import org.springframework.context.ApplicationEvent;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 上午11:32:56
|
* @date: 2020年5月6日 上午11:32:56
|
||||||
*/
|
*/
|
||||||
public class OnlineEvent extends ApplicationEvent {
|
public class OnlineEvent extends ApplicationEvent {
|
||||||
|
|
|
@ -14,7 +14,7 @@ import com.genersoft.iot.vmp.utils.redis.RedisUtil;
|
||||||
* @Description: 在线事件监听器,监听到离线后,修改设备离在线状态。 设备在线有两个来源:
|
* @Description: 在线事件监听器,监听到离线后,修改设备离在线状态。 设备在线有两个来源:
|
||||||
* 1、设备主动注销,发送注销指令,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.RegisterRequestProcessor}
|
* 1、设备主动注销,发送注销指令,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.RegisterRequestProcessor}
|
||||||
* 2、设备未知原因离线,心跳超时,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.MessageRequestProcessor}
|
* 2、设备未知原因离线,心跳超时,{@link com.genersoft.iot.vmp.gb28181.transmit.request.impl.MessageRequestProcessor}
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午1:51:23
|
* @date: 2020年5月6日 下午1:51:23
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -9,7 +9,7 @@ import com.genersoft.iot.vmp.utils.SpringBeanFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:SIP信令中的SSRC工具类。SSRC值由10位十进制整数组成的字符串,第一位为0代表实况,为1则代表回放;第二位至第六位由监控域ID的第4位到第8位组成;最后4位为不重复的4个整数
|
* @Description:SIP信令中的SSRC工具类。SSRC值由10位十进制整数组成的字符串,第一位为0代表实况,为1则代表回放;第二位至第六位由监控域ID的第4位到第8位组成;最后4位为不重复的4个整数
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月10日 上午11:57:57
|
* @date: 2020年5月10日 上午11:57:57
|
||||||
*/
|
*/
|
||||||
public class SsrcUtil {
|
public class SsrcUtil {
|
||||||
|
|
|
@ -8,7 +8,7 @@ import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:视频流session管理器,管理视频预览、预览回放的通信句柄
|
* @Description:视频流session管理器,管理视频预览、预览回放的通信句柄
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月13日 下午4:03:02
|
* @date: 2020年5月13日 下午4:03:02
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -26,7 +26,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.response.impl.OtherResponseProcess
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:24:37
|
* @date: 2020年5月3日 下午4:24:37
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -10,7 +10,7 @@ import org.springframework.web.context.request.async.DeferredResult;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午7:59:05
|
* @date: 2020年5月8日 下午7:59:05
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -2,7 +2,7 @@ package com.genersoft.iot.vmp.gb28181.transmit.callback;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:TODO(这里用一句话描述这个类的作用)
|
* @Description:TODO(这里用一句话描述这个类的作用)
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午1:09:18
|
* @date: 2020年5月8日 下午1:09:18
|
||||||
*/
|
*/
|
||||||
public class RequestMessage {
|
public class RequestMessage {
|
||||||
|
|
|
@ -4,7 +4,7 @@ import com.genersoft.iot.vmp.gb28181.bean.Device;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备能力接口,用于定义设备的控制、查询能力
|
* @Description:设备能力接口,用于定义设备的控制、查询能力
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午9:16:34
|
* @date: 2020年5月3日 下午9:16:34
|
||||||
*/
|
*/
|
||||||
public interface ISIPCommander {
|
public interface ISIPCommander {
|
||||||
|
|
|
@ -25,7 +25,7 @@ import com.genersoft.iot.vmp.gb28181.bean.Host;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:摄像头命令request创造器 TODO 冗余代码太多待优化
|
* @Description:摄像头命令request创造器 TODO 冗余代码太多待优化
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 上午9:29:02
|
* @date: 2020年5月6日 上午9:29:02
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
@ -79,7 +79,8 @@ public class SIPRequestHeaderProvider {
|
||||||
SipURI requestLine = layer.getAddressFactory().createSipURI(channelId, host.getAddress());
|
SipURI requestLine = layer.getAddressFactory().createSipURI(channelId, host.getAddress());
|
||||||
//via
|
//via
|
||||||
ArrayList<ViaHeader> viaHeaders = new ArrayList<ViaHeader>();
|
ArrayList<ViaHeader> viaHeaders = new ArrayList<ViaHeader>();
|
||||||
ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(sipConfig.getSipIp(), sipConfig.getSipPort(), device.getTransport(), viaTag);
|
// ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(sipConfig.getSipIp(), sipConfig.getSipPort(), device.getTransport(), viaTag);
|
||||||
|
ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(device.getHost().getIp(), device.getHost().getPort(), device.getTransport(), viaTag);
|
||||||
viaHeader.setRPort();
|
viaHeader.setRPort();
|
||||||
viaHeaders.add(viaHeader);
|
viaHeaders.add(viaHeader);
|
||||||
//from
|
//from
|
||||||
|
@ -108,6 +109,7 @@ public class SIPRequestHeaderProvider {
|
||||||
request = layer.getMessageFactory().createRequest(requestLine, Request.INVITE, callIdHeader, cSeqHeader,fromHeader, toHeader, viaHeaders, maxForwards);
|
request = layer.getMessageFactory().createRequest(requestLine, Request.INVITE, callIdHeader, cSeqHeader,fromHeader, toHeader, viaHeaders, maxForwards);
|
||||||
|
|
||||||
Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), sipConfig.getSipIp()+":"+sipConfig.getSipPort()));
|
Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), sipConfig.getSipIp()+":"+sipConfig.getSipPort()));
|
||||||
|
// Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), device.getHost().getIp()+":"+device.getHost().getPort()));
|
||||||
request.addHeader(layer.getHeaderFactory().createContactHeader(concatAddress));
|
request.addHeader(layer.getHeaderFactory().createContactHeader(concatAddress));
|
||||||
|
|
||||||
ContentTypeHeader contentTypeHeader = layer.getHeaderFactory().createContentTypeHeader("Application", "SDP");
|
ContentTypeHeader contentTypeHeader = layer.getHeaderFactory().createContentTypeHeader("Application", "SDP");
|
||||||
|
@ -122,7 +124,8 @@ public class SIPRequestHeaderProvider {
|
||||||
SipURI requestLine = layer.getAddressFactory().createSipURI(device.getDeviceId(), host.getAddress());
|
SipURI requestLine = layer.getAddressFactory().createSipURI(device.getDeviceId(), host.getAddress());
|
||||||
//via
|
//via
|
||||||
ArrayList<ViaHeader> viaHeaders = new ArrayList<ViaHeader>();
|
ArrayList<ViaHeader> viaHeaders = new ArrayList<ViaHeader>();
|
||||||
ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(sipConfig.getSipIp(), sipConfig.getSipPort(), device.getTransport(), viaTag);
|
// ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(sipConfig.getSipIp(), sipConfig.getSipPort(), device.getTransport(), viaTag);
|
||||||
|
ViaHeader viaHeader = layer.getHeaderFactory().createViaHeader(device.getHost().getIp(), device.getHost().getPort(), device.getTransport(), viaTag);
|
||||||
viaHeader.setRPort();
|
viaHeader.setRPort();
|
||||||
viaHeaders.add(viaHeader);
|
viaHeaders.add(viaHeader);
|
||||||
//from
|
//from
|
||||||
|
@ -151,6 +154,7 @@ public class SIPRequestHeaderProvider {
|
||||||
request = layer.getMessageFactory().createRequest(requestLine, Request.INVITE, callIdHeader, cSeqHeader,fromHeader, toHeader, viaHeaders, maxForwards);
|
request = layer.getMessageFactory().createRequest(requestLine, Request.INVITE, callIdHeader, cSeqHeader,fromHeader, toHeader, viaHeaders, maxForwards);
|
||||||
|
|
||||||
Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), sipConfig.getSipIp()+":"+sipConfig.getSipPort()));
|
Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), sipConfig.getSipIp()+":"+sipConfig.getSipPort()));
|
||||||
|
// Address concatAddress = layer.getAddressFactory().createAddress(layer.getAddressFactory().createSipURI(sipConfig.getSipId(), device.getHost().getIp()+":"+device.getHost().getPort()));
|
||||||
request.addHeader(layer.getHeaderFactory().createContactHeader(concatAddress));
|
request.addHeader(layer.getHeaderFactory().createContactHeader(concatAddress));
|
||||||
|
|
||||||
ContentTypeHeader contentTypeHeader = layer.getHeaderFactory().createContentTypeHeader("Application", "SDP");
|
ContentTypeHeader contentTypeHeader = layer.getHeaderFactory().createContentTypeHeader("Application", "SDP");
|
||||||
|
|
|
@ -7,10 +7,13 @@ import javax.sip.Dialog;
|
||||||
import javax.sip.InvalidArgumentException;
|
import javax.sip.InvalidArgumentException;
|
||||||
import javax.sip.SipException;
|
import javax.sip.SipException;
|
||||||
import javax.sip.TransactionDoesNotExistException;
|
import javax.sip.TransactionDoesNotExistException;
|
||||||
|
import javax.sip.address.Address;
|
||||||
|
import javax.sip.address.SipURI;
|
||||||
import javax.sip.header.ViaHeader;
|
import javax.sip.header.ViaHeader;
|
||||||
import javax.sip.message.Request;
|
import javax.sip.message.Request;
|
||||||
|
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.boot.autoconfigure.security.SecurityProperties.Headers;
|
||||||
import org.springframework.stereotype.Component;
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
import com.genersoft.iot.vmp.conf.SipConfig;
|
import com.genersoft.iot.vmp.conf.SipConfig;
|
||||||
|
@ -21,9 +24,12 @@ import com.genersoft.iot.vmp.gb28181.transmit.cmd.ISIPCommander;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.cmd.SIPRequestHeaderProvider;
|
import com.genersoft.iot.vmp.gb28181.transmit.cmd.SIPRequestHeaderProvider;
|
||||||
import com.genersoft.iot.vmp.gb28181.utils.DateUtil;
|
import com.genersoft.iot.vmp.gb28181.utils.DateUtil;
|
||||||
|
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
import java.util.regex.Matcher;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:设备能力接口,用于定义设备的控制、查询能力
|
* @Description:设备能力接口,用于定义设备的控制、查询能力
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午9:22:48
|
* @date: 2020年5月3日 下午9:22:48
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
@ -94,6 +100,49 @@ public class SIPCommander implements ISIPCommander {
|
||||||
return ptzCmd(device, channelId, 0, 0, inOut, 0, zoomSpeed);
|
return ptzCmd(device, channelId, 0, 0, inOut, 0, zoomSpeed);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 云台指令码计算
|
||||||
|
*
|
||||||
|
* @param leftRight 镜头左移右移 0:停止 1:左移 2:右移
|
||||||
|
* @param upDown 镜头上移下移 0:停止 1:上移 2:下移
|
||||||
|
* @param inOut 镜头放大缩小 0:停止 1:缩小 2:放大
|
||||||
|
* @param moveSpeed 镜头移动速度 默认 0XFF (0-255)
|
||||||
|
* @param zoomSpeed 镜头缩放速度 默认 0X1 (0-255)
|
||||||
|
*/
|
||||||
|
public static String cmdString(int leftRight, int upDown, int inOut, int moveSpeed, int zoomSpeed) {
|
||||||
|
int cmdCode = 0;
|
||||||
|
if (leftRight == 2) {
|
||||||
|
cmdCode|=0x01; // 右移
|
||||||
|
} else if(leftRight == 1) {
|
||||||
|
cmdCode|=0x02; // 左移
|
||||||
|
}
|
||||||
|
if (upDown == 2) {
|
||||||
|
cmdCode|=0x04; // 下移
|
||||||
|
} else if(upDown == 1) {
|
||||||
|
cmdCode|=0x08; // 上移
|
||||||
|
}
|
||||||
|
if (inOut == 2) {
|
||||||
|
cmdCode |= 0x10; // 放大
|
||||||
|
} else if(inOut == 1) {
|
||||||
|
cmdCode |= 0x20; // 缩小
|
||||||
|
}
|
||||||
|
StringBuilder builder = new StringBuilder("A50F01");
|
||||||
|
String strTmp;
|
||||||
|
strTmp = String.format("%02X", cmdCode);
|
||||||
|
builder.append(strTmp, 0, 2);
|
||||||
|
strTmp = String.format("%02X", moveSpeed);
|
||||||
|
builder.append(strTmp, 0, 2);
|
||||||
|
builder.append(strTmp, 0, 2);
|
||||||
|
strTmp = String.format("%X", zoomSpeed);
|
||||||
|
builder.append(strTmp, 0, 1).append("0");
|
||||||
|
//计算校验码
|
||||||
|
int checkCode = (0XA5 + 0X0F + 0X01 + cmdCode + moveSpeed + moveSpeed + (zoomSpeed /*<< 4*/ & 0XF0)) % 0X100;
|
||||||
|
strTmp = String.format("%02X", checkCode);
|
||||||
|
builder.append(strTmp, 0, 2);
|
||||||
|
return builder.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 云台控制,支持方向与缩放控制
|
* 云台控制,支持方向与缩放控制
|
||||||
*
|
*
|
||||||
|
@ -109,13 +158,14 @@ public class SIPCommander implements ISIPCommander {
|
||||||
public boolean ptzCmd(Device device, String channelId, int leftRight, int upDown, int inOut, int moveSpeed,
|
public boolean ptzCmd(Device device, String channelId, int leftRight, int upDown, int inOut, int moveSpeed,
|
||||||
int zoomSpeed) {
|
int zoomSpeed) {
|
||||||
try {
|
try {
|
||||||
|
String cmdStr= cmdString(leftRight, upDown, inOut, moveSpeed, zoomSpeed);
|
||||||
StringBuffer ptzXml = new StringBuffer(200);
|
StringBuffer ptzXml = new StringBuffer(200);
|
||||||
ptzXml.append("<?xml version=\"1.0\" ?>");
|
ptzXml.append("<?xml version=\"1.0\" ?>");
|
||||||
ptzXml.append("<Control>");
|
ptzXml.append("<Control>");
|
||||||
ptzXml.append("<CmdType>DeviceControl</CmdType>");
|
ptzXml.append("<CmdType>DeviceControl</CmdType>");
|
||||||
ptzXml.append("<SN>" + (int)((Math.random()*9+1)*100000) + "</SN>");
|
ptzXml.append("<SN>" + (int)((Math.random()*9+1)*100000) + "</SN>");
|
||||||
ptzXml.append("<DeviceID>" + channelId + "</DeviceID>");
|
ptzXml.append("<DeviceID>" + channelId + "</DeviceID>");
|
||||||
ptzXml.append("<PTZCmd>" + "</PTZCmd>");
|
ptzXml.append("<PTZCmd>" + cmdStr + "</PTZCmd>");
|
||||||
ptzXml.append("<Info>");
|
ptzXml.append("<Info>");
|
||||||
ptzXml.append("</Info>");
|
ptzXml.append("</Info>");
|
||||||
ptzXml.append("</Control>");
|
ptzXml.append("</Control>");
|
||||||
|
@ -123,7 +173,6 @@ public class SIPCommander implements ISIPCommander {
|
||||||
Request request = headerProvider.createMessageRequest(device, ptzXml.toString(), "ViaPtzBranch", "FromPtzTag", "ToPtzTag");
|
Request request = headerProvider.createMessageRequest(device, ptzXml.toString(), "ViaPtzBranch", "FromPtzTag", "ToPtzTag");
|
||||||
|
|
||||||
transmitRequest(device, request);
|
transmitRequest(device, request);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
} catch (SipException | ParseException | InvalidArgumentException e) {
|
} catch (SipException | ParseException | InvalidArgumentException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
|
@ -245,6 +294,13 @@ public class SIPCommander implements ISIPCommander {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Request byeRequest = dialog.createRequest(Request.BYE);
|
Request byeRequest = dialog.createRequest(Request.BYE);
|
||||||
|
SipURI byeURI = (SipURI) byeRequest.getRequestURI();
|
||||||
|
String vh = transaction.getRequest().getHeader(ViaHeader.NAME).toString();
|
||||||
|
Pattern p = Pattern.compile("(\\d+\\.\\d+\\.\\d+\\.\\d+)\\:(\\d+)");
|
||||||
|
Matcher matcher = p.matcher(vh);
|
||||||
|
if (matcher.find()) {
|
||||||
|
byeURI.setHost(matcher.group(1));
|
||||||
|
}
|
||||||
ViaHeader viaHeader = (ViaHeader) byeRequest.getHeader(ViaHeader.NAME);
|
ViaHeader viaHeader = (ViaHeader) byeRequest.getHeader(ViaHeader.NAME);
|
||||||
String protocol = viaHeader.getTransport().toUpperCase();
|
String protocol = viaHeader.getTransport().toUpperCase();
|
||||||
ClientTransaction clientTransaction = null;
|
ClientTransaction clientTransaction = null;
|
||||||
|
@ -258,6 +314,8 @@ public class SIPCommander implements ISIPCommander {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
} catch (SipException e) {
|
} catch (SipException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
|
} catch (ParseException e) {
|
||||||
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@ import com.genersoft.iot.vmp.gb28181.SipLayer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:处理接收IPCamera发来的SIP协议请求消息
|
* @Description:处理接收IPCamera发来的SIP协议请求消息
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:42:22
|
* @date: 2020年5月3日 下午4:42:22
|
||||||
*/
|
*/
|
||||||
public interface ISIPRequestProcessor {
|
public interface ISIPRequestProcessor {
|
||||||
|
|
|
@ -16,7 +16,7 @@ import gov.nist.javax.sip.header.CSeq;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:ACK请求处理器
|
* @Description:ACK请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:31:45
|
* @date: 2020年5月3日 下午5:31:45
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.request.ISIPRequestProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description: BYE请求处理器
|
* @Description: BYE请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:05
|
* @date: 2020年5月3日 下午5:32:05
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.request.ISIPRequestProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:CANCEL请求处理器
|
* @Description:CANCEL请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:23
|
* @date: 2020年5月3日 下午5:32:23
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.request.ISIPRequestProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:处理INVITE请求
|
* @Description:处理INVITE请求
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:43:52
|
* @date: 2020年5月3日 下午4:43:52
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -43,7 +43,7 @@ import com.genersoft.iot.vmp.utils.redis.RedisUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:MESSAGE请求处理器
|
* @Description:MESSAGE请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:41
|
* @date: 2020年5月3日 下午5:32:41
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
@ -100,6 +100,7 @@ public class MessageRequestProcessor implements ISIPRequestProcessor {
|
||||||
|
|
||||||
Request request = evt.getRequest();
|
Request request = evt.getRequest();
|
||||||
SAXReader reader = new SAXReader();
|
SAXReader reader = new SAXReader();
|
||||||
|
reader.setEncoding("gbk");
|
||||||
Document xml;
|
Document xml;
|
||||||
try {
|
try {
|
||||||
xml = reader.read(new ByteArrayInputStream(request.getRawContent()));
|
xml = reader.read(new ByteArrayInputStream(request.getRawContent()));
|
||||||
|
@ -375,7 +376,7 @@ public class MessageRequestProcessor implements ISIPRequestProcessor {
|
||||||
private Element getRootElement(RequestEvent evt) throws DocumentException {
|
private Element getRootElement(RequestEvent evt) throws DocumentException {
|
||||||
Request request = evt.getRequest();
|
Request request = evt.getRequest();
|
||||||
SAXReader reader = new SAXReader();
|
SAXReader reader = new SAXReader();
|
||||||
reader.setEncoding("GB2312");
|
reader.setEncoding("gbk");
|
||||||
Document xml = reader.read(new ByteArrayInputStream(request.getRawContent()));
|
Document xml = reader.read(new ByteArrayInputStream(request.getRawContent()));
|
||||||
return xml.getRootElement();
|
return xml.getRootElement();
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.request.ISIPRequestProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:暂不支持的消息请求处理器
|
* @Description:暂不支持的消息请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:59
|
* @date: 2020年5月3日 下午5:32:59
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -38,7 +38,7 @@ import gov.nist.javax.sip.header.Expires;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:收到注册请求 处理
|
* @Description:收到注册请求 处理
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:47:25
|
* @date: 2020年5月3日 下午4:47:25
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -17,7 +17,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.request.ISIPRequestProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:SUBSCRIBE请求处理器
|
* @Description:SUBSCRIBE请求处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:31:20
|
* @date: 2020年5月3日 下午5:31:20
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
package com.genersoft.iot.vmp.gb28181.transmit.response;
|
package com.genersoft.iot.vmp.gb28181.transmit.response;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
|
|
||||||
import javax.sip.ResponseEvent;
|
import javax.sip.ResponseEvent;
|
||||||
|
|
||||||
import com.genersoft.iot.vmp.conf.SipConfig;
|
import com.genersoft.iot.vmp.conf.SipConfig;
|
||||||
|
@ -7,11 +9,11 @@ import com.genersoft.iot.vmp.gb28181.SipLayer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:处理接收IPCamera发来的SIP协议响应消息
|
* @Description:处理接收IPCamera发来的SIP协议响应消息
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:42:22
|
* @date: 2020年5月3日 下午4:42:22
|
||||||
*/
|
*/
|
||||||
public interface ISIPResponseProcessor {
|
public interface ISIPResponseProcessor {
|
||||||
|
|
||||||
public void process(ResponseEvent evt, SipLayer layer, SipConfig config);
|
public void process(ResponseEvent evt, SipLayer layer, SipConfig config) throws ParseException;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.response.ISIPResponseProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description: BYE请求响应器
|
* @Description: BYE请求响应器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:05
|
* @date: 2020年5月3日 下午5:32:05
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.response.ISIPResponseProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:CANCEL响应处理器
|
* @Description:CANCEL响应处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:23
|
* @date: 2020年5月3日 下午5:32:23
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -2,7 +2,6 @@ package com.genersoft.iot.vmp.gb28181.transmit.response.impl;
|
||||||
|
|
||||||
import java.text.ParseException;
|
import java.text.ParseException;
|
||||||
|
|
||||||
import javax.sip.ClientTransaction;
|
|
||||||
import javax.sip.Dialog;
|
import javax.sip.Dialog;
|
||||||
import javax.sip.InvalidArgumentException;
|
import javax.sip.InvalidArgumentException;
|
||||||
import javax.sip.ResponseEvent;
|
import javax.sip.ResponseEvent;
|
||||||
|
@ -22,9 +21,10 @@ import com.genersoft.iot.vmp.gb28181.SipLayer;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.SIPProcessorFactory;
|
import com.genersoft.iot.vmp.gb28181.transmit.SIPProcessorFactory;
|
||||||
import com.genersoft.iot.vmp.gb28181.transmit.response.ISIPResponseProcessor;
|
import com.genersoft.iot.vmp.gb28181.transmit.response.ISIPResponseProcessor;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:处理INVITE响应
|
* @Description:处理INVITE响应
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午4:43:52
|
* @date: 2020年5月3日 下午4:43:52
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
@ -35,17 +35,16 @@ public class InviteResponseProcessor implements ISIPResponseProcessor {
|
||||||
/**
|
/**
|
||||||
* 处理invite响应
|
* 处理invite响应
|
||||||
*
|
*
|
||||||
* @param evt
|
* @param evt 响应消息
|
||||||
* 响应消息
|
* @throws ParseException
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void process(ResponseEvent evt, SipLayer layer, SipConfig config) {
|
public void process(ResponseEvent evt, SipLayer layer, SipConfig config) throws ParseException {
|
||||||
try {
|
try {
|
||||||
Response response = evt.getResponse();
|
Response response = evt.getResponse();
|
||||||
int statusCode = response.getStatusCode();
|
int statusCode = response.getStatusCode();
|
||||||
// trying不会回复
|
// trying不会回复
|
||||||
if (statusCode == Response.TRYING) {
|
if (statusCode == Response.TRYING) {
|
||||||
|
|
||||||
}
|
}
|
||||||
// 成功响应
|
// 成功响应
|
||||||
// 下发ack
|
// 下发ack
|
||||||
|
@ -56,8 +55,9 @@ public class InviteResponseProcessor implements ISIPResponseProcessor {
|
||||||
// return;
|
// return;
|
||||||
// }
|
// }
|
||||||
// Dialog clientDialog = clientTransaction.getDialog();
|
// Dialog clientDialog = clientTransaction.getDialog();
|
||||||
//
|
|
||||||
// CSeqHeader clientCSeqHeader = (CSeqHeader) response.getHeader(CSeqHeader.NAME);
|
// CSeqHeader clientCSeqHeader = (CSeqHeader)
|
||||||
|
// response.getHeader(CSeqHeader.NAME);
|
||||||
// long cseqId = clientCSeqHeader.getSeqNumber();
|
// long cseqId = clientCSeqHeader.getSeqNumber();
|
||||||
// /*
|
// /*
|
||||||
// createAck函数,创建的ackRequest,会采用Invite响应的200OK,中的contact字段中的地址,作为目标地址。
|
// createAck函数,创建的ackRequest,会采用Invite响应的200OK,中的contact字段中的地址,作为目标地址。
|
||||||
|
@ -67,12 +67,30 @@ public class InviteResponseProcessor implements ISIPResponseProcessor {
|
||||||
// Request ackRequest = clientDialog.createAck(cseqId);
|
// Request ackRequest = clientDialog.createAck(cseqId);
|
||||||
// SipURI requestURI = (SipURI) ackRequest.getRequestURI();
|
// SipURI requestURI = (SipURI) ackRequest.getRequestURI();
|
||||||
// ViaHeader viaHeader = (ViaHeader) response.getHeader(ViaHeader.NAME);
|
// ViaHeader viaHeader = (ViaHeader) response.getHeader(ViaHeader.NAME);
|
||||||
|
// try {
|
||||||
// requestURI.setHost(viaHeader.getHost());
|
// requestURI.setHost(viaHeader.getHost());
|
||||||
|
// } catch (Exception e) {
|
||||||
|
// e.printStackTrace();
|
||||||
|
// }
|
||||||
// requestURI.setPort(viaHeader.getPort());
|
// requestURI.setPort(viaHeader.getPort());
|
||||||
// clientDialog.sendAck(ackRequest);
|
// clientDialog.sendAck(ackRequest);
|
||||||
|
|
||||||
Dialog dialog = evt.getDialog();
|
Dialog dialog = evt.getDialog();
|
||||||
Request reqAck =dialog.createAck(1L);
|
CSeqHeader cseq = (CSeqHeader) response.getHeader(CSeqHeader.NAME);
|
||||||
|
Request reqAck = dialog.createAck(cseq.getSeqNumber());
|
||||||
|
|
||||||
|
SipURI requestURI = (SipURI) reqAck.getRequestURI();
|
||||||
|
ViaHeader viaHeader = (ViaHeader) response.getHeader(ViaHeader.NAME);
|
||||||
|
// String viaHost =viaHeader.getHost();
|
||||||
|
//getHost()函数取回的IP地址是“[xxx.xxx.xxx.xxx:yyyy]”的格式,需用正则表达式截取为“xxx.xxx.xxx.xxx"格式
|
||||||
|
// Pattern p = Pattern.compile("(?<=//|)((\\w)+\\.)+\\w+");
|
||||||
|
// Matcher matcher = p.matcher(viaHeader.getHost());
|
||||||
|
// if (matcher.find()) {
|
||||||
|
// requestURI.setHost(matcher.group());
|
||||||
|
// }
|
||||||
|
requestURI.setHost(viaHeader.getHost());
|
||||||
|
requestURI.setPort(viaHeader.getPort());
|
||||||
|
reqAck.setRequestURI(requestURI);
|
||||||
dialog.sendAck(reqAck);
|
dialog.sendAck(reqAck);
|
||||||
}
|
}
|
||||||
} catch (InvalidArgumentException | SipException e) {
|
} catch (InvalidArgumentException | SipException e) {
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.genersoft.iot.vmp.gb28181.transmit.response.ISIPResponseProcessor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:暂不支持的消息响应处理器
|
* @Description:暂不支持的消息响应处理器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月3日 下午5:32:59
|
* @date: 2020年5月3日 下午5:32:59
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -7,7 +7,7 @@ import java.util.Locale;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:时间工具类,主要处理ISO 8601格式转换
|
* @Description:时间工具类,主要处理ISO 8601格式转换
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 下午3:24:42
|
* @date: 2020年5月8日 下午3:24:42
|
||||||
*/
|
*/
|
||||||
public class DateUtil {
|
public class DateUtil {
|
||||||
|
|
|
@ -18,11 +18,11 @@ import com.genersoft.iot.vmp.gb28181.transmit.cmd.impl.SIPCommander;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:针对 ZLMediaServer的hook事件监听
|
* @Description:针对 ZLMediaServer的hook事件监听
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月8日 上午10:46:48
|
* @date: 2020年5月8日 上午10:46:48
|
||||||
*/
|
*/
|
||||||
@RestController
|
@RestController
|
||||||
@RequestMapping("/hook/zlm")
|
@RequestMapping("/index/hook")
|
||||||
public class ZLMHttpHookListener {
|
public class ZLMHttpHookListener {
|
||||||
|
|
||||||
private final static Logger logger = LoggerFactory.getLogger(ZLMHttpHookListener.class);
|
private final static Logger logger = LoggerFactory.getLogger(ZLMHttpHookListener.class);
|
||||||
|
|
|
@ -6,7 +6,7 @@ import com.genersoft.iot.vmp.gb28181.bean.Device;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:视频设备数据存储接口
|
* @Description:视频设备数据存储接口
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午2:14:31
|
* @date: 2020年5月6日 下午2:14:31
|
||||||
*/
|
*/
|
||||||
public interface IVideoManagerStorager {
|
public interface IVideoManagerStorager {
|
||||||
|
|
|
@ -8,7 +8,7 @@ import com.genersoft.iot.vmp.conf.VManagerConfig;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:视频设备数据存储工厂,根据存储策略,返回对应的存储器
|
* @Description:视频设备数据存储工厂,根据存储策略,返回对应的存储器
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午2:15:16
|
* @date: 2020年5月6日 下午2:15:16
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -11,7 +11,7 @@ import com.genersoft.iot.vmp.storager.IVideoManagerStorager;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:视频设备数据存储-jdbc实现
|
* @Description:视频设备数据存储-jdbc实现
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午2:28:12
|
* @date: 2020年5月6日 下午2:28:12
|
||||||
*/
|
*/
|
||||||
@Component("jdbcStorager")
|
@Component("jdbcStorager")
|
||||||
|
|
|
@ -13,7 +13,7 @@ import com.genersoft.iot.vmp.utils.redis.RedisUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:视频设备数据存储-redis实现
|
* @Description:视频设备数据存储-redis实现
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午2:31:42
|
* @date: 2020年5月6日 下午2:31:42
|
||||||
*/
|
*/
|
||||||
@Component("redisStorager")
|
@Component("redisStorager")
|
||||||
|
|
|
@ -7,7 +7,7 @@ import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:spring bean获取工厂,获取spring中的已初始化的bean
|
* @Description:spring bean获取工厂,获取spring中的已初始化的bean
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2019年6月25日 下午4:51:52
|
* @date: 2019年6月25日 下午4:51:52
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -10,7 +10,7 @@ import com.alibaba.fastjson.serializer.SerializerFeature;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:使用fastjson实现redis的序列化
|
* @Description:使用fastjson实现redis的序列化
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午8:40:11
|
* @date: 2020年5月6日 下午8:40:11
|
||||||
*/
|
*/
|
||||||
public class FastJsonRedisSerializer<T> implements RedisSerializer<T> {
|
public class FastJsonRedisSerializer<T> implements RedisSerializer<T> {
|
||||||
|
|
|
@ -13,7 +13,7 @@ import org.springframework.util.CollectionUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Description:Redis工具类
|
* @Description:Redis工具类
|
||||||
* @author: swwheihei
|
* @author: songww
|
||||||
* @date: 2020年5月6日 下午8:27:29
|
* @date: 2020年5月6日 下午8:27:29
|
||||||
*/
|
*/
|
||||||
@Component
|
@Component
|
||||||
|
|
|
@ -26,7 +26,8 @@ spring:
|
||||||
server:
|
server:
|
||||||
port: 8080
|
port: 8080
|
||||||
sip:
|
sip:
|
||||||
ip: 10.200.64.63
|
# ip: 10.200.64.63
|
||||||
|
ip: 192.168.0.102
|
||||||
port: 5060
|
port: 5060
|
||||||
# 根据国标6.1.2中规定,domain宜采用ID统一编码的前十位编码。国标附录D中定义前8位为中心编码(由省级、市级、区级、基层编号组成,参照GB/T 2260-2007)
|
# 根据国标6.1.2中规定,domain宜采用ID统一编码的前十位编码。国标附录D中定义前8位为中心编码(由省级、市级、区级、基层编号组成,参照GB/T 2260-2007)
|
||||||
# 后两位为行业编码,定义参照附录D.3
|
# 后两位为行业编码,定义参照附录D.3
|
||||||
|
@ -34,7 +35,8 @@ sip:
|
||||||
domain: 3701020049
|
domain: 3701020049
|
||||||
id: 37010200492000000001
|
id: 37010200492000000001
|
||||||
# 默认设备认证密码,后续扩展使用设备单独密码
|
# 默认设备认证密码,后续扩展使用设备单独密码
|
||||||
password: admin
|
password: admin123
|
||||||
media:
|
media:
|
||||||
ip: 10.200.64.88
|
# ip: 10.200.64.88
|
||||||
|
ip: 192.168.0.102
|
||||||
port: 10000
|
port: 10000
|
Binary file not shown.
Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 169 KiB |
Loading…
Reference in New Issue