mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
refactor gb28181, supporting SIP server enable, multiplex, API interface
This commit is contained in:
parent
8b4f84e336
commit
c99fb99ab6
15 changed files with 2524 additions and 980 deletions
|
@ -6,28 +6,35 @@ daemon off;
|
||||||
pid ./objs/srs28181.pid;
|
pid ./objs/srs28181.pid;
|
||||||
srs_log_file ./objs/srs28181.log;
|
srs_log_file ./objs/srs28181.log;
|
||||||
srs_log_tank console;
|
srs_log_tank console;
|
||||||
|
|
||||||
|
http_api {
|
||||||
|
enabled on;
|
||||||
|
listen 1985;
|
||||||
|
}
|
||||||
|
|
||||||
stream_caster {
|
stream_caster {
|
||||||
enabled on;
|
enabled on;
|
||||||
caster gb28181;
|
caster gb28181;
|
||||||
|
|
||||||
#rtmp输出地址,可以参数化
|
#转发流到rtmp服务器地址与端口
|
||||||
#[stream] 代表客户端sip设备编号
|
output 127.0.0.1:1935;
|
||||||
#[timestamp] 时间戳
|
|
||||||
output rtmp://127.0.0.1/live/[stream];
|
|
||||||
#sip监听udp端口
|
|
||||||
listen 15060;
|
|
||||||
|
|
||||||
#服务器主机号,可以域名或ip地址
|
#接收设备端rtp流的多路复用端口
|
||||||
#也就是设备端将媒体发送的地址,如果是服务器是内外网
|
listen 9000;
|
||||||
#需要写外网地址
|
|
||||||
host 192.168.1.27;
|
|
||||||
|
|
||||||
#服务器端编号
|
#rtp接收监听端口范围,最小值
|
||||||
#设备端配置编号需要与该值一致,否则无法注册
|
rtp_port_min 58200;
|
||||||
serial 34020000002000000001;
|
#rtp接收监听端口范围,最大值
|
||||||
|
rtp_port_max 58300;
|
||||||
|
|
||||||
#服务器端域
|
#是否等待关键帧之后,再转发,
|
||||||
realm 3402000000;
|
#off:不需等待,直接转发
|
||||||
|
#on:等第一个关键帧后,再转发
|
||||||
|
wait_keyframe off;
|
||||||
|
|
||||||
|
#rtp包空闲等待时间,如果指定时间没有收到任何包
|
||||||
|
#rtp监听连接自动停止,发送BYE命令
|
||||||
|
rtp_idle_timeout 30;
|
||||||
|
|
||||||
#是否转发音频流
|
#是否转发音频流
|
||||||
#目前只支持aac格式,所以需要设备支持aac格式
|
#目前只支持aac格式,所以需要设备支持aac格式
|
||||||
|
@ -41,32 +48,50 @@ stream_caster {
|
||||||
#因为flash,只支持11025 22050 44100
|
#因为flash,只支持11025 22050 44100
|
||||||
audio_enable on;
|
audio_enable on;
|
||||||
|
|
||||||
#服务端发送ack后,接收回应的超时时间,单位为秒
|
#服务器主机号,可以域名或ip地址
|
||||||
#如果指定时间没有回应,认为失败
|
#也就是设备端将媒体发送的地址,如果是服务器是内外网
|
||||||
ack_timeout 30;
|
#需要写外网地址,
|
||||||
|
#调用api创建stream session时返回ip地址也是host
|
||||||
|
host 192.168.1.27;
|
||||||
|
|
||||||
#设备心跳维持时间,如果指定时间内(秒)没有接收一个心跳
|
sip {
|
||||||
#认为设备离线
|
#是否启用srs内部sip信令
|
||||||
keepalive_timeout 30;
|
#为on信令走srs, off 只转发ps流
|
||||||
|
enabled on;
|
||||||
|
|
||||||
#是否等待关键帧之后,再转发,
|
#sip监听udp端口
|
||||||
#off:不需等待,直接转发
|
listen 15060;
|
||||||
#on:等第一个关键帧后,再转发
|
|
||||||
wait_keyframe off;
|
|
||||||
|
|
||||||
#日志打印是否打印sip信息
|
#服务器端编号
|
||||||
#off:不打印
|
#设备端配置编号需要与该值一致,否则无法注册
|
||||||
#on:打印接收或发送sip命令信息
|
serial 34020000002000000001;
|
||||||
print_sip_message off;
|
|
||||||
|
|
||||||
#rtp包空闲等待时间,如果指定时间没有收到任何包
|
#服务器端域
|
||||||
#rtp监听连接自动停止,发送BYE命令
|
realm 3402000000;
|
||||||
rtp_idle_timeout 30;
|
|
||||||
|
|
||||||
#rtp接收监听端口范围,最小值
|
#服务端发送ack后,接收回应的超时时间,单位为秒
|
||||||
rtp_port_min 58200;
|
#如果指定时间没有回应,认为失败
|
||||||
#rtp接收监听端口范围,最大值
|
ack_timeout 30;
|
||||||
rtp_port_max 58300;
|
|
||||||
|
#设备心跳维持时间,如果指定时间内(秒)没有接收一个心跳
|
||||||
|
#认为设备离线
|
||||||
|
keepalive_timeout 30;
|
||||||
|
|
||||||
|
#日志打印是否打印sip信息
|
||||||
|
#off:不打印
|
||||||
|
#on:打印接收或发送sip命令信息
|
||||||
|
print_sip_message off;
|
||||||
|
|
||||||
|
#注册之后是否自动给设备端发送invite
|
||||||
|
#on: 是 off 不是,需要通过api控制
|
||||||
|
auto_play on;
|
||||||
|
|
||||||
|
#设备将流发送的端口,是否固定
|
||||||
|
#on 发送流到多路复用端口 如9000
|
||||||
|
#off 自动从rtp_mix_port - rtp_max_port 之间的值中
|
||||||
|
#选一个可以用的端口
|
||||||
|
invite_port_fixed on;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
vhost __defaultVhost__ {
|
vhost __defaultVhost__ {
|
||||||
}
|
}
|
||||||
|
|
2
trunk/configure
vendored
2
trunk/configure
vendored
|
@ -257,7 +257,7 @@ if [ $SRS_EXPORT_LIBRTMP_PROJECT = NO ]; then
|
||||||
"srs_app_mpegts_udp" "srs_app_rtsp" "srs_app_listener" "srs_app_async_call"
|
"srs_app_mpegts_udp" "srs_app_rtsp" "srs_app_listener" "srs_app_async_call"
|
||||||
"srs_app_caster_flv" "srs_app_process" "srs_app_ng_exec"
|
"srs_app_caster_flv" "srs_app_process" "srs_app_ng_exec"
|
||||||
"srs_app_hourglass" "srs_app_dash" "srs_app_fragment" "srs_app_dvr"
|
"srs_app_hourglass" "srs_app_dash" "srs_app_fragment" "srs_app_dvr"
|
||||||
"srs_app_coworkers" "srs_app_hybrid" "srs_app_gb28181")
|
"srs_app_coworkers" "srs_app_hybrid" "srs_app_gb28181" "srs_app_gb28181_sip")
|
||||||
DEFINES=""
|
DEFINES=""
|
||||||
# add each modules for app
|
# add each modules for app
|
||||||
for SRS_MODULE in ${SRS_MODULES[*]}; do
|
for SRS_MODULE in ${SRS_MODULES[*]}; do
|
||||||
|
|
|
@ -2160,6 +2160,10 @@ srs_error_t SrsConfig::global_to_json(SrsJsonObject* obj)
|
||||||
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
||||||
} else if (sdir->name == "print_sip_message") {
|
} else if (sdir->name == "print_sip_message") {
|
||||||
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
||||||
|
} else if (sdir->name == "invite_port_fixed") {
|
||||||
|
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
||||||
|
} else if (sdir->name == "auto_play") {
|
||||||
|
sobj->set(sdir->name, sdir->dumps_arg0_to_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -3672,12 +3676,24 @@ srs_error_t SrsConfig::check_normal_config()
|
||||||
string n = conf->name;
|
string n = conf->name;
|
||||||
if (n != "enabled" && n != "caster" && n != "output"
|
if (n != "enabled" && n != "caster" && n != "output"
|
||||||
&& n != "listen" && n != "rtp_port_min" && n != "rtp_port_max"
|
&& n != "listen" && n != "rtp_port_min" && n != "rtp_port_max"
|
||||||
&& n != "rtp_idle_timeout" && n != "ack_timeout" && n != "keepalive_timeout"
|
&& n != "rtp_idle_timeout" && n != "sip"
|
||||||
&& n != "host" && n != "serial" && n != "realm"
|
|
||||||
&& n != "audio_enable" && n != "wait_keyframe"
|
&& n != "audio_enable" && n != "wait_keyframe"
|
||||||
&& n != "print_sip_message") {
|
&& n != "host") {
|
||||||
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal stream_caster.%s", n.c_str());
|
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal stream_caster.%s", n.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (n == "sip") {
|
||||||
|
for (int j = 0; j < (int)conf->directives.size(); j++) {
|
||||||
|
string m = conf->at(j)->name;
|
||||||
|
if (m != "enabled" && m != "listen"
|
||||||
|
&& m != "ack_timeout" && m != "keepalive_timeout"
|
||||||
|
&& m != "host" && m != "serial" && m != "realm"
|
||||||
|
&& m != "print_sip_message" && m != "auto_play"
|
||||||
|
&& m != "invite_port_fixed") {
|
||||||
|
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal stream_caster.%s", m.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4207,7 +4223,7 @@ int SrsConfig::get_stream_caster_rtp_port_max(SrsConfDirective* conf)
|
||||||
return ::atoi(conf->arg0().c_str());
|
return ::atoi(conf->arg0().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
int SrsConfig::get_stream_caster_gb28181_rtp_ide_timeout(SrsConfDirective* conf)
|
int SrsConfig::get_stream_caster_gb28181_rtp_idle_timeout(SrsConfDirective* conf)
|
||||||
{
|
{
|
||||||
static int DEFAULT = 30;
|
static int DEFAULT = 30;
|
||||||
|
|
||||||
|
@ -4215,7 +4231,7 @@ int SrsConfig::get_stream_caster_gb28181_rtp_ide_timeout(SrsConfDirective* conf)
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
conf = conf->get("rtp_ide_timeout");
|
conf = conf->get("rtp_idle_timeout");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
@ -4231,6 +4247,11 @@ int SrsConfig::get_stream_caster_gb28181_ack_timeout(SrsConfDirective* conf)
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
conf = conf->get("ack_timeout");
|
conf = conf->get("ack_timeout");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4247,6 +4268,11 @@ int SrsConfig::get_stream_caster_gb28181_keepalive_timeout(SrsConfDirective* con
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
conf = conf->get("keepalive_timeout");
|
conf = conf->get("keepalive_timeout");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4257,7 +4283,7 @@ int SrsConfig::get_stream_caster_gb28181_keepalive_timeout(SrsConfDirective* con
|
||||||
|
|
||||||
string SrsConfig::get_stream_caster_gb28181_host(SrsConfDirective* conf)
|
string SrsConfig::get_stream_caster_gb28181_host(SrsConfDirective* conf)
|
||||||
{
|
{
|
||||||
static string DEFAULT = "127.0.0.1";
|
static string DEFAULT = "";
|
||||||
|
|
||||||
if (!conf) {
|
if (!conf) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4279,6 +4305,11 @@ string SrsConfig::get_stream_caster_gb28181_serial(SrsConfDirective* conf)
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
conf = conf->get("serial");
|
conf = conf->get("serial");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4295,6 +4326,11 @@ string SrsConfig::get_stream_caster_gb28181_realm(SrsConfDirective* conf)
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
conf = conf->get("realm");
|
conf = conf->get("realm");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4327,6 +4363,11 @@ bool SrsConfig::get_stream_caster_gb28181_print_sip_message(SrsConfDirective* co
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
conf = conf->get("print_sip_message");
|
conf = conf->get("print_sip_message");
|
||||||
if (!conf || conf->arg0().empty()) {
|
if (!conf || conf->arg0().empty()) {
|
||||||
return DEFAULT;
|
return DEFAULT;
|
||||||
|
@ -4351,6 +4392,95 @@ bool SrsConfig::get_stream_caster_gb28181_wait_keyframe(SrsConfDirective* conf)
|
||||||
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool SrsConfig::get_stream_caster_gb28181_sip_enable(SrsConfDirective* conf)
|
||||||
|
{
|
||||||
|
static bool DEFAULT = false;
|
||||||
|
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("enabled");
|
||||||
|
if (!conf || conf->arg0().empty()) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SrsConfig::get_stream_caster_gb28181_sip_auto_play(SrsConfDirective* conf)
|
||||||
|
{
|
||||||
|
static bool DEFAULT = false;
|
||||||
|
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("auto_play");
|
||||||
|
if (!conf || conf->arg0().empty()) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
int SrsConfig::get_stream_caster_gb28181_sip_listen(SrsConfDirective* conf)
|
||||||
|
{
|
||||||
|
static int DEFAULT = 0;
|
||||||
|
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("listen");
|
||||||
|
if (!conf || conf->arg0().empty()) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ::atoi(conf->arg0().c_str());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SrsConfig::get_stream_caster_gb28181_sip_invite_port_fixed(SrsConfDirective* conf)
|
||||||
|
{
|
||||||
|
static bool DEFAULT = true;
|
||||||
|
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("sip");
|
||||||
|
if (!conf) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
conf = conf->get("invite_port_fixed");
|
||||||
|
if (!conf || conf->arg0().empty()) {
|
||||||
|
return DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SRS_CONF_PERFER_FALSE(conf->arg0());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
SrsConfDirective* SrsConfig::get_vhost(string vhost, bool try_default_vhost)
|
SrsConfDirective* SrsConfig::get_vhost(string vhost, bool try_default_vhost)
|
||||||
{
|
{
|
||||||
srs_assert(root);
|
srs_assert(root);
|
||||||
|
|
|
@ -486,7 +486,7 @@ public:
|
||||||
// Get the max udp port for rtp of stream caster rtsp.
|
// Get the max udp port for rtp of stream caster rtsp.
|
||||||
virtual int get_stream_caster_rtp_port_max(SrsConfDirective* conf);
|
virtual int get_stream_caster_rtp_port_max(SrsConfDirective* conf);
|
||||||
|
|
||||||
virtual int get_stream_caster_gb28181_rtp_ide_timeout(SrsConfDirective* conf);
|
virtual int get_stream_caster_gb28181_rtp_idle_timeout(SrsConfDirective* conf);
|
||||||
virtual int get_stream_caster_gb28181_ack_timeout(SrsConfDirective* conf);
|
virtual int get_stream_caster_gb28181_ack_timeout(SrsConfDirective* conf);
|
||||||
virtual int get_stream_caster_gb28181_keepalive_timeout(SrsConfDirective* conf);
|
virtual int get_stream_caster_gb28181_keepalive_timeout(SrsConfDirective* conf);
|
||||||
virtual bool get_stream_caster_gb28181_audio_enable(SrsConfDirective* conf);
|
virtual bool get_stream_caster_gb28181_audio_enable(SrsConfDirective* conf);
|
||||||
|
@ -495,6 +495,11 @@ public:
|
||||||
virtual std::string get_stream_caster_gb28181_realm(SrsConfDirective* conf);
|
virtual std::string get_stream_caster_gb28181_realm(SrsConfDirective* conf);
|
||||||
virtual bool get_stream_caster_gb28181_print_sip_message(SrsConfDirective* conf);
|
virtual bool get_stream_caster_gb28181_print_sip_message(SrsConfDirective* conf);
|
||||||
virtual bool get_stream_caster_gb28181_wait_keyframe(SrsConfDirective* conf);
|
virtual bool get_stream_caster_gb28181_wait_keyframe(SrsConfDirective* conf);
|
||||||
|
virtual bool get_stream_caster_gb28181_sip_enable(SrsConfDirective* conf);
|
||||||
|
virtual bool get_stream_caster_gb28181_sip_auto_play(SrsConfDirective* conf);
|
||||||
|
virtual int get_stream_caster_gb28181_sip_listen(SrsConfDirective* conf);
|
||||||
|
virtual bool get_stream_caster_gb28181_sip_invite_port_fixed(SrsConfDirective* conf);
|
||||||
|
|
||||||
|
|
||||||
// vhost specified section
|
// vhost specified section
|
||||||
public:
|
public:
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -29,6 +29,7 @@
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <queue>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
#include <srs_app_st.hpp>
|
#include <srs_app_st.hpp>
|
||||||
|
@ -38,16 +39,14 @@
|
||||||
#include <srs_kernel_stream.hpp>
|
#include <srs_kernel_stream.hpp>
|
||||||
#include <srs_app_log.hpp>
|
#include <srs_app_log.hpp>
|
||||||
#include <srs_kernel_file.hpp>
|
#include <srs_kernel_file.hpp>
|
||||||
|
#include <srs_protocol_json.hpp>
|
||||||
|
#include <srs_app_gb28181_sip.hpp>
|
||||||
|
|
||||||
|
#define RTP_PORT_MODE_FIXED "fixed"
|
||||||
|
#define RTP_PORT_MODE_RANDOM "random"
|
||||||
|
|
||||||
class SrsStSocket;
|
|
||||||
class SrsRtpConn;
|
|
||||||
class SrsRtspConn;
|
|
||||||
class SrsRtspStack;
|
|
||||||
class SrsRtspCaster;
|
|
||||||
class SrsConfDirective;
|
class SrsConfDirective;
|
||||||
class SrsRtpPacket;
|
class SrsRtpPacket;
|
||||||
class SrsRequest;
|
|
||||||
class SrsStSocket;
|
|
||||||
class SrsRtmpClient;
|
class SrsRtmpClient;
|
||||||
class SrsRawH264Stream;
|
class SrsRawH264Stream;
|
||||||
class SrsRawAacStream;
|
class SrsRawAacStream;
|
||||||
|
@ -58,52 +57,16 @@ class SrsSimpleStream;
|
||||||
class SrsPithyPrint;
|
class SrsPithyPrint;
|
||||||
class SrsSimpleRtmpClient;
|
class SrsSimpleRtmpClient;
|
||||||
class SrsSipStack;
|
class SrsSipStack;
|
||||||
class SrsGb28181Caster;
|
class SrsGb28181Manger;
|
||||||
class SrsRtspJitter;
|
class SrsRtspJitter;
|
||||||
class SrsRtspAudioCache;
|
|
||||||
class SrsSipRequest;
|
class SrsSipRequest;
|
||||||
class SrsGb28181Conn;
|
class SrsGb28181RtmpMuxer;
|
||||||
class SrsGb28281ClientInfo;
|
class SrsGb28181Config;
|
||||||
|
class SrsGb28181PsRtpProcessor;
|
||||||
/* gb28181 program stream struct define
|
class SrsGb28181SipService;
|
||||||
|
class SrsGb28181StreamChannel;
|
||||||
*/
|
|
||||||
|
|
||||||
struct SrsPsPacketStartCode
|
|
||||||
{
|
|
||||||
uint8_t start_code[3];
|
|
||||||
uint8_t stream_id[1];
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SrsPsPacketHeader
|
|
||||||
{
|
|
||||||
SrsPsPacketStartCode start;// 4
|
|
||||||
uint8_t info[9];
|
|
||||||
uint8_t stuffing_length;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SrsPsPacketBBHeader
|
|
||||||
{
|
|
||||||
SrsPsPacketStartCode start;
|
|
||||||
uint16_t length;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SrsPsePacket
|
|
||||||
{
|
|
||||||
SrsPsPacketStartCode start;
|
|
||||||
uint16_t length;
|
|
||||||
uint8_t info[2];
|
|
||||||
uint8_t stuffing_length;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SrsPsMapPacket
|
|
||||||
{
|
|
||||||
SrsPsPacketStartCode start;
|
|
||||||
uint16_t length;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//ps rtp header packet parse
|
||||||
class SrsPsRtpPacket: public SrsRtpPacket
|
class SrsPsRtpPacket: public SrsRtpPacket
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -113,19 +76,115 @@ public:
|
||||||
virtual srs_error_t decode(SrsBuffer* stream);
|
virtual srs_error_t decode(SrsBuffer* stream);
|
||||||
};
|
};
|
||||||
|
|
||||||
// A rtp connection which transport a stream.
|
//randomly assigned ports receive gb28281 device streams
|
||||||
class SrsPsRtpConn: public ISrsUdpHandler
|
class SrsPsRtpListener: public ISrsUdpHandler
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
SrsUdpListener* listener;
|
||||||
|
SrsGb28181PsRtpProcessor* rtp_processor;
|
||||||
|
int _port;
|
||||||
|
public:
|
||||||
|
SrsPsRtpListener(SrsGb28181Config* c, int p, std::string s);
|
||||||
|
virtual ~SrsPsRtpListener();
|
||||||
|
public:
|
||||||
|
virtual int port();
|
||||||
|
virtual srs_error_t listen();
|
||||||
|
// Interface ISrsUdpHandler
|
||||||
|
public:
|
||||||
|
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
||||||
|
};
|
||||||
|
|
||||||
|
//multiplexing service, single port receiving all gb28281 device streams
|
||||||
|
class SrsGb28181RtpMuxService : public ISrsUdpHandler
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
SrsGb28181Config *config;
|
||||||
|
SrsGb28181PsRtpProcessor *rtp_processor;
|
||||||
|
public:
|
||||||
|
SrsGb28181RtpMuxService(SrsConfDirective* c);
|
||||||
|
virtual ~SrsGb28181RtpMuxService();
|
||||||
|
|
||||||
|
// Interface ISrsUdpHandler
|
||||||
|
public:
|
||||||
|
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
//process gb28281 RTP package, generate a completed PS stream data,
|
||||||
|
//call the PS stream parser, parse the original video and audio
|
||||||
|
class SrsGb28181PsRtpProcessor: public ISrsUdpHandler
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
SrsPithyPrint* pprint;
|
SrsPithyPrint* pprint;
|
||||||
SrsUdpListener* listener;
|
SrsGb28181Config* config;
|
||||||
SrsGb28181Conn* gb28181;
|
std::map<std::string, SrsPsRtpPacket*> cache_ps_rtp_packet;
|
||||||
SrsPsRtpPacket* cache;
|
std::map<std::string, SrsPsRtpPacket*> pre_packet;
|
||||||
std::map<uint32_t, SrsSimpleStream*> cache_payload;
|
std::string channel_id;
|
||||||
std::string session_id;
|
bool auto_create_channel;
|
||||||
int _port;
|
public:
|
||||||
uint32_t pre_timestamp;
|
SrsGb28181PsRtpProcessor(SrsGb28181Config* c, std::string sid);
|
||||||
|
virtual ~SrsGb28181PsRtpProcessor();
|
||||||
|
private:
|
||||||
|
bool can_send_ps_av_packet();
|
||||||
|
void dispose();
|
||||||
|
// Interface ISrsUdpHandler
|
||||||
|
public:
|
||||||
|
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
||||||
|
};
|
||||||
|
|
||||||
|
//ps stream processing parsing interface
|
||||||
|
class ISrsPsStreamHander
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ISrsPsStreamHander();
|
||||||
|
virtual ~ISrsPsStreamHander();
|
||||||
|
public:
|
||||||
|
virtual srs_error_t on_rtp_video(SrsSimpleStream* stream, int64_t dts)=0;
|
||||||
|
virtual srs_error_t on_rtp_audio(SrsSimpleStream* stream, int64_t dts)=0;
|
||||||
|
};
|
||||||
|
|
||||||
|
//analysis of PS stream and
|
||||||
|
//extraction of H264 raw data and audio data
|
||||||
|
//then process the flow through PS stream hander,
|
||||||
|
//such as RTMP multiplexer, and composited into RTMP av stream
|
||||||
|
class SrsPsStreamDemixer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// gb28181 program stream struct define
|
||||||
|
struct SrsPsPacketStartCode
|
||||||
|
{
|
||||||
|
uint8_t start_code[3];
|
||||||
|
uint8_t stream_id[1];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SrsPsPacketHeader
|
||||||
|
{
|
||||||
|
SrsPsPacketStartCode start;// 4
|
||||||
|
uint8_t info[9];
|
||||||
|
uint8_t stuffing_length;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SrsPsPacketBBHeader
|
||||||
|
{
|
||||||
|
SrsPsPacketStartCode start;
|
||||||
|
uint16_t length;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SrsPsePacket
|
||||||
|
{
|
||||||
|
SrsPsPacketStartCode start;
|
||||||
|
uint16_t length;
|
||||||
|
uint8_t info[2];
|
||||||
|
uint8_t stuffing_length;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SrsPsMapPacket
|
||||||
|
{
|
||||||
|
SrsPsPacketStartCode start;
|
||||||
|
uint16_t length;
|
||||||
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
SrsFileWriter ps_fw;
|
SrsFileWriter ps_fw;
|
||||||
SrsFileWriter video_fw;
|
SrsFileWriter video_fw;
|
||||||
SrsFileWriter audio_fw;
|
SrsFileWriter audio_fw;
|
||||||
|
@ -133,117 +192,89 @@ private:
|
||||||
bool first_keyframe_flag;
|
bool first_keyframe_flag;
|
||||||
bool wait_first_keyframe;
|
bool wait_first_keyframe;
|
||||||
bool audio_enable;
|
bool audio_enable;
|
||||||
|
std::string channel_id;
|
||||||
|
|
||||||
|
ISrsPsStreamHander *hander;
|
||||||
public:
|
public:
|
||||||
SrsPsRtpConn(SrsGb28181Conn* r, int p, std::string sid, bool a, bool k);
|
SrsPsStreamDemixer(ISrsPsStreamHander *h, std::string sid, bool a, bool k);
|
||||||
virtual ~SrsPsRtpConn();
|
virtual ~SrsPsStreamDemixer();
|
||||||
|
|
||||||
private:
|
|
||||||
int64_t parse_ps_timestamp(const uint8_t* p);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool can_send_ps_av_packet();
|
bool can_send_ps_av_packet();
|
||||||
void dispose();
|
|
||||||
public:
|
public:
|
||||||
virtual int port();
|
int64_t parse_ps_timestamp(const uint8_t* p);
|
||||||
virtual srs_error_t listen();
|
virtual srs_error_t on_ps_stream(char* ps_data, int ps_size, uint32_t timestamp, uint32_t ssrc);
|
||||||
// Interface ISrsUdpHandler
|
|
||||||
public:
|
|
||||||
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
|
||||||
virtual srs_error_t on_ps_stream(char* ps_data, int ps_size, uint32_t timestamp);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class SrsGb28281ClientInfo {
|
|
||||||
public:
|
|
||||||
SrsGb28281ClientInfo();
|
|
||||||
virtual ~SrsGb28281ClientInfo();
|
|
||||||
|
|
||||||
public:
|
//RTMP multiplexer, which processes the raw H264 / AAC,
|
||||||
sockaddr* sock_from;
|
//then publish it to RTMP server
|
||||||
int sock_fromlen;
|
class SrsGb28181RtmpMuxer : public ISrsCoroutineHandler,
|
||||||
srs_netfd_t stfd;
|
public ISrsConnection, public ISrsPsStreamHander
|
||||||
SrsSipRequest *req;
|
|
||||||
};
|
|
||||||
|
|
||||||
enum Srs28181CtrlStatusType{
|
|
||||||
Srs28181Unkonw = 0,
|
|
||||||
Srs28181RegisterOk = 1,
|
|
||||||
Srs28181AliveOk = 2,
|
|
||||||
Srs28181InviteOk = 3,
|
|
||||||
Srs28181Trying = 4,
|
|
||||||
Srs28181Bye = 5,
|
|
||||||
};
|
|
||||||
|
|
||||||
class SrsGb28181Conn : public ISrsCoroutineHandler, public ISrsConnection
|
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
std::string output_template;
|
|
||||||
SrsPithyPrint* pprint;
|
SrsPithyPrint* pprint;
|
||||||
public:
|
SrsGb28181StreamChannel *channel;
|
||||||
Srs28181CtrlStatusType register_status;
|
int stream_idle_timeout;
|
||||||
Srs28181CtrlStatusType alive_status;
|
srs_utime_t recv_stream_time;
|
||||||
Srs28181CtrlStatusType invite_status;
|
|
||||||
srs_utime_t register_time;
|
|
||||||
srs_utime_t alive_time;
|
|
||||||
srs_utime_t invite_time;
|
|
||||||
srs_utime_t recv_rtp_time;
|
|
||||||
|
|
||||||
std::string rtmp_url;
|
|
||||||
int reg_expires;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::string session_id;
|
std::string channel_id;
|
||||||
// video stream.
|
std::string _rtmp_url;
|
||||||
int video_id;
|
std::string video_ssrc;
|
||||||
std::string video_codec;
|
std::string audio_ssrc;
|
||||||
SrsPsRtpConn* video_rtp;
|
|
||||||
// audio stream.
|
|
||||||
int audio_id;
|
|
||||||
std::string audio_codec;
|
|
||||||
int audio_sample_rate;
|
int audio_sample_rate;
|
||||||
int audio_channel;
|
int audio_channel;
|
||||||
SrsPsRtpConn* audio_rtp;
|
|
||||||
public:
|
SrsGb28181Manger* gb28181_manger;
|
||||||
SrsGb28281ClientInfo* info;
|
|
||||||
private:
|
|
||||||
SrsStSocket* skt;
|
|
||||||
SrsSipStack* sip;
|
|
||||||
SrsGb28181Caster* caster;
|
|
||||||
SrsCoroutine* trd;
|
SrsCoroutine* trd;
|
||||||
private:
|
SrsPsStreamDemixer* ps_demixer;
|
||||||
SrsSipRequest* req;
|
srs_cond_t wait_ps_queue;
|
||||||
|
|
||||||
SrsSimpleRtmpClient* sdk;
|
SrsSimpleRtmpClient* sdk;
|
||||||
SrsRtspJitter* vjitter;
|
SrsRtspJitter* vjitter;
|
||||||
SrsRtspJitter* ajitter;
|
SrsRtspJitter* ajitter;
|
||||||
private:
|
|
||||||
SrsRawH264Stream* avc;
|
SrsRawH264Stream* avc;
|
||||||
std::string h264_sps;
|
std::string h264_sps;
|
||||||
std::string h264_pps;
|
std::string h264_pps;
|
||||||
bool h264_sps_changed;
|
bool h264_sps_changed;
|
||||||
bool h264_pps_changed;
|
bool h264_pps_changed;
|
||||||
bool h264_sps_pps_sent;
|
bool h264_sps_pps_sent;
|
||||||
private:
|
|
||||||
SrsRawAacStream* aac;
|
SrsRawAacStream* aac;
|
||||||
std::string aac_specific_config;
|
std::string aac_specific_config;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SrsGb28181Conn(SrsGb28181Caster* c, std::string id);
|
std::queue<SrsPsRtpPacket*> ps_queue;
|
||||||
virtual ~SrsGb28181Conn();
|
|
||||||
|
public:
|
||||||
|
SrsGb28181RtmpMuxer(SrsGb28181Manger* m, std::string id, bool a, bool k);
|
||||||
|
virtual ~SrsGb28181RtmpMuxer();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
virtual srs_error_t serve();
|
virtual srs_error_t serve();
|
||||||
virtual std::string remote_ip();
|
virtual void stop();
|
||||||
virtual void set_request_info(SrsSipRequest *req);
|
|
||||||
virtual std::string get_session_id();
|
virtual std::string get_channel_id();
|
||||||
|
virtual void ps_packet_enqueue(SrsPsRtpPacket *pkt);
|
||||||
|
virtual void copy_channel(SrsGb28181StreamChannel *s);
|
||||||
|
virtual void set_channel_peer_ip(std::string ip);
|
||||||
|
virtual void set_channel_peer_port(int port);
|
||||||
|
virtual int channel_peer_port();
|
||||||
|
virtual std::string channel_peer_ip();
|
||||||
|
virtual void set_rtmp_url(std::string url);
|
||||||
|
virtual std::string rtmp_url();
|
||||||
|
virtual SrsGb28181StreamChannel get_channel();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
virtual srs_error_t do_cycle();
|
virtual srs_error_t do_cycle();
|
||||||
// internal methods
|
virtual void destroy();
|
||||||
public:
|
|
||||||
virtual srs_error_t start_rtp_listen(int port);
|
|
||||||
virtual srs_error_t stop_rtp_listen();
|
|
||||||
// Interface ISrsOneCycleThreadHandler
|
// Interface ISrsOneCycleThreadHandler
|
||||||
public:
|
public:
|
||||||
virtual srs_error_t cycle();
|
virtual srs_error_t cycle();
|
||||||
|
virtual std::string remote_ip();
|
||||||
public:
|
public:
|
||||||
virtual srs_error_t on_rtp_video(SrsSimpleStream* stream, int64_t dts, int keyframe);
|
virtual srs_error_t on_rtp_video(SrsSimpleStream* stream, int64_t dts);
|
||||||
virtual srs_error_t on_rtp_audio(SrsSimpleStream* stream, int64_t dts);
|
virtual srs_error_t on_rtp_audio(SrsSimpleStream* stream, int64_t dts);
|
||||||
private:
|
private:
|
||||||
virtual srs_error_t write_h264_sps_pps(uint32_t dts, uint32_t pts);
|
virtual srs_error_t write_h264_sps_pps(uint32_t dts, uint32_t pts);
|
||||||
|
@ -255,75 +286,152 @@ private:
|
||||||
virtual srs_error_t connect();
|
virtual srs_error_t connect();
|
||||||
// Close the connection to RTMP server.
|
// Close the connection to RTMP server.
|
||||||
virtual void close();
|
virtual void close();
|
||||||
|
public:
|
||||||
|
virtual void rtmp_close();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//system parameter configuration of gb28281 module,
|
||||||
|
//read file from configuration file to generate
|
||||||
class SrsGb28181Config
|
class SrsGb28181Config
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
std::string sip_host;
|
std::string host;
|
||||||
std::string sip_port;
|
|
||||||
std::string sip_serial;
|
|
||||||
std::string sip_realm;
|
|
||||||
int sip_ack_timeout;
|
|
||||||
int sip_keepalive_timeout;
|
|
||||||
int rtp_idle_timeout;
|
int rtp_idle_timeout;
|
||||||
bool audio_enable;
|
bool audio_enable;
|
||||||
|
bool wait_keyframe;
|
||||||
std::string output;
|
std::string output;
|
||||||
int rtp_port_min;
|
int rtp_port_min;
|
||||||
int rtp_port_max;
|
int rtp_port_max;
|
||||||
int listen_port;
|
int rtp_mux_port;
|
||||||
|
|
||||||
|
//sip config
|
||||||
|
int sip_port;
|
||||||
|
std::string sip_serial;
|
||||||
|
std::string sip_realm;
|
||||||
|
bool sip_enable;
|
||||||
|
int sip_ack_timeout;
|
||||||
|
int sip_keepalive_timeout;
|
||||||
bool print_sip_message;
|
bool print_sip_message;
|
||||||
bool wait_keyframe;
|
bool sip_auto_play;
|
||||||
|
bool sip_invite_port_fixed;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SrsGb28181Config(SrsConfDirective* c);
|
SrsGb28181Config(SrsConfDirective* c);
|
||||||
virtual ~SrsGb28181Config();
|
virtual ~SrsGb28181Config();
|
||||||
};
|
};
|
||||||
|
|
||||||
//gb28181 conn manager
|
class SrsGb28181StreamChannel
|
||||||
class SrsGb28181Caster : public ISrsUdpHandler
|
{
|
||||||
|
private:
|
||||||
|
std::string channel_id;
|
||||||
|
std::string port_mode;
|
||||||
|
std::string app;
|
||||||
|
std::string stream;
|
||||||
|
|
||||||
|
std::string ip;
|
||||||
|
int rtp_port;
|
||||||
|
int rtmp_port;
|
||||||
|
uint32_t ssrc;
|
||||||
|
|
||||||
|
//send rtp stream client local port
|
||||||
|
int rtp_peer_port;
|
||||||
|
//send rtp stream client local ip
|
||||||
|
std::string rtp_peer_ip;
|
||||||
|
|
||||||
|
public:
|
||||||
|
SrsGb28181StreamChannel();
|
||||||
|
virtual ~SrsGb28181StreamChannel();
|
||||||
|
|
||||||
|
std::string get_channel_id() const { return channel_id; }
|
||||||
|
std::string get_port_mode() const { return port_mode; }
|
||||||
|
std::string get_app() const { return app; }
|
||||||
|
std::string get_stream() const { return stream; }
|
||||||
|
std::string get_ip() const { return ip; }
|
||||||
|
int get_rtp_port() const { return rtp_port; }
|
||||||
|
int get_rtmp_port() const { return rtmp_port; }
|
||||||
|
uint32_t get_ssrc() const { return ssrc; }
|
||||||
|
uint32_t get_rtp_peer_port() const { return rtp_peer_port; }
|
||||||
|
std::string get_rtp_peer_ip() const { return rtp_peer_ip; }
|
||||||
|
|
||||||
|
void set_channel_id(const std::string &i) { channel_id = i; }
|
||||||
|
void set_port_mode(const std::string &p) { port_mode = p; }
|
||||||
|
void set_app(const std::string &a) { app = a; }
|
||||||
|
void set_stream(const std::string &s) { stream = s; }
|
||||||
|
void set_ip(const std::string &i) { ip = i; }
|
||||||
|
void set_rtp_port( const int &p) { rtp_port = p; }
|
||||||
|
void set_rtmp_port( const int &p) { rtmp_port = p; }
|
||||||
|
void set_ssrc( const int &s) { ssrc = s;}
|
||||||
|
void set_rtp_peer_ip( const std::string &p) { rtp_peer_ip = p; }
|
||||||
|
void set_rtp_peer_port( const int &s) { rtp_peer_port = s;}
|
||||||
|
|
||||||
|
void copy(const SrsGb28181StreamChannel *s);
|
||||||
|
void dumps(SrsJsonObject* obj);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
// Global singleton instance.
|
||||||
|
extern SrsGb28181Manger* _srs_gb28181;
|
||||||
|
|
||||||
|
//gb28181 module management, management of all RTMP multiplexers,
|
||||||
|
//random assignment of RTP listeners, and external control interfaces
|
||||||
|
class SrsGb28181Manger
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
SrsGb28181Config *config;
|
SrsGb28181Config *config;
|
||||||
// The key: port, value: whether used.
|
// The key: port, value: whether used.
|
||||||
std::map<int, bool> used_ports;
|
std::map<int, bool> used_ports;
|
||||||
SrsSipStack *sip;
|
std::map<uint32_t, SrsPsRtpListener*> rtp_pool;
|
||||||
srs_netfd_t lfd;
|
std::map<uint32_t, SrsGb28181RtmpMuxer*> rtmpmuxers_ssrc;
|
||||||
private:
|
std::map<std::string, SrsGb28181RtmpMuxer*> rtmpmuxers;
|
||||||
std::map<std::string, SrsGb28181Conn*> clients;
|
|
||||||
SrsCoroutineManager* manager;
|
SrsCoroutineManager* manager;
|
||||||
|
|
||||||
|
SrsGb28181SipService* sip_service;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SrsGb28181Caster(SrsConfDirective* c);
|
SrsGb28181Manger(SrsConfDirective* c);
|
||||||
virtual ~SrsGb28181Caster();
|
virtual ~SrsGb28181Manger();
|
||||||
|
|
||||||
|
public:
|
||||||
|
srs_error_t fetch_or_create_rtmpmuxer(std::string id, SrsGb28181RtmpMuxer** gb28181);
|
||||||
|
SrsGb28181RtmpMuxer* fetch_rtmpmuxer(std::string id);
|
||||||
|
SrsGb28181RtmpMuxer* fetch_rtmpmuxer_by_ssrc(uint32_t ssrc);
|
||||||
|
void rtmpmuxer_map_by_ssrc(SrsGb28181RtmpMuxer*muxer, uint32_t ssrc);
|
||||||
|
void rtmpmuxer_unmap_by_ssrc(uint32_t ssrc);
|
||||||
|
uint32_t generate_ssrc(std::string id);
|
||||||
|
uint32_t hash_code(std::string str);
|
||||||
|
|
||||||
|
void set_sip_service(SrsGb28181SipService *s) { sip_service = s; }
|
||||||
|
SrsGb28181SipService* get_sip_service() { return sip_service; }
|
||||||
|
|
||||||
|
public:
|
||||||
|
//stream channel api
|
||||||
|
uint32_t create_stream_channel(SrsGb28181StreamChannel *channel);
|
||||||
|
uint32_t delete_stream_channel(std::string id);
|
||||||
|
uint32_t queue_stream_channel(std::string id, SrsJsonArray* arr);
|
||||||
|
//sip api
|
||||||
|
uint32_t notify_sip_invite(std::string id, std::string ip, int port, uint32_t ssrc);
|
||||||
|
uint32_t notify_sip_bye(std::string id);
|
||||||
|
uint32_t notify_sip_raw_data(std::string id, std::string data);
|
||||||
|
uint32_t notify_sip_unregister(std::string id);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
srs_error_t fetch_or_create(SrsSipRequest* r, SrsGb28181Conn** gb28181);
|
void destroy();
|
||||||
virtual SrsGb28181Conn* fetch(const SrsSipRequest* r);
|
|
||||||
virtual void destroy();
|
|
||||||
public:
|
public:
|
||||||
// Alloc a rtp port from local ports pool.
|
// Alloc a rtp port from local ports pool.
|
||||||
// @param pport output the rtp port.
|
// @param pport output the rtp port.
|
||||||
virtual srs_error_t alloc_port(int* pport);
|
void alloc_port(int* pport);
|
||||||
// Free the alloced rtp port.
|
// Free the alloced rtp port.
|
||||||
virtual void free_port(int lpmin, int lpmax);
|
void free_port(int lpmin, int lpmax);
|
||||||
virtual srs_error_t initialize();
|
srs_error_t initialize();
|
||||||
|
|
||||||
virtual void set_stfd(srs_netfd_t fd);
|
SrsGb28181Config get_gb28181_config();
|
||||||
virtual SrsGb28181Config GetGb28181Config();
|
srs_error_t start_ps_rtp_listen(std::string id, int port);
|
||||||
|
void stop_rtp_listen(std::string id);
|
||||||
|
|
||||||
// Interface ISrsUdpHandler
|
|
||||||
public:
|
public:
|
||||||
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
void remove(SrsGb28181RtmpMuxer* conn);
|
||||||
private:
|
|
||||||
virtual srs_error_t on_udp_bytes(std::string host, int port, char* buf, int nb_buf, sockaddr* from, int fromlen);
|
|
||||||
// internal methods.
|
|
||||||
public:
|
|
||||||
virtual srs_error_t send_message(sockaddr* f, int l, std::stringstream& ss);
|
|
||||||
virtual srs_error_t send_bye(SrsSipRequest *req, sockaddr *f, int l);
|
|
||||||
virtual srs_error_t send_ack(SrsSipRequest *req, sockaddr *f, int l);
|
|
||||||
virtual srs_error_t send_invite(SrsSipRequest *req, sockaddr *f, int l, int port);
|
|
||||||
virtual srs_error_t send_status(SrsSipRequest *req, sockaddr *f, int l);
|
|
||||||
virtual void remove(SrsGb28181Conn* conn);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
471
trunk/src/app/srs_app_gb28181_sip.cpp
Normal file
471
trunk/src/app/srs_app_gb28181_sip.cpp
Normal file
|
@ -0,0 +1,471 @@
|
||||||
|
/**
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013-2020 Winlin
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <srs_app_gb28181_sip.hpp>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#include <netdb.h>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
#include <srs_app_config.hpp>
|
||||||
|
#include <srs_kernel_error.hpp>
|
||||||
|
|
||||||
|
#include <srs_kernel_log.hpp>
|
||||||
|
#include <srs_app_utility.hpp>
|
||||||
|
#include <srs_core_autofree.hpp>
|
||||||
|
#include <srs_kernel_buffer.hpp>
|
||||||
|
#include <srs_kernel_stream.hpp>
|
||||||
|
#include <srs_protocol_utility.hpp>
|
||||||
|
#include <srs_kernel_utility.hpp>
|
||||||
|
#include <srs_kernel_codec.hpp>
|
||||||
|
#include <srs_app_pithy_print.hpp>
|
||||||
|
#include <srs_sip_stack.hpp>
|
||||||
|
#include <srs_app_gb28181.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
SrsGb28181SipSession::SrsGb28181SipSession(SrsGb28181SipService *c, SrsSipRequest* r)
|
||||||
|
{
|
||||||
|
caster = c;
|
||||||
|
req = new SrsSipRequest();
|
||||||
|
req->copy(r);
|
||||||
|
|
||||||
|
_register_status = SrsGb28181SipSessionUnkonw;
|
||||||
|
_alive_status = SrsGb28181SipSessionUnkonw;
|
||||||
|
_invite_status = SrsGb28181SipSessionUnkonw;
|
||||||
|
_register_time = 0;
|
||||||
|
_alive_time = 0;
|
||||||
|
_invite_time = 0;
|
||||||
|
_recv_rtp_time = 0;
|
||||||
|
_reg_expires = 0;
|
||||||
|
|
||||||
|
_peer_ip = "";
|
||||||
|
_peer_port = 0;
|
||||||
|
|
||||||
|
_from = NULL;
|
||||||
|
_fromlen = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGb28181SipSession::~SrsGb28181SipSession()
|
||||||
|
{
|
||||||
|
srs_freep(req);
|
||||||
|
}
|
||||||
|
|
||||||
|
//gb28181 sip Service
|
||||||
|
SrsGb28181SipService::SrsGb28181SipService(SrsConfDirective* c)
|
||||||
|
{
|
||||||
|
// TODO: FIXME: support reload.
|
||||||
|
config = new SrsGb28181Config(c);
|
||||||
|
sip = new SrsSipStack();
|
||||||
|
|
||||||
|
if (_srs_gb28181){
|
||||||
|
_srs_gb28181->set_sip_service(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGb28181SipService::~SrsGb28181SipService()
|
||||||
|
{
|
||||||
|
destroy();
|
||||||
|
srs_freep(sip);
|
||||||
|
srs_freep(config);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SrsGb28181SipService::set_stfd(srs_netfd_t fd)
|
||||||
|
{
|
||||||
|
lfd = fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
srs_error_t SrsGb28181SipService::on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf)
|
||||||
|
{
|
||||||
|
char address_string[64];
|
||||||
|
char port_string[16];
|
||||||
|
if(getnameinfo(from, fromlen,
|
||||||
|
(char*)&address_string, sizeof(address_string),
|
||||||
|
(char*)&port_string, sizeof(port_string),
|
||||||
|
NI_NUMERICHOST|NI_NUMERICSERV)) {
|
||||||
|
return srs_error_new(ERROR_SYSTEM_IP_INVALID, "bad address");
|
||||||
|
}
|
||||||
|
std::string peer_ip = std::string(address_string);
|
||||||
|
int peer_port = atoi(port_string);
|
||||||
|
|
||||||
|
srs_error_t err = on_udp_sip(peer_ip, peer_port, buf, nb_buf, (sockaddr*)from, fromlen);
|
||||||
|
if (err != srs_success) {
|
||||||
|
return srs_error_wrap(err, "process udp");
|
||||||
|
}
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
srs_error_t SrsGb28181SipService::on_udp_sip(string peer_ip, int peer_port,
|
||||||
|
char* buf, int nb_buf, sockaddr* from, const int fromlen)
|
||||||
|
{
|
||||||
|
srs_error_t err = srs_success;
|
||||||
|
|
||||||
|
if (config->print_sip_message)
|
||||||
|
{
|
||||||
|
srs_trace("gb28181: request peer_ip=%s, peer_port=%d nbbuf=%d", peer_ip.c_str(), peer_port, nb_buf);
|
||||||
|
srs_trace("gb28181: request recv message=%s", buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nb_buf < 10) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsSipRequest* req = NULL;
|
||||||
|
|
||||||
|
if ((err = sip->parse_request(&req, buf, nb_buf)) != srs_success) {
|
||||||
|
return srs_error_wrap(err, "parse sip request");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config->print_sip_message)
|
||||||
|
{
|
||||||
|
srs_trace("gb28181: %s method=%s, uri=%s, version=%s ",
|
||||||
|
req->get_cmdtype_str().c_str(), req->method.c_str(), req->uri.c_str(), req->version.c_str());
|
||||||
|
srs_trace("gb28281: request client id=%s", req->sip_auth_id.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
req->peer_ip = peer_ip;
|
||||||
|
req->peer_port = peer_port;
|
||||||
|
SrsAutoFree(SrsSipRequest, req);
|
||||||
|
|
||||||
|
std::string session_id = req->sip_auth_id;
|
||||||
|
|
||||||
|
if (req->is_register()) {
|
||||||
|
std::vector<std::string> serial = srs_string_split(srs_string_replace(req->uri,"sip:", ""), "@");
|
||||||
|
if (serial.at(0) != config->sip_serial){
|
||||||
|
srs_trace("gb28181: client:%s request serial and server serial inconformity(%s:%s)",
|
||||||
|
req->sip_auth_id.c_str(), serial.at(0).c_str(), config->sip_serial.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
srs_trace("gb28181: request peer_ip=%s, peer_port=%d", peer_ip.c_str(), peer_port, nb_buf);
|
||||||
|
srs_trace("gb28181: request %s method=%s, uri=%s, version=%s ",
|
||||||
|
req->get_cmdtype_str().c_str(), req->method.c_str(), req->uri.c_str(), req->version.c_str());
|
||||||
|
srs_trace("gb28281: request client id=%s", req->sip_auth_id.c_str());
|
||||||
|
|
||||||
|
SrsGb28181SipSession* sip_session = create_sip_session(req);
|
||||||
|
if (!sip_session) {
|
||||||
|
srs_trace("gb28181: create sip session faild:%s", req->uri.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
send_status(req, from, fromlen);
|
||||||
|
sip_session->set_register_status(SrsGb28181SipSessionRegisterOk);
|
||||||
|
sip_session->set_register_time(srs_get_system_time());
|
||||||
|
sip_session->set_reg_expires(req->expires);
|
||||||
|
sip_session->set_sockaddr(from);
|
||||||
|
sip_session->set_sockaddr_len(fromlen);
|
||||||
|
sip_session->set_peer_ip(peer_ip);
|
||||||
|
sip_session->set_peer_port(peer_port);
|
||||||
|
}else if (req->is_message()) {
|
||||||
|
SrsGb28181SipSession* sip_session = fetch(session_id);
|
||||||
|
if (!sip_session || sip_session->register_status() == SrsGb28181SipSessionUnkonw){
|
||||||
|
srs_trace("gb28181: %s client not registered", req->sip_auth_id.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
//reponse status
|
||||||
|
send_status(req, from, fromlen);
|
||||||
|
sip_session->set_register_status(SrsGb28181SipSessionRegisterOk);
|
||||||
|
sip_session->set_register_time(srs_get_system_time());
|
||||||
|
sip_session->set_alive_status(SrsGb28181SipSessionAliveOk);
|
||||||
|
sip_session->set_alive_time(srs_get_system_time());
|
||||||
|
sip_session->set_sockaddr(from);
|
||||||
|
sip_session->set_sockaddr_len(fromlen);
|
||||||
|
sip_session->set_peer_port(peer_port);
|
||||||
|
sip_session->set_peer_ip(peer_ip);
|
||||||
|
|
||||||
|
//send invite, play client av
|
||||||
|
//start ps rtp listen, recv ps stream
|
||||||
|
if (config->sip_auto_play && sip_session->register_status() == SrsGb28181SipSessionRegisterOk &&
|
||||||
|
sip_session->alive_status() == SrsGb28181SipSessionAliveOk &&
|
||||||
|
sip_session->invite_status() == SrsGb28181SipSessionUnkonw)
|
||||||
|
{
|
||||||
|
//stop the possible stream and push a new stream
|
||||||
|
//send_bye(req, from, fromlen);
|
||||||
|
|
||||||
|
SrsGb28181StreamChannel ch;
|
||||||
|
ch.set_channel_id(session_id);
|
||||||
|
ch.set_ip(config->host);
|
||||||
|
ch.set_stream(session_id);
|
||||||
|
ch.set_app("live");
|
||||||
|
if (config->sip_invite_port_fixed){
|
||||||
|
ch.set_port_mode(RTP_PORT_MODE_FIXED);
|
||||||
|
}else {
|
||||||
|
ch.set_port_mode(RTP_PORT_MODE_RANDOM);
|
||||||
|
}
|
||||||
|
|
||||||
|
int code = _srs_gb28181->create_stream_channel(&ch);
|
||||||
|
if (code == ERROR_SUCCESS){
|
||||||
|
code = send_invite(req, ch.get_ip(),
|
||||||
|
ch.get_rtp_port(), ch.get_ssrc());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (code == ERROR_SUCCESS){
|
||||||
|
sip_session->set_invite_status(SrsGb28181SipSessionTrying);
|
||||||
|
sip_session->set_invite_time(srs_get_system_time());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}else if (req->is_invite()) {
|
||||||
|
SrsGb28181SipSession* sip_session = fetch(session_id);
|
||||||
|
|
||||||
|
srs_trace("gb28181: request peer_ip=%s, peer_port=%d", peer_ip.c_str(), peer_port, nb_buf);
|
||||||
|
srs_trace("gb28181: request %s method=%s, uri=%s, version=%s ",
|
||||||
|
req->get_cmdtype_str().c_str(), req->method.c_str(), req->uri.c_str(), req->version.c_str());
|
||||||
|
srs_trace("gb28281: request client id=%s", req->sip_auth_id.c_str());
|
||||||
|
|
||||||
|
if (!sip_session){
|
||||||
|
send_bye(req);
|
||||||
|
srs_trace("gb28181: %s client not registered", req->sip_auth_id.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sip_session->register_status() == SrsGb28181SipSessionUnkonw ||
|
||||||
|
sip_session->alive_status() == SrsGb28181SipSessionUnkonw) {
|
||||||
|
srs_trace("gb28181: %s client not registered or not alive", req->sip_auth_id.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req->cmdtype == SrsSipCmdRespone && req->status == "200") {
|
||||||
|
srs_trace("gb28181: INVITE response %s client status=%s", req->sip_auth_id.c_str(), req->status.c_str());
|
||||||
|
send_ack(req, from, fromlen);
|
||||||
|
sip_session->set_invite_status(SrsGb28181SipSessionInviteOk);
|
||||||
|
sip_session->set_invite_time(srs_get_system_time());
|
||||||
|
//Record tag and branch, which are required by the 'bye' command,
|
||||||
|
sip_session->set_request(req);
|
||||||
|
}else{
|
||||||
|
sip_session->set_invite_status(SrsGb28181SipSessionUnkonw);
|
||||||
|
sip_session->set_invite_time(0);
|
||||||
|
}
|
||||||
|
}else if (req->is_bye()) {
|
||||||
|
srs_trace("gb28181: request peer_ip=%s, peer_port=%d", peer_ip.c_str(), peer_port, nb_buf);
|
||||||
|
srs_trace("gb28181: request %s method=%s, uri=%s, version=%s ",
|
||||||
|
req->get_cmdtype_str().c_str(), req->method.c_str(), req->uri.c_str(), req->version.c_str());
|
||||||
|
srs_trace("gb28281: request client id=%s", req->sip_auth_id.c_str());
|
||||||
|
|
||||||
|
SrsGb28181SipSession* sip_session = fetch(session_id);
|
||||||
|
send_status(req, from, fromlen);
|
||||||
|
|
||||||
|
if (!sip_session){
|
||||||
|
srs_trace("gb28181: %s client not registered", req->sip_auth_id.c_str());
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
sip_session->set_invite_status(SrsGb28181SipSessionBye);
|
||||||
|
sip_session->set_invite_time(0);
|
||||||
|
|
||||||
|
}else{
|
||||||
|
srs_trace("gb28181: ingor request method=%s", req->method.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_message(sockaddr* from, int fromlen, std::stringstream& ss)
|
||||||
|
{
|
||||||
|
std::string str = ss.str();
|
||||||
|
if (config->print_sip_message)
|
||||||
|
srs_trace("gb28181: send_message:%s", str.c_str());
|
||||||
|
srs_assert(!str.empty());
|
||||||
|
|
||||||
|
int ret = srs_sendto(lfd, (char*)str.c_str(), (int)str.length(), from, fromlen, SRS_UTIME_NO_TIMEOUT);
|
||||||
|
if (ret <= 0){
|
||||||
|
srs_trace("gb28181: send_message falid (%d)", ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_ack(SrsSipRequest *req, sockaddr *f, int l)
|
||||||
|
{
|
||||||
|
srs_assert(req);
|
||||||
|
|
||||||
|
std::stringstream ss;
|
||||||
|
|
||||||
|
req->host = config->host;
|
||||||
|
req->host_port = config->sip_port;
|
||||||
|
req->realm = config->sip_realm;
|
||||||
|
req->serial = config->sip_serial;
|
||||||
|
|
||||||
|
sip->resp_ack(ss, req);
|
||||||
|
return send_message(f, l, ss);
|
||||||
|
}
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_status(SrsSipRequest *req, sockaddr *f, int l)
|
||||||
|
{
|
||||||
|
srs_assert(req);
|
||||||
|
|
||||||
|
std::stringstream ss;
|
||||||
|
|
||||||
|
req->host = config->host;
|
||||||
|
req->host_port = config->sip_port;
|
||||||
|
req->realm = config->sip_realm;
|
||||||
|
req->serial = config->sip_serial;
|
||||||
|
|
||||||
|
sip->resp_status(ss, req);
|
||||||
|
return send_message(f, l, ss);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_invite(SrsSipRequest *req, string ip, int port, uint32_t ssrc)
|
||||||
|
{
|
||||||
|
srs_assert(req);
|
||||||
|
|
||||||
|
SrsGb28181SipSession *sip_session = fetch(req->sip_auth_id);
|
||||||
|
|
||||||
|
if (!sip_session){
|
||||||
|
return ERROR_GB28181_SESSION_IS_NOTEXIST;
|
||||||
|
}
|
||||||
|
|
||||||
|
//if you are inviting or succeed in invite,
|
||||||
|
//you cannot invite again. you need to 'bye' and try again
|
||||||
|
if (sip_session->invite_status() == SrsGb28181SipSessionTrying ||
|
||||||
|
sip_session->invite_status() == SrsGb28181SipSessionInviteOk){
|
||||||
|
return ERROR_GB28281_SIP_IS_INVITING;
|
||||||
|
}
|
||||||
|
|
||||||
|
req->host = config->host;
|
||||||
|
req->host_port = config->sip_port;
|
||||||
|
req->realm = config->sip_realm;
|
||||||
|
req->serial = config->sip_serial;
|
||||||
|
|
||||||
|
std::stringstream ss;
|
||||||
|
sip->req_invite(ss, req, ip, port, ssrc);
|
||||||
|
|
||||||
|
if (send_message(sip_session->sockaddr_from(), sip_session->sockaddr_fromlen(), ss) <= 0)
|
||||||
|
{
|
||||||
|
return ERROR_GB28281_SIP_INVITE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
sip_session->set_invite_status(SrsGb28181SipSessionTrying);
|
||||||
|
|
||||||
|
return ERROR_SUCCESS;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_bye(SrsSipRequest *req)
|
||||||
|
{
|
||||||
|
srs_assert(req);
|
||||||
|
|
||||||
|
SrsGb28181SipSession *sip_session = fetch(req->sip_auth_id);
|
||||||
|
|
||||||
|
if (!sip_session){
|
||||||
|
return ERROR_GB28181_SESSION_IS_NOTEXIST;
|
||||||
|
}
|
||||||
|
|
||||||
|
//prame branch, from_tag, to_tag, call_id,
|
||||||
|
//The parameter of 'bye' must be the same as 'invite'
|
||||||
|
SrsSipRequest r = sip_session->request();
|
||||||
|
req->copy(&r);
|
||||||
|
|
||||||
|
req->host = config->host;
|
||||||
|
req->host_port = config->sip_port;
|
||||||
|
req->realm = config->sip_realm;
|
||||||
|
req->serial = config->sip_serial;
|
||||||
|
|
||||||
|
//get protocol stack
|
||||||
|
std::stringstream ss;
|
||||||
|
sip->req_bye(ss, req);
|
||||||
|
|
||||||
|
if (send_message(sip_session->sockaddr_from(), sip_session->sockaddr_fromlen(), ss) <= 0)
|
||||||
|
{
|
||||||
|
return ERROR_GB28281_SIP_BYE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ERROR_SUCCESS;
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int SrsGb28181SipService::send_sip_raw_data(SrsSipRequest *req, std::string data)
|
||||||
|
{
|
||||||
|
srs_assert(req);
|
||||||
|
|
||||||
|
SrsGb28181SipSession *sip_session = fetch(req->sip_auth_id);
|
||||||
|
|
||||||
|
if (!sip_session){
|
||||||
|
return ERROR_GB28181_SESSION_IS_NOTEXIST;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << data;
|
||||||
|
|
||||||
|
if (send_message(sip_session->sockaddr_from(), sip_session->sockaddr_fromlen(), ss) <= 0)
|
||||||
|
{
|
||||||
|
return ERROR_GB28281_SIP_BYE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ERROR_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGb28181SipSession* SrsGb28181SipService::create_sip_session(SrsSipRequest *req)
|
||||||
|
{
|
||||||
|
SrsGb28181SipSession *sess = NULL;
|
||||||
|
|
||||||
|
std::map<std::string, SrsGb28181SipSession*>::iterator it = sessions.find(req->sip_auth_id);
|
||||||
|
if (it == sessions.end()){
|
||||||
|
sess = new SrsGb28181SipSession(this, req);
|
||||||
|
}else{
|
||||||
|
return it->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
sessions[req->sip_auth_id] = sess;
|
||||||
|
return sess;
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGb28181SipSession* SrsGb28181SipService::fetch(std::string sid)
|
||||||
|
{
|
||||||
|
std::map<std::string, SrsGb28181SipSession*>::iterator it = sessions.find(sid);
|
||||||
|
if (it == sessions.end()){
|
||||||
|
return NULL;
|
||||||
|
}else{
|
||||||
|
return it->second;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SrsGb28181SipService::remove_session(std::string sid)
|
||||||
|
{
|
||||||
|
std::map<std::string, SrsGb28181SipSession*>::iterator it = sessions.find(sid);
|
||||||
|
if (it != sessions.end()){
|
||||||
|
srs_freep(it->second);
|
||||||
|
sessions.erase(it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void SrsGb28181SipService::destroy()
|
||||||
|
{
|
||||||
|
//destory all sip session
|
||||||
|
std::map<std::string, SrsGb28181SipSession*>::iterator it;
|
||||||
|
for (it = sessions.begin(); it != sessions.end(); ++it) {
|
||||||
|
srs_freep(it->second);
|
||||||
|
}
|
||||||
|
sessions.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
161
trunk/src/app/srs_app_gb28181_sip.hpp
Normal file
161
trunk/src/app/srs_app_gb28181_sip.hpp
Normal file
|
@ -0,0 +1,161 @@
|
||||||
|
/**
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013-2020 Winlin
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef SRS_APP_GB28181_SIP_HPP
|
||||||
|
#define SRS_APP_GB28181_SIP_HPP
|
||||||
|
|
||||||
|
#include <srs_core.hpp>
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
#include <srs_app_log.hpp>
|
||||||
|
#include <srs_sip_stack.hpp>
|
||||||
|
#include <srs_app_gb28181.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
class SrsConfDirective;
|
||||||
|
class SrsSipRequest;
|
||||||
|
class SrsGb28181Config;
|
||||||
|
class SrsSipStack;
|
||||||
|
class SrsGb28181SipService;
|
||||||
|
|
||||||
|
enum SrsGb28281SipSessionStatusType{
|
||||||
|
SrsGb28181SipSessionUnkonw = 0,
|
||||||
|
SrsGb28181SipSessionRegisterOk = 1,
|
||||||
|
SrsGb28181SipSessionAliveOk = 2,
|
||||||
|
SrsGb28181SipSessionInviteOk = 3,
|
||||||
|
SrsGb28181SipSessionTrying = 4,
|
||||||
|
SrsGb28181SipSessionBye = 5,
|
||||||
|
};
|
||||||
|
|
||||||
|
class SrsGb28181SipSession
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
//SrsSipRequest *req;
|
||||||
|
SrsGb28181SipService *caster;
|
||||||
|
std::string session_id;
|
||||||
|
private:
|
||||||
|
SrsGb28281SipSessionStatusType _register_status;
|
||||||
|
SrsGb28281SipSessionStatusType _alive_status;
|
||||||
|
SrsGb28281SipSessionStatusType _invite_status;
|
||||||
|
srs_utime_t _register_time;
|
||||||
|
srs_utime_t _alive_time;
|
||||||
|
srs_utime_t _invite_time;
|
||||||
|
srs_utime_t _recv_rtp_time;
|
||||||
|
int _reg_expires;
|
||||||
|
|
||||||
|
std::string _peer_ip;
|
||||||
|
int _peer_port;
|
||||||
|
|
||||||
|
sockaddr *_from;
|
||||||
|
int _fromlen;
|
||||||
|
SrsSipRequest *req;
|
||||||
|
|
||||||
|
public:
|
||||||
|
void set_register_status(SrsGb28281SipSessionStatusType s) { _register_status = s;}
|
||||||
|
void set_alive_status(SrsGb28281SipSessionStatusType s) { _alive_status = s;}
|
||||||
|
void set_invite_status(SrsGb28281SipSessionStatusType s) { _invite_status = s;}
|
||||||
|
void set_register_time(srs_utime_t t) { _register_time = t;}
|
||||||
|
void set_alive_time(srs_utime_t t) { _alive_time = t;}
|
||||||
|
void set_invite_time(srs_utime_t t) { _invite_time = t;}
|
||||||
|
void set_recv_rtp_time(srs_utime_t t) { _recv_rtp_time = t;}
|
||||||
|
void set_reg_expires(int e) { _reg_expires = e;}
|
||||||
|
void set_peer_ip(std::string i) { _peer_ip = i;}
|
||||||
|
void set_peer_port(int o) { _peer_port = o;}
|
||||||
|
void set_sockaddr(sockaddr *f) { _from = f;}
|
||||||
|
void set_sockaddr_len(int l) { _fromlen = l;}
|
||||||
|
void set_request(SrsSipRequest *r) { req->copy(r);}
|
||||||
|
|
||||||
|
SrsGb28281SipSessionStatusType register_status() { return _register_status;}
|
||||||
|
SrsGb28281SipSessionStatusType alive_status() { return _alive_status;}
|
||||||
|
SrsGb28281SipSessionStatusType invite_status() { return _invite_status;}
|
||||||
|
srs_utime_t register_time() { return _register_time;}
|
||||||
|
srs_utime_t alive_time() { return _alive_time;}
|
||||||
|
srs_utime_t invite_time() { return _invite_time;}
|
||||||
|
srs_utime_t recv_rtp_time() { return _recv_rtp_time;}
|
||||||
|
int reg_expires() { return _reg_expires;}
|
||||||
|
std::string peer_ip() { return _peer_ip;}
|
||||||
|
int peer_port() { return _peer_port;}
|
||||||
|
sockaddr* sockaddr_from() { return _from;}
|
||||||
|
int sockaddr_fromlen() { return _fromlen;}
|
||||||
|
SrsSipRequest request() { return *req;}
|
||||||
|
|
||||||
|
public:
|
||||||
|
SrsGb28181SipSession(SrsGb28181SipService *c, SrsSipRequest* r);
|
||||||
|
virtual ~SrsGb28181SipSession();
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class SrsGb28181SipService : public ISrsUdpHandler
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
SrsSipStack *sip;
|
||||||
|
SrsGb28181Config *config;
|
||||||
|
srs_netfd_t lfd;
|
||||||
|
|
||||||
|
std::map<std::string, SrsGb28181SipSession*> sessions;
|
||||||
|
public:
|
||||||
|
SrsGb28181SipService(SrsConfDirective* c);
|
||||||
|
virtual ~SrsGb28181SipService();
|
||||||
|
|
||||||
|
// Interface ISrsUdpHandler
|
||||||
|
public:
|
||||||
|
virtual srs_error_t on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf);
|
||||||
|
virtual void set_stfd(srs_netfd_t fd);
|
||||||
|
private:
|
||||||
|
void destroy();
|
||||||
|
srs_error_t on_udp_sip(std::string host, int port, char* buf, int nb_buf, sockaddr* from, int fromlen);
|
||||||
|
public:
|
||||||
|
int send_message(sockaddr* f, int l, std::stringstream& ss);
|
||||||
|
|
||||||
|
int send_ack(SrsSipRequest *req, sockaddr *f, int l);
|
||||||
|
int send_status(SrsSipRequest *req, sockaddr *f, int l);
|
||||||
|
|
||||||
|
int send_invite(SrsSipRequest *req, std::string ip, int port, uint32_t ssrc);
|
||||||
|
int send_bye(SrsSipRequest *req);
|
||||||
|
|
||||||
|
// The SIP command is transmitted through HTTP API,
|
||||||
|
// and the body content is transmitted to the device,
|
||||||
|
// mainly for testing and debugging, For example, here is HTTP body:
|
||||||
|
// BYE sip:34020000001320000003@3402000000 SIP/2.0
|
||||||
|
// Via: SIP/2.0/UDP 39.100.155.146:15063;rport;branch=z9hG4bK34205410
|
||||||
|
// From: <sip:34020000002000000001@3402000000>;tag=512355410
|
||||||
|
// To: <sip:34020000001320000003@3402000000>;tag=680367414
|
||||||
|
// Call-ID: 200003304
|
||||||
|
// CSeq: 21 BYE
|
||||||
|
// Max-Forwards: 70
|
||||||
|
// User-Agent: SRS/4.0.4(Leo)
|
||||||
|
// Content-Length: 0
|
||||||
|
//
|
||||||
|
//
|
||||||
|
int send_sip_raw_data(SrsSipRequest *req, std::string data);
|
||||||
|
|
||||||
|
SrsGb28181SipSession* create_sip_session(SrsSipRequest *req);
|
||||||
|
SrsGb28181SipSession* fetch(std::string id);
|
||||||
|
void remove_session(std::string id);
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -1301,6 +1301,131 @@ srs_error_t SrsGoApiError::serve_http(ISrsHttpResponseWriter* w, ISrsHttpMessage
|
||||||
return srs_api_response_code(w, r, 100);
|
return srs_api_response_code(w, r, 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SrsGoApiGb28181::SrsGoApiGb28181()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGoApiGb28181::~SrsGoApiGb28181()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
srs_error_t SrsGoApiGb28181::serve_http(ISrsHttpResponseWriter* w, ISrsHttpMessage* r)
|
||||||
|
{
|
||||||
|
SrsJsonObject* obj = SrsJsonAny::object();
|
||||||
|
SrsAutoFree(SrsJsonObject, obj);
|
||||||
|
|
||||||
|
obj->set("code", SrsJsonAny::integer(ERROR_SUCCESS));
|
||||||
|
SrsJsonObject* data = SrsJsonAny::object();
|
||||||
|
obj->set("data", data);
|
||||||
|
|
||||||
|
string id = r->query_get("id");
|
||||||
|
string action = r->query_get("action");
|
||||||
|
string vhost = r->query_get("vhost");
|
||||||
|
string app = r->query_get("app");
|
||||||
|
string stream = r->query_get("stream");
|
||||||
|
//fixed, random
|
||||||
|
string port_mode = r->query_get("port_mode");
|
||||||
|
|
||||||
|
if (_srs_gb28181) {
|
||||||
|
if(action == "create_channel"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
SrsGb28181StreamChannel channel;
|
||||||
|
channel.set_channel_id(id);
|
||||||
|
channel.set_app(app);
|
||||||
|
channel.set_stream(stream);
|
||||||
|
channel.set_port_mode(port_mode);
|
||||||
|
|
||||||
|
uint32_t code = _srs_gb28181->create_stream_channel(&channel);
|
||||||
|
if (code != ERROR_SUCCESS) {
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
|
||||||
|
data->set("query", SrsJsonAny::object()
|
||||||
|
->set("id", SrsJsonAny::str(channel.get_channel_id().c_str()))
|
||||||
|
->set("ip", SrsJsonAny::str(channel.get_ip().c_str()))
|
||||||
|
->set("rtmp_port", SrsJsonAny::integer(channel.get_rtmp_port()))
|
||||||
|
->set("app", SrsJsonAny::str(channel.get_app().c_str()))
|
||||||
|
->set("stream", SrsJsonAny::str(channel.get_stream().c_str()))
|
||||||
|
->set("rtp_port", SrsJsonAny::integer(channel.get_rtp_port()))
|
||||||
|
->set("ssrc", SrsJsonAny::integer(channel.get_ssrc())));
|
||||||
|
return srs_api_response(w, r, obj->dumps());
|
||||||
|
|
||||||
|
}
|
||||||
|
else if(action == "delete_channel"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t code = _srs_gb28181->delete_stream_channel(id);
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
else if(action == "query_channel") {
|
||||||
|
SrsJsonArray* arr = SrsJsonAny::array();
|
||||||
|
data->set("channels", arr);
|
||||||
|
|
||||||
|
uint32_t code = _srs_gb28181->queue_stream_channel(id, arr);
|
||||||
|
if (code != ERROR_SUCCESS) {
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
|
||||||
|
return srs_api_response(w, r, obj->dumps());
|
||||||
|
}
|
||||||
|
else if(action == "sip_invite"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
string ssrc = r->query_get("ssrc");
|
||||||
|
string rtp_port = r->query_get("rtp_port");
|
||||||
|
string ip = r->query_get("ip");
|
||||||
|
|
||||||
|
int _port = strtoul(rtp_port.c_str(), NULL, 10);
|
||||||
|
uint32_t _ssrc = (uint32_t)(strtoul(ssrc.c_str(), NULL, 10));
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
int code = _srs_gb28181->notify_sip_invite(id, ip, _port, _ssrc);
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
else if(action == "sip_bye"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
int code = _srs_gb28181->notify_sip_bye(id);
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
else if(action == "sip_raw_data"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string body;
|
||||||
|
r->body_read_all(body);
|
||||||
|
int code = _srs_gb28181->notify_sip_raw_data(id, body);
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
else if(action == "sip_unregister"){
|
||||||
|
if (id.empty()){
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_VALUE_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
int code = _srs_gb28181->notify_sip_unregister(id);
|
||||||
|
return srs_api_response_code(w, r, code);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_ACTION_INVALID);
|
||||||
|
}
|
||||||
|
|
||||||
|
}else {
|
||||||
|
return srs_api_response_code(w, r, ERROR_GB28181_SERVER_NOT_RUN);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
SrsHttpApi::SrsHttpApi(IConnectionManager* cm, srs_netfd_t fd, SrsHttpServeMux* m, string cip)
|
SrsHttpApi::SrsHttpApi(IConnectionManager* cm, srs_netfd_t fd, SrsHttpServeMux* m, string cip)
|
||||||
: SrsConnection(cm, fd, cip)
|
: SrsConnection(cm, fd, cip)
|
||||||
{
|
{
|
||||||
|
|
|
@ -210,6 +210,16 @@ public:
|
||||||
virtual srs_error_t serve_http(ISrsHttpResponseWriter* w, ISrsHttpMessage* r);
|
virtual srs_error_t serve_http(ISrsHttpResponseWriter* w, ISrsHttpMessage* r);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class SrsGoApiGb28181 : public ISrsHttpHandler
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
SrsGoApiGb28181();
|
||||||
|
virtual ~SrsGoApiGb28181();
|
||||||
|
public:
|
||||||
|
virtual srs_error_t serve_http(ISrsHttpResponseWriter* w, ISrsHttpMessage* r);
|
||||||
|
};
|
||||||
|
|
||||||
class SrsHttpApi : virtual public SrsConnection, virtual public ISrsReloadHandler
|
class SrsHttpApi : virtual public SrsConnection, virtual public ISrsReloadHandler
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -52,6 +52,7 @@ using namespace std;
|
||||||
#include <srs_app_thread.hpp>
|
#include <srs_app_thread.hpp>
|
||||||
#include <srs_app_coworkers.hpp>
|
#include <srs_app_coworkers.hpp>
|
||||||
#include <srs_app_gb28181.hpp>
|
#include <srs_app_gb28181.hpp>
|
||||||
|
#include <srs_app_gb28181_sip.hpp>
|
||||||
|
|
||||||
// system interval in srs_utime_t,
|
// system interval in srs_utime_t,
|
||||||
// all resolution times should be times togother,
|
// all resolution times should be times togother,
|
||||||
|
@ -109,8 +110,10 @@ std::string srs_listener_type2string(SrsListenerType type)
|
||||||
return "RTSP";
|
return "RTSP";
|
||||||
case SrsListenerFlv:
|
case SrsListenerFlv:
|
||||||
return "HTTP-FLV";
|
return "HTTP-FLV";
|
||||||
case SrsListenerGb28181:
|
case SrsListenerGb28181Sip:
|
||||||
return "GB28181-SIP over UDP";
|
return "GB28181-SIP over UDP";
|
||||||
|
case SrsListenerGb28181RtpMux:
|
||||||
|
return "GB28181-Stream over RTP";
|
||||||
default:
|
default:
|
||||||
return "UNKONWN";
|
return "UNKONWN";
|
||||||
}
|
}
|
||||||
|
@ -301,7 +304,9 @@ srs_error_t SrsUdpStreamListener::listen(string i, int p)
|
||||||
|
|
||||||
// the caller already ensure the type is ok,
|
// the caller already ensure the type is ok,
|
||||||
// we just assert here for unknown stream caster.
|
// we just assert here for unknown stream caster.
|
||||||
srs_assert(type == SrsListenerMpegTsOverUdp || type == SrsListenerGb28181);
|
srs_assert(type == SrsListenerMpegTsOverUdp
|
||||||
|
|| type == SrsListenerGb28181Sip
|
||||||
|
|| type == SrsListenerGb28181RtpMux);
|
||||||
|
|
||||||
ip = i;
|
ip = i;
|
||||||
port = p;
|
port = p;
|
||||||
|
@ -344,9 +349,13 @@ SrsGb28181Listener::SrsGb28181Listener(SrsServer* svr, SrsListenerType t, SrsCon
|
||||||
{
|
{
|
||||||
// the caller already ensure the type is ok,
|
// the caller already ensure the type is ok,
|
||||||
// we just assert here for unknown stream caster.
|
// we just assert here for unknown stream caster.
|
||||||
srs_assert(type == SrsListenerGb28181);
|
srs_assert(type == SrsListenerGb28181Sip
|
||||||
if (type == SrsListenerGb28181) {
|
||type == SrsListenerGb28181RtpMux);
|
||||||
caster = new SrsGb28181Caster(c);
|
|
||||||
|
if (type == SrsListenerGb28181Sip) {
|
||||||
|
caster = new SrsGb28181SipService(c);
|
||||||
|
}else if(type == SrsListenerGb28181RtpMux){
|
||||||
|
caster = new SrsGb28181RtpMuxService(c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -526,6 +535,9 @@ void SrsServer::destroy()
|
||||||
|
|
||||||
srs_freep(signal_manager);
|
srs_freep(signal_manager);
|
||||||
srs_freep(conn_manager);
|
srs_freep(conn_manager);
|
||||||
|
|
||||||
|
//free global gb28281 manager
|
||||||
|
srs_freep(_srs_gb28181);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SrsServer::dispose()
|
void SrsServer::dispose()
|
||||||
|
@ -772,6 +784,10 @@ srs_error_t SrsServer::http_handle()
|
||||||
return srs_error_wrap(err, "handle raw");
|
return srs_error_wrap(err, "handle raw");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((err = http_api_mux->handle("/api/v1/gb28181", new SrsGoApiGb28181())) != srs_success) {
|
||||||
|
return srs_error_wrap(err, "handle raw");
|
||||||
|
}
|
||||||
|
|
||||||
// test the request info.
|
// test the request info.
|
||||||
if ((err = http_api_mux->handle("/api/v1/tests/requests", new SrsGoApiRequests())) != srs_success) {
|
if ((err = http_api_mux->handle("/api/v1/tests/requests", new SrsGoApiRequests())) != srs_success) {
|
||||||
return srs_error_wrap(err, "handle tests requests");
|
return srs_error_wrap(err, "handle tests requests");
|
||||||
|
@ -1088,6 +1104,30 @@ srs_error_t SrsServer::listen_http_stream()
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
srs_error_t SrsServer::listen_gb28281_sip(SrsConfDirective* stream_caster)
|
||||||
|
{
|
||||||
|
srs_error_t err = srs_success;
|
||||||
|
|
||||||
|
SrsListener* sip_listener = NULL;
|
||||||
|
sip_listener = new SrsGb28181Listener(this, SrsListenerGb28181Sip, stream_caster);
|
||||||
|
|
||||||
|
int port = _srs_config->get_stream_caster_gb28181_sip_listen(stream_caster);
|
||||||
|
if (port <= 0) {
|
||||||
|
return srs_error_new(ERROR_STREAM_CASTER_PORT, "invalid sip port=%d", port);
|
||||||
|
}
|
||||||
|
|
||||||
|
srs_assert(sip_listener != NULL);
|
||||||
|
|
||||||
|
listeners.push_back(sip_listener);
|
||||||
|
|
||||||
|
// TODO: support listen at <[ip:]port>
|
||||||
|
if ((err = sip_listener->listen(srs_any_address_for_listener(), port)) != srs_success) {
|
||||||
|
return srs_error_wrap(err, "listen at %d", port);
|
||||||
|
}
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
srs_error_t SrsServer::listen_stream_caster()
|
srs_error_t SrsServer::listen_stream_caster()
|
||||||
{
|
{
|
||||||
srs_error_t err = srs_success;
|
srs_error_t err = srs_success;
|
||||||
|
@ -1113,19 +1153,33 @@ srs_error_t SrsServer::listen_stream_caster()
|
||||||
} else if (srs_stream_caster_is_flv(caster)) {
|
} else if (srs_stream_caster_is_flv(caster)) {
|
||||||
listener = new SrsHttpFlvListener(this, SrsListenerFlv, stream_caster);
|
listener = new SrsHttpFlvListener(this, SrsListenerFlv, stream_caster);
|
||||||
} else if (srs_stream_caster_is_gb28181(caster)) {
|
} else if (srs_stream_caster_is_gb28181(caster)) {
|
||||||
listener = new SrsGb28181Listener(this, SrsListenerGb28181, stream_caster);
|
//init global gb28281 manger
|
||||||
|
if (_srs_gb28181 == NULL){
|
||||||
|
_srs_gb28181 = new SrsGb28181Manger(stream_caster);
|
||||||
|
if ((err = _srs_gb28181->initialize()) != srs_success){
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//sip listener
|
||||||
|
if (_srs_config->get_stream_caster_gb28181_sip_enable(stream_caster)){
|
||||||
|
if ((err = listen_gb28281_sip(stream_caster)) != srs_success){
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//gb28281 stream listener
|
||||||
|
listener = new SrsGb28181Listener(this, SrsListenerGb28181RtpMux, stream_caster);
|
||||||
} else {
|
} else {
|
||||||
return srs_error_new(ERROR_STREAM_CASTER_ENGINE, "invalid caster %s", caster.c_str());
|
return srs_error_new(ERROR_STREAM_CASTER_ENGINE, "invalid caster %s", caster.c_str());
|
||||||
}
|
}
|
||||||
srs_assert(listener != NULL);
|
srs_assert(listener != NULL);
|
||||||
|
|
||||||
listeners.push_back(listener);
|
listeners.push_back(listener);
|
||||||
|
|
||||||
int port = _srs_config->get_stream_caster_listen(stream_caster);
|
int port = _srs_config->get_stream_caster_listen(stream_caster);
|
||||||
if (port <= 0) {
|
if (port <= 0) {
|
||||||
return srs_error_new(ERROR_STREAM_CASTER_PORT, "invalid port=%d", port);
|
return srs_error_new(ERROR_STREAM_CASTER_PORT, "invalid port=%d", port);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: support listen at <[ip:]port>
|
// TODO: support listen at <[ip:]port>
|
||||||
if ((err = listener->listen(srs_any_address_for_listener(), port)) != srs_success) {
|
if ((err = listener->listen(srs_any_address_for_listener(), port)) != srs_success) {
|
||||||
return srs_error_wrap(err, "listen at %d", port);
|
return srs_error_wrap(err, "listen at %d", port);
|
||||||
|
|
|
@ -36,6 +36,8 @@
|
||||||
#include <srs_app_listener.hpp>
|
#include <srs_app_listener.hpp>
|
||||||
#include <srs_app_conn.hpp>
|
#include <srs_app_conn.hpp>
|
||||||
#include <srs_service_st.hpp>
|
#include <srs_service_st.hpp>
|
||||||
|
#include <srs_app_gb28181.hpp>
|
||||||
|
#include <srs_app_gb28181_sip.hpp>
|
||||||
|
|
||||||
class SrsServer;
|
class SrsServer;
|
||||||
class SrsConnection;
|
class SrsConnection;
|
||||||
|
@ -52,6 +54,8 @@ class SrsTcpListener;
|
||||||
class SrsAppCasterFlv;
|
class SrsAppCasterFlv;
|
||||||
class SrsRtspCaster;
|
class SrsRtspCaster;
|
||||||
class SrsCoroutineManager;
|
class SrsCoroutineManager;
|
||||||
|
class SrsGb28181Caster;
|
||||||
|
|
||||||
|
|
||||||
// The listener type for server to identify the connection,
|
// The listener type for server to identify the connection,
|
||||||
// that is, use different type to process the connection.
|
// that is, use different type to process the connection.
|
||||||
|
@ -69,8 +73,10 @@ enum SrsListenerType
|
||||||
SrsListenerRtsp = 4,
|
SrsListenerRtsp = 4,
|
||||||
// TCP stream, FLV stream over HTTP.
|
// TCP stream, FLV stream over HTTP.
|
||||||
SrsListenerFlv = 5,
|
SrsListenerFlv = 5,
|
||||||
// UDP stream, gb28181 stream
|
// UDP stream, gb28181 ps stream over rtp,
|
||||||
SrsListenerGb28181 = 6,
|
SrsListenerGb28181RtpMux = 6,
|
||||||
|
// UDP gb28181 sip server
|
||||||
|
SrsListenerGb28181Sip = 7,
|
||||||
};
|
};
|
||||||
|
|
||||||
// A common tcp listener, for RTMP/HTTP server.
|
// A common tcp listener, for RTMP/HTTP server.
|
||||||
|
@ -158,7 +164,7 @@ public:
|
||||||
virtual ~SrsUdpCasterListener();
|
virtual ~SrsUdpCasterListener();
|
||||||
};
|
};
|
||||||
|
|
||||||
// A UDP sip listener, for sip server.
|
// A UDP gb28181 listener, for sip and rtp stream mux server.
|
||||||
class SrsGb28181Listener : public SrsUdpStreamListener
|
class SrsGb28181Listener : public SrsUdpStreamListener
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -290,6 +296,7 @@ private:
|
||||||
virtual srs_error_t listen_http_api();
|
virtual srs_error_t listen_http_api();
|
||||||
virtual srs_error_t listen_http_stream();
|
virtual srs_error_t listen_http_stream();
|
||||||
virtual srs_error_t listen_stream_caster();
|
virtual srs_error_t listen_stream_caster();
|
||||||
|
virtual srs_error_t listen_gb28281_sip(SrsConfDirective* c);
|
||||||
// Close the listeners for specified type,
|
// Close the listeners for specified type,
|
||||||
// Remove the listen object from manager.
|
// Remove the listen object from manager.
|
||||||
virtual void close_listeners(SrsListenerType type);
|
virtual void close_listeners(SrsListenerType type);
|
||||||
|
|
|
@ -322,6 +322,22 @@
|
||||||
#define ERROR_BASE64_DECODE 4039
|
#define ERROR_BASE64_DECODE 4039
|
||||||
#define ERROR_HTTP_STREAM_EOF 4040
|
#define ERROR_HTTP_STREAM_EOF 4040
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////
|
||||||
|
// GB28181 API error.
|
||||||
|
///////////////////////////////////////////////////////
|
||||||
|
#define ERROR_GB28181_SERVER_NOT_RUN 6000
|
||||||
|
#define ERROR_GB28181_SESSION_IS_EXIST 6001
|
||||||
|
#define ERROR_GB28181_SESSION_IS_NOTEXIST 6002
|
||||||
|
#define ERROR_GB28181_RTP_PORT_FULL 6003
|
||||||
|
#define ERROR_GB28181_PORT_MODE_INVALID 6004
|
||||||
|
#define ERROR_GB28181_VALUE_EMPTY 6005
|
||||||
|
#define ERROR_GB28181_ACTION_INVALID 6006
|
||||||
|
#define ERROR_GB28181_SIP_NOT_RUN 6007
|
||||||
|
#define ERROR_GB28281_SIP_INVITE_FAILED 6008
|
||||||
|
#define ERROR_GB28281_SIP_BYE_FAILED 6009
|
||||||
|
#define ERROR_GB28281_SIP_IS_INVITING 6010
|
||||||
|
#define ERROR_GB28281_CREATER_RTMPMUXER_FAILED 6011
|
||||||
|
|
||||||
///////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////
|
||||||
// HTTP API error.
|
// HTTP API error.
|
||||||
///////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////
|
||||||
|
|
|
@ -43,8 +43,6 @@ using namespace std;
|
||||||
#include <srs_kernel_codec.hpp>
|
#include <srs_kernel_codec.hpp>
|
||||||
#include <srs_rtsp_stack.hpp>
|
#include <srs_rtsp_stack.hpp>
|
||||||
|
|
||||||
#define SIP_MAX_HEADER_LEN 2049
|
|
||||||
|
|
||||||
unsigned int srs_sip_random(int min,int max)
|
unsigned int srs_sip_random(int min,int max)
|
||||||
{
|
{
|
||||||
srand(int(time(0)));
|
srand(int(time(0)));
|
||||||
|
@ -68,10 +66,34 @@ std::string srs_sip_get_form_to_uri(std::string msg)
|
||||||
return msg;
|
return msg;
|
||||||
}
|
}
|
||||||
|
|
||||||
msg = msg.substr(0, pos2-1);
|
msg = msg.substr(0, pos2);
|
||||||
return msg;
|
return msg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string srs_sip_get_utc_date()
|
||||||
|
{
|
||||||
|
// clock time
|
||||||
|
timeval tv;
|
||||||
|
if (gettimeofday(&tv, NULL) == -1) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// to calendar time
|
||||||
|
struct tm* tm;
|
||||||
|
if ((tm = gmtime(&tv.tv_sec)) == NULL) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
//Date: 2020-03-21T14:20:57.638
|
||||||
|
std::string utc_date = "";
|
||||||
|
char buffer[25] = {0};
|
||||||
|
snprintf(buffer, 25,
|
||||||
|
"%d-%02d-%02dT%02d:%02d:%02d.%03d",
|
||||||
|
1900 + tm->tm_year, 1 + tm->tm_mon, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec, (int)(tv.tv_usec / 1000));
|
||||||
|
utc_date = buffer;
|
||||||
|
return utc_date;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
std::string srs_sip_get_param(std::string msg, std::string param)
|
std::string srs_sip_get_param(std::string msg, std::string param)
|
||||||
{
|
{
|
||||||
|
@ -120,6 +142,8 @@ SrsSipRequest::SrsSipRequest()
|
||||||
status = "";
|
status = "";
|
||||||
expires = 3600;
|
expires = 3600;
|
||||||
max_forwards = 70;
|
max_forwards = 70;
|
||||||
|
www_authenticate = "";
|
||||||
|
authorization = "";
|
||||||
cmdtype = SrsSipCmdRequest;
|
cmdtype = SrsSipCmdRequest;
|
||||||
|
|
||||||
host = "127.0.0.1";;
|
host = "127.0.0.1";;
|
||||||
|
@ -202,6 +226,8 @@ void SrsSipRequest::copy(SrsSipRequest* src)
|
||||||
status = src->status;
|
status = src->status;
|
||||||
expires = src->expires;
|
expires = src->expires;
|
||||||
max_forwards = src->max_forwards;
|
max_forwards = src->max_forwards;
|
||||||
|
www_authenticate = src->www_authenticate;
|
||||||
|
authorization = src->authorization;
|
||||||
cmdtype = src->cmdtype;
|
cmdtype = src->cmdtype;
|
||||||
|
|
||||||
host = src->host;
|
host = src->host;
|
||||||
|
@ -215,10 +241,8 @@ void SrsSipRequest::copy(SrsSipRequest* src)
|
||||||
sip_username = src->sip_username;
|
sip_username = src->sip_username;
|
||||||
peer_ip = src->peer_ip;
|
peer_ip = src->peer_ip;
|
||||||
peer_port = src->peer_port;
|
peer_port = src->peer_port;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
SrsSipStack::SrsSipStack()
|
SrsSipStack::SrsSipStack()
|
||||||
{
|
{
|
||||||
buf = new SrsSimpleStream();
|
buf = new SrsSimpleStream();
|
||||||
|
@ -256,8 +280,8 @@ srs_error_t SrsSipStack::do_parse_request(SrsSipRequest* req, const char* recv_m
|
||||||
body = header_body.at(1);
|
body = header_body.at(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
//srs_trace("sip: header=%s\n", header.c_str());
|
srs_info("sip: header=%s\n", header.c_str());
|
||||||
//srs_trace("sip: body=%s\n", body.c_str());
|
srs_info("sip: body=%s\n", body.c_str());
|
||||||
|
|
||||||
// parse one by one.
|
// parse one by one.
|
||||||
char* start = (char*)header.c_str();
|
char* start = (char*)header.c_str();
|
||||||
|
@ -274,7 +298,7 @@ srs_error_t SrsSipStack::do_parse_request(SrsSipRequest* req, const char* recv_m
|
||||||
|
|
||||||
if (firstline == ""){
|
if (firstline == ""){
|
||||||
firstline = oneline;
|
firstline = oneline;
|
||||||
//srs_trace("=== first line=%s", firstline.c_str());
|
srs_info("sip: first line=%s", firstline.c_str());
|
||||||
}else{
|
}else{
|
||||||
size_t pos = oneline.find(":");
|
size_t pos = oneline.find(":");
|
||||||
if (pos != string::npos){
|
if (pos != string::npos){
|
||||||
|
@ -322,6 +346,7 @@ srs_error_t SrsSipStack::do_parse_request(SrsSipRequest* req, const char* recv_m
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (!strcasecmp(phead, "via:")) {
|
else if (!strcasecmp(phead, "via:")) {
|
||||||
|
std::vector<std::string> vec_seq = srs_string_split(content, ";");
|
||||||
req->via = content;
|
req->via = content;
|
||||||
req->branch = srs_sip_get_param(content.c_str(), "branch");
|
req->branch = srs_sip_get_param(content.c_str(), "branch");
|
||||||
}
|
}
|
||||||
|
@ -334,12 +359,18 @@ srs_error_t SrsSipStack::do_parse_request(SrsSipRequest* req, const char* recv_m
|
||||||
else if (!strcasecmp(phead, "max-forwards:")){
|
else if (!strcasecmp(phead, "max-forwards:")){
|
||||||
req->max_forwards = strtoul(content.c_str(), NULL, 10);
|
req->max_forwards = strtoul(content.c_str(), NULL, 10);
|
||||||
}
|
}
|
||||||
|
else if (!strcasecmp(phead, "www-authenticate:")){
|
||||||
|
req->www_authenticate = content;
|
||||||
|
}
|
||||||
|
else if (!strcasecmp(phead, "authorization:")){
|
||||||
|
req->authorization = content;
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
|
//TODO: fixme
|
||||||
srs_trace("sip: unkonw message head %s content=%s", phead, content.c_str());
|
srs_trace("sip: unkonw message head %s content=%s", phead, content.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//srs_trace("====new line=%s", oneline.c_str());
|
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
p++;
|
p++;
|
||||||
|
@ -370,24 +401,24 @@ srs_error_t SrsSipStack::do_parse_request(SrsSipRequest* req, const char* recv_m
|
||||||
|
|
||||||
req->sip_username = req->sip_auth_id;
|
req->sip_username = req->sip_auth_id;
|
||||||
|
|
||||||
//srs_trace("sip: method=%s uri=%s version=%s cmdtype=%s",
|
srs_info("sip: method=%s uri=%s version=%s cmdtype=%s",
|
||||||
// req->method.c_str(), req->uri.c_str(), req->version.c_str(), req->get_cmdtype_str().c_str());
|
req->method.c_str(), req->uri.c_str(), req->version.c_str(), req->get_cmdtype_str().c_str());
|
||||||
// srs_trace("via=%s", req->via.c_str());
|
srs_info("via=%s", req->via.c_str());
|
||||||
// srs_trace("via_branch=%s", req->branch.c_str());
|
srs_info("via_branch=%s", req->branch.c_str());
|
||||||
//srs_trace("cseq=%d", req->seq);
|
srs_info("cseq=%d", req->seq);
|
||||||
// srs_trace("contact=%s", req->contact.c_str());
|
srs_info("contact=%s", req->contact.c_str());
|
||||||
//srs_trace("from=%s", req->from.c_str());
|
srs_info("from=%s", req->from.c_str());
|
||||||
//srs_trace("to=%s", req->to.c_str());
|
srs_info("to=%s", req->to.c_str());
|
||||||
//srs_trace("callid=%s", req->call_id.c_str());
|
srs_info("callid=%s", req->call_id.c_str());
|
||||||
// srs_trace("status=%s", req->status.c_str());
|
srs_info("status=%s", req->status.c_str());
|
||||||
// srs_trace("from_tag=%s", req->from_tag.c_str());
|
srs_info("from_tag=%s", req->from_tag.c_str());
|
||||||
// srs_trace("to_tag=%s", req->to_tag.c_str());
|
srs_info("to_tag=%s", req->to_tag.c_str());
|
||||||
//srs_trace("sip_auth_id=%s", req->sip_auth_id.c_str());
|
srs_info("sip_auth_id=%s", req->sip_auth_id.c_str());
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
srs_error_t SrsSipStack::resp_keepalive(std::stringstream& ss, SrsSipRequest *req){
|
void SrsSipStack::resp_keepalive(std::stringstream& ss, SrsSipRequest *req){
|
||||||
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
||||||
<< "Via: " << SRS_SIP_VERSION << "/UDP " << req->host << ":" << req->host_port << ";branch=" << req->branch << SRS_RTSP_CRLF
|
<< "Via: " << SRS_SIP_VERSION << "/UDP " << req->host << ":" << req->host_port << ";branch=" << req->branch << SRS_RTSP_CRLF
|
||||||
<< "From: <sip:" << req->from.c_str() << ">;tag=" << req->from_tag << SRS_RTSP_CRLF
|
<< "From: <sip:" << req->from.c_str() << ">;tag=" << req->from_tag << SRS_RTSP_CRLF
|
||||||
|
@ -398,30 +429,43 @@ srs_error_t SrsSipStack::resp_keepalive(std::stringstream& ss, SrsSipRequest *re
|
||||||
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
||||||
<< "User-Agent: "<< SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
<< "User-Agent: "<< SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
||||||
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
||||||
|
|
||||||
return srs_success;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
srs_error_t SrsSipStack::resp_ack(std::stringstream& ss, SrsSipRequest *req){
|
void SrsSipStack::resp_status(stringstream& ss, SrsSipRequest *req)
|
||||||
|
|
||||||
ss << "ACK " << "sip:" << req->sip_auth_id << "@" << req->realm << " "<< SRS_SIP_VERSION << SRS_RTSP_CRLF
|
|
||||||
<< "Via: " << SRS_SIP_VERSION << "/UDP " << req->host << ":" << req->host_port << ";branch=" << req->branch << SRS_RTSP_CRLF
|
|
||||||
<< "From: <sip:" << req->serial << "@" << req->host + ":" << req->host_port << ">;tag=" << req->from_tag << SRS_RTSP_CRLF
|
|
||||||
<< "To: <sip:"<< req->sip_auth_id << "@" << req->realm << ">\r\n"
|
|
||||||
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
|
||||||
<< "CSeq: " << req->seq << " " << req->method << SRS_RTSP_CRLF
|
|
||||||
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
|
||||||
<< "User-Agent: "<< SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
|
||||||
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
|
||||||
|
|
||||||
return srs_success;
|
|
||||||
}
|
|
||||||
|
|
||||||
srs_error_t SrsSipStack::resp_status(stringstream& ss, SrsSipRequest *req)
|
|
||||||
{
|
{
|
||||||
srs_error_t err = srs_success;
|
|
||||||
|
|
||||||
if (req->method == "REGISTER"){
|
if (req->method == "REGISTER"){
|
||||||
|
/*
|
||||||
|
//request: sip-agent-----REGISTER------->sip-server
|
||||||
|
REGISTER sip:34020000002000000001@3402000000 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP 192.168.137.11:5060;rport;branch=z9hG4bK1371463273
|
||||||
|
From: <sip:34020000001320000003@3402000000>;tag=2043466181
|
||||||
|
To: <sip:34020000001320000003@3402000000>
|
||||||
|
Call-ID: 1011047669
|
||||||
|
CSeq: 1 REGISTER
|
||||||
|
Contact: <sip:34020000001320000003@192.168.137.11:5060>
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: IP Camera
|
||||||
|
Expires: 3600
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
//response: sip-agent<-----200 OK--------sip-server
|
||||||
|
SIP/2.0 200 OK
|
||||||
|
Via: SIP/2.0/UDP 192.168.137.11:5060;rport;branch=z9hG4bK1371463273
|
||||||
|
From: <sip:34020000001320000003@3402000000>
|
||||||
|
To: <sip:34020000001320000003@3402000000>
|
||||||
|
CSeq: 1 REGISTER
|
||||||
|
Call-ID: 1011047669
|
||||||
|
Contact: <sip:34020000001320000003@192.168.137.11:5060>
|
||||||
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
|
Expires: 3600
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
*/
|
||||||
|
if (req->authorization.empty()){
|
||||||
|
//TODO: fixme supoort 401
|
||||||
|
//return req_401_unauthorized(ss, req);
|
||||||
|
}
|
||||||
|
|
||||||
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
||||||
<< "Via: " << req->via << SRS_RTSP_CRLF
|
<< "Via: " << req->via << SRS_RTSP_CRLF
|
||||||
<< "From: <sip:"<< req->from << ">" << SRS_RTSP_CRLF
|
<< "From: <sip:"<< req->from << ">" << SRS_RTSP_CRLF
|
||||||
|
@ -430,8 +474,43 @@ srs_error_t SrsSipStack::resp_status(stringstream& ss, SrsSipRequest *req)
|
||||||
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
||||||
<< "Contact: " << req->contact << SRS_RTSP_CRLF
|
<< "Contact: " << req->contact << SRS_RTSP_CRLF
|
||||||
<< "User-Agent: " << SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
<< "User-Agent: " << SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
||||||
|
<< "Expires: " << req->expires << SRS_RTSP_CRLF
|
||||||
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
||||||
}else{
|
}else{
|
||||||
|
/*
|
||||||
|
//request: sip-agnet-------MESSAGE------->sip-server
|
||||||
|
MESSAGE sip:34020000002000000001@3402000000 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP 192.168.137.11:5060;rport;branch=z9hG4bK1066375804
|
||||||
|
From: <sip:34020000001320000003@3402000000>;tag=1925919231
|
||||||
|
To: <sip:34020000002000000001@3402000000>
|
||||||
|
Call-ID: 1185236415
|
||||||
|
CSeq: 20 MESSAGE
|
||||||
|
Content-Type: Application/MANSCDP+xml
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: IP Camera
|
||||||
|
Content-Length: 175
|
||||||
|
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<Notify>
|
||||||
|
<CmdType>Keepalive</CmdType>
|
||||||
|
<SN>1</SN>
|
||||||
|
<DeviceID>34020000001320000003</DeviceID>
|
||||||
|
<Status>OK</Status>
|
||||||
|
<Info>
|
||||||
|
</Info>
|
||||||
|
</Notify>
|
||||||
|
//response: sip-agent------200 OK --------> sip-server
|
||||||
|
SIP/2.0 200 OK
|
||||||
|
Via: SIP/2.0/UDP 192.168.137.11:5060;rport;branch=z9hG4bK1066375804
|
||||||
|
From: <sip:34020000001320000003@3402000000>
|
||||||
|
To: <sip:34020000002000000001@3402000000>
|
||||||
|
CSeq: 20 MESSAGE
|
||||||
|
Call-ID: 1185236415
|
||||||
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
ss << SRS_SIP_VERSION <<" 200 OK" << SRS_RTSP_CRLF
|
||||||
<< "Via: " << req->via << SRS_RTSP_CRLF
|
<< "Via: " << req->via << SRS_RTSP_CRLF
|
||||||
<< "From: <sip:"<< req->from << ">" << SRS_RTSP_CRLF
|
<< "From: <sip:"<< req->from << ">" << SRS_RTSP_CRLF
|
||||||
|
@ -442,61 +521,104 @@ srs_error_t SrsSipStack::resp_status(stringstream& ss, SrsSipRequest *req)
|
||||||
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
||||||
}
|
}
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
srs_error_t SrsSipStack::req_invite(stringstream& ss, SrsSipRequest *req, int port)
|
void SrsSipStack::req_invite(stringstream& ss, SrsSipRequest *req, string ip, int port, uint32_t ssrc)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
INVITE sip:34020000001320000001@3402000000 SIP/2.0
|
//request: sip-agent <-------INVITE------ sip-server
|
||||||
Via: SIP/2.0/UDP 192.168.1.22:15060;rport;branch=z9hG4bK369961166
|
INVITE sip:34020000001320000003@3402000000 SIP/2.0
|
||||||
From: <sip:34020000002000000001@3402000000>;tag=536961166
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport;branch=z9hG4bK34208805
|
||||||
To: <sip:34020000001320000001@3402000000>
|
From: <sip:34020000002000000001@39.100.155.146:15063>;tag=512358805
|
||||||
Call-ID: 929961057
|
To: <sip:34020000001320000003@3402000000>
|
||||||
CSeq: 3 INVITE
|
Call-ID: 200008805
|
||||||
Content-Type: APPLICATION/SDP
|
CSeq: 20 INVITE
|
||||||
Contact: <sip:34020000002000000001@192.168.1.22:15060>
|
Content-Type: Application/SDP
|
||||||
|
Contact: <sip:34020000001320000003@3402000000>
|
||||||
Max-Forwards: 70
|
Max-Forwards: 70
|
||||||
User-Agent: XXXXXXX XXXXXXX
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
Subject: 34020000001320000001:0200000001,34020000002020000001:0
|
Subject: 34020000001320000003:630886,34020000002000000001:0
|
||||||
Content-Length: 247
|
Content-Length: 164
|
||||||
|
|
||||||
v=0
|
v=0
|
||||||
o=34020000002000000001 0 0 IN IP4 192.168.1.23
|
o=34020000001320000003 0 0 IN IP4 39.100.155.146
|
||||||
s=Play
|
s=Play
|
||||||
c=IN IP4 192.168.1.23
|
c=IN IP4 39.100.155.146
|
||||||
t=0 0
|
t=0 0
|
||||||
m=video 30000 RTP/AVP 96 97 98 99
|
m=video 9000 RTP/AVP 96
|
||||||
a=recvonly
|
a=recvonly
|
||||||
a=rtpmap:96 PS/90000
|
a=rtpmap:96 PS/90000
|
||||||
a=rtpmap:97 MPEG4/90000
|
y=630886
|
||||||
a=rtpmap:98 H264/90000
|
//response: sip-agent --------100 Trying--------> sip-server
|
||||||
a=rtpmap:99 H265/90000
|
SIP/2.0 100 Trying
|
||||||
y=0200000001
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport=15063;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@39.100.155.146:15063>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 20 INVITE
|
||||||
|
User-Agent: IP Camera
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
//response: sip-agent -------200 OK--------> sip-server
|
||||||
|
SIP/2.0 200 OK
|
||||||
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport=15063;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@39.100.155.146:15063>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>;tag=1083111311
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 20 INVITE
|
||||||
|
Contact: <sip:34020000001320000003@192.168.137.11:5060>
|
||||||
|
Content-Type: application/sdp
|
||||||
|
User-Agent: IP Camera
|
||||||
|
Content-Length: 263
|
||||||
|
|
||||||
|
v=0
|
||||||
|
o=34020000001320000003 1073 1073 IN IP4 192.168.137.11
|
||||||
|
s=Play
|
||||||
|
c=IN IP4 192.168.137.11
|
||||||
|
t=0 0
|
||||||
|
m=video 15060 RTP/AVP 96
|
||||||
|
a=setup:active
|
||||||
|
a=sendonly
|
||||||
|
a=rtpmap:96 PS/90000
|
||||||
|
a=username:34020000001320000003
|
||||||
|
a=password:12345678
|
||||||
|
a=filesize:0
|
||||||
|
y=0000630886
|
||||||
|
f=
|
||||||
|
//request: sip-agent <------ ACK ------- sip-server
|
||||||
|
ACK sip:34020000001320000003@3402000000 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@39.100.155.146:15063>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 20 ACK
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
|
Content-Length: 0
|
||||||
*/
|
*/
|
||||||
|
|
||||||
srs_error_t err = srs_success;
|
|
||||||
int ssrc = srs_sip_random(10000, 99999);
|
|
||||||
std::stringstream sdp;
|
std::stringstream sdp;
|
||||||
sdp << "v=0" << SRS_RTSP_CRLF
|
sdp << "v=0" << SRS_RTSP_CRLF
|
||||||
<< "o=" << req->sip_auth_id << " 0 0 IN IP4 " << req->host << SRS_RTSP_CRLF
|
<< "o=" << req->sip_auth_id << " 0 0 IN IP4 " << ip << SRS_RTSP_CRLF
|
||||||
<< "s=Play" << SRS_RTSP_CRLF
|
<< "s=Play" << SRS_RTSP_CRLF
|
||||||
<< "c=IN IP4 " << req->host << SRS_RTSP_CRLF
|
<< "c=IN IP4 " << ip << SRS_RTSP_CRLF
|
||||||
<< "t=0 0" << SRS_RTSP_CRLF
|
<< "t=0 0" << SRS_RTSP_CRLF
|
||||||
<< "m=video " << port <<" RTP/AVP 96 97 98 99" << SRS_RTSP_CRLF
|
//TODO 97 98 99 current no support
|
||||||
|
//<< "m=video " << port <<" RTP/AVP 96 97 98 99" << SRS_RTSP_CRLF
|
||||||
|
<< "m=video " << port <<" RTP/AVP 96" << SRS_RTSP_CRLF
|
||||||
<< "a=recvonly" << SRS_RTSP_CRLF
|
<< "a=recvonly" << SRS_RTSP_CRLF
|
||||||
<< "a=rtpmap:96 PS/90000" << SRS_RTSP_CRLF
|
<< "a=rtpmap:96 PS/90000" << SRS_RTSP_CRLF
|
||||||
<< "a=rtpmap:97 MPEG4/90000" << SRS_RTSP_CRLF
|
//TODO: current no support
|
||||||
<< "a=rtpmap:98 H264/90000" << SRS_RTSP_CRLF
|
//<< "a=rtpmap:97 MPEG4/90000" << SRS_RTSP_CRLF
|
||||||
<< "a=rtpmap:99 H265/90000" << SRS_RTSP_CRLF
|
//<< "a=rtpmap:98 H264/90000" << SRS_RTSP_CRLF
|
||||||
<< "y=00181" << ssrc << SRS_RTSP_CRLF;
|
//<< "a=rtpmap:99 H265/90000" << SRS_RTSP_CRLF
|
||||||
|
|
||||||
//<< "a=streamMode:MAIN\r\n"
|
//<< "a=streamMode:MAIN\r\n"
|
||||||
//<< "a=filesize:0\r\n"
|
//<< "a=filesize:0\r\n"
|
||||||
|
<< "y=" << ssrc << SRS_RTSP_CRLF;
|
||||||
|
|
||||||
|
|
||||||
int rand = srs_sip_random(1000, 9999);
|
int rand = srs_sip_random(1000, 9999);
|
||||||
std::stringstream from, to, uri;
|
std::stringstream from, to, uri, branch, from_tag;
|
||||||
//"INVITE sip:34020000001320000001@3402000000 SIP/2.0\r\n
|
//"INVITE sip:34020000001320000001@3402000000 SIP/2.0\r\n
|
||||||
uri << "sip:" << req->sip_auth_id << "@" << req->realm;
|
uri << "sip:" << req->sip_auth_id << "@" << req->realm;
|
||||||
//From: <sip:34020000002000000001@%s:%s>;tag=500485%d\r\n
|
//From: <sip:34020000002000000001@%s:%s>;tag=500485%d\r\n
|
||||||
|
@ -507,9 +629,14 @@ srs_error_t SrsSipStack::req_invite(stringstream& ss, SrsSipRequest *req, int po
|
||||||
req->to = to.str();
|
req->to = to.str();
|
||||||
req->uri = uri.str();
|
req->uri = uri.str();
|
||||||
|
|
||||||
|
branch << "z9hG4bK3420" << rand;
|
||||||
|
from_tag << "51235" << rand;
|
||||||
|
req->branch = branch.str();
|
||||||
|
req->from_tag = from_tag.str();
|
||||||
|
|
||||||
ss << "INVITE " << req->uri << " " << SRS_SIP_VERSION << SRS_RTSP_CRLF
|
ss << "INVITE " << req->uri << " " << SRS_SIP_VERSION << SRS_RTSP_CRLF
|
||||||
<< "Via: " << SRS_SIP_VERSION << "/UDP "<< req->host << ":" << req->host_port << ";rport;branch=z9hG4bK3420" << rand << SRS_RTSP_CRLF
|
<< "Via: " << SRS_SIP_VERSION << "/UDP "<< req->host << ":" << req->host_port << ";rport;branch=" << req->branch << SRS_RTSP_CRLF
|
||||||
<< "From: <sip:" << req->from << ">;tag=51235" << rand << SRS_RTSP_CRLF
|
<< "From: <sip:" << req->from << ">;tag=" << req->from_tag << SRS_RTSP_CRLF
|
||||||
<< "To: <sip:" << req->to << ">" << SRS_RTSP_CRLF
|
<< "To: <sip:" << req->to << ">" << SRS_RTSP_CRLF
|
||||||
<< "Call-ID: 20000" << rand <<SRS_RTSP_CRLF
|
<< "Call-ID: 20000" << rand <<SRS_RTSP_CRLF
|
||||||
<< "CSeq: 20 INVITE" << SRS_RTSP_CRLF
|
<< "CSeq: 20 INVITE" << SRS_RTSP_CRLF
|
||||||
|
@ -517,40 +644,133 @@ srs_error_t SrsSipStack::req_invite(stringstream& ss, SrsSipRequest *req, int po
|
||||||
<< "Contact: <sip:" << req->to << ">" << SRS_RTSP_CRLF
|
<< "Contact: <sip:" << req->to << ">" << SRS_RTSP_CRLF
|
||||||
<< "Max-Forwards: 70" << " \r\n"
|
<< "Max-Forwards: 70" << " \r\n"
|
||||||
<< "User-Agent: " << SRS_SIP_USER_AGENT <<SRS_RTSP_CRLF
|
<< "User-Agent: " << SRS_SIP_USER_AGENT <<SRS_RTSP_CRLF
|
||||||
<< "Subject: "<< req->sip_auth_id << ":00181" << ssrc << "," << req->serial << ":0" << SRS_RTSP_CRLF
|
<< "Subject: "<< req->sip_auth_id << ":" << ssrc << "," << req->serial << ":0" << SRS_RTSP_CRLF
|
||||||
<< "Content-Length: " << sdp.str().length() << SRS_RTSP_CRLFCRLF
|
<< "Content-Length: " << sdp.str().length() << SRS_RTSP_CRLFCRLF
|
||||||
<< sdp.str();
|
<< sdp.str();
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
srs_error_t SrsSipStack::req_bye(std::stringstream& ss, SrsSipRequest *req)
|
|
||||||
|
void SrsSipStack::req_401_unauthorized(std::stringstream& ss, SrsSipRequest *req)
|
||||||
{
|
{
|
||||||
srs_error_t err = srs_success;
|
/* sip-agent <-----401 Unauthorized ------ sip-server
|
||||||
|
SIP/2.0 401 Unauthorized
|
||||||
|
Via: SIP/2.0/UDP 192.168.137.92:5061;rport=61378;received=192.168.1.13;branch=z9hG4bK802519080
|
||||||
|
From: <sip:34020000001320000004@192.168.137.92:5061>;tag=611442989
|
||||||
|
To: <sip:34020000001320000004@192.168.137.92:5061>;tag=102092689
|
||||||
|
CSeq: 1 REGISTER
|
||||||
|
Call-ID: 1650345118
|
||||||
|
User-Agent: LiveGBS v200228
|
||||||
|
Contact: <sip:34020000002000000001@192.168.1.23:15060>
|
||||||
|
Content-Length: 0
|
||||||
|
WWW-Authenticate: Digest realm="3402000000",qop="auth",nonce="f1da98bd160f3e2efe954c6eedf5f75a"
|
||||||
|
*/
|
||||||
|
|
||||||
|
ss << SRS_SIP_VERSION <<" 401 Unauthorized" << SRS_RTSP_CRLF
|
||||||
|
//<< "Via: " << req->via << SRS_RTSP_CRLF
|
||||||
|
<< "Via: " << req->via << ";rport=" << req->peer_port << ";received=" << req->peer_ip << ";branch=" << req->branch << SRS_RTSP_CRLF
|
||||||
|
<< "From: <sip:"<< req->from << ">" << SRS_RTSP_CRLF
|
||||||
|
<< "To: <sip:"<< req->to << ">" << SRS_RTSP_CRLF
|
||||||
|
<< "CSeq: "<< req->seq << " " << req->method << SRS_RTSP_CRLF
|
||||||
|
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
||||||
|
<< "Contact: " << req->contact << SRS_RTSP_CRLF
|
||||||
|
<< "User-Agent: " << SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
||||||
|
<< "Content-Length: 0" << SRS_RTSP_CRLF
|
||||||
|
<< "WWW-Authenticate: Digest realm=\"3402000000\",qop=\"auth\",nonce=\"f1da98bd160f3e2efe954c6eedf5f75a\"" << SRS_RTSP_CRLFCRLF;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SrsSipStack::resp_ack(std::stringstream& ss, SrsSipRequest *req){
|
||||||
|
/*
|
||||||
|
//request: sip-agent <------ ACK ------- sip-server
|
||||||
|
ACK sip:34020000001320000003@3402000000 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@39.100.155.146:15063>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 20 ACK
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
|
Content-Length: 0
|
||||||
|
*/
|
||||||
|
|
||||||
|
ss << "ACK " << "sip:" << req->sip_auth_id << "@" << req->realm << " "<< SRS_SIP_VERSION << SRS_RTSP_CRLF
|
||||||
|
<< "Via: " << SRS_SIP_VERSION << "/UDP " << req->host << ":" << req->host_port << ";rport;branch=" << req->branch << SRS_RTSP_CRLF
|
||||||
|
<< "From: <sip:" << req->serial << "@" << req->host + ":" << req->host_port << ">;tag=" << req->from_tag << SRS_RTSP_CRLF
|
||||||
|
<< "To: <sip:"<< req->sip_auth_id << "@" << req->realm << ">\r\n"
|
||||||
|
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
||||||
|
<< "CSeq: " << req->seq << " ACK"<< SRS_RTSP_CRLF
|
||||||
|
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
||||||
|
<< "User-Agent: "<< SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
||||||
|
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SrsSipStack::req_bye(std::stringstream& ss, SrsSipRequest *req)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
//request: sip-agent <------BYE------ sip-server
|
||||||
|
BYE sip:34020000001320000003@3402000000 SIP/2.0
|
||||||
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@3402000000>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>;tag=1083111311
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 79 BYE
|
||||||
|
Max-Forwards: 70
|
||||||
|
User-Agent: SRS/4.0.4(Leo)
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
//response: sip-agent ------200 OK ------> sip-server
|
||||||
|
SIP/2.0 200 OK
|
||||||
|
Via: SIP/2.0/UDP 39.100.155.146:15063;rport=15063;branch=z9hG4bK34208805
|
||||||
|
From: <sip:34020000002000000001@3402000000>;tag=512358805
|
||||||
|
To: <sip:34020000001320000003@3402000000>;tag=1083111311
|
||||||
|
Call-ID: 200008805
|
||||||
|
CSeq: 79 BYE
|
||||||
|
User-Agent: IP Camera
|
||||||
|
Content-Length: 0
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
std::stringstream from, to, uri;
|
std::stringstream from, to, uri;
|
||||||
uri << "sip:" << req->sip_auth_id << "@" << req->realm;
|
uri << "sip:" << req->sip_auth_id << "@" << req->realm;
|
||||||
from << req->serial << "@" << req->host << ":" << req->host_port;
|
from << req->serial << "@" << req->realm;
|
||||||
to << req->sip_auth_id << "@" << req->realm;
|
to << req->sip_auth_id << "@" << req->realm;
|
||||||
|
|
||||||
req->from = from.str();
|
req->from = from.str();
|
||||||
req->to = to.str();
|
req->to = to.str();
|
||||||
req->uri = uri.str();
|
req->uri = uri.str();
|
||||||
|
|
||||||
int rand = srs_sip_random(1000, 9999);
|
string to_tag, from_tag, branch;
|
||||||
|
|
||||||
|
if (req->branch.empty()){
|
||||||
|
branch = "";
|
||||||
|
}else {
|
||||||
|
branch = ";branch=" + req->branch;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req->from_tag.empty()){
|
||||||
|
from_tag = "";
|
||||||
|
}else {
|
||||||
|
from_tag = ";tag=" + req->from_tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req->to_tag.empty()){
|
||||||
|
to_tag = "";
|
||||||
|
}else {
|
||||||
|
to_tag = ";tag=" + req->to_tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
int seq = srs_sip_random(22, 99);
|
||||||
ss << "BYE " << req->uri << " "<< SRS_SIP_VERSION << SRS_RTSP_CRLF
|
ss << "BYE " << req->uri << " "<< SRS_SIP_VERSION << SRS_RTSP_CRLF
|
||||||
<< "Via: "<< SRS_SIP_VERSION << "/UDP "<< req->host << ":" << req->host_port << ";branch=z9hG4bK3420" << rand << SRS_RTSP_CRLF
|
<< "Via: "<< SRS_SIP_VERSION << "/UDP "<< req->host << ":" << req->host_port << ";rport" << branch << SRS_RTSP_CRLF
|
||||||
<< "From: <sip:" << req->from << ">;tag=51235" << rand << SRS_RTSP_CRLF
|
<< "From: <sip:" << req->from << ">" << from_tag << SRS_RTSP_CRLF
|
||||||
<< "To: <sip:" << req->to << ">" << SRS_RTSP_CRLF
|
<< "To: <sip:" << req->to << ">" << to_tag << SRS_RTSP_CRLF
|
||||||
<< "Call-ID: 20000" << rand << SRS_RTSP_CRLF
|
<< "Call-ID: " << req->call_id << SRS_RTSP_CRLF
|
||||||
<< "CSeq: 21 BYE" << SRS_RTSP_CRLF
|
<< "CSeq: "<< seq <<" BYE" << SRS_RTSP_CRLF
|
||||||
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
<< "Max-Forwards: 70" << SRS_RTSP_CRLF
|
||||||
<< "User-Agent: " << SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
<< "User-Agent: " << SRS_SIP_USER_AGENT << SRS_RTSP_CRLF
|
||||||
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
<< "Content-Length: 0" << SRS_RTSP_CRLFCRLF;
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -83,6 +83,9 @@ public:
|
||||||
long expires;
|
long expires;
|
||||||
int max_forwards;
|
int max_forwards;
|
||||||
|
|
||||||
|
std::string www_authenticate;
|
||||||
|
std::string authorization;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
std::string serial;
|
std::string serial;
|
||||||
std::string realm;
|
std::string realm;
|
||||||
|
@ -128,12 +131,13 @@ protected:
|
||||||
virtual srs_error_t do_parse_request(SrsSipRequest* req, const char *recv_msg);
|
virtual srs_error_t do_parse_request(SrsSipRequest* req, const char *recv_msg);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
virtual srs_error_t resp_status(std::stringstream& ss, SrsSipRequest *req);
|
virtual void resp_status(std::stringstream& ss, SrsSipRequest *req);
|
||||||
virtual srs_error_t resp_keepalive(std::stringstream& ss, SrsSipRequest *req);
|
virtual void resp_keepalive(std::stringstream& ss, SrsSipRequest *req);
|
||||||
virtual srs_error_t resp_ack(std::stringstream& ss, SrsSipRequest *req);
|
virtual void resp_ack(std::stringstream& ss, SrsSipRequest *req);
|
||||||
|
|
||||||
virtual srs_error_t req_invite(std::stringstream& ss, SrsSipRequest *req, int port);
|
virtual void req_invite(std::stringstream& ss, SrsSipRequest *req, std::string ip, int port, uint32_t ssrc);
|
||||||
virtual srs_error_t req_bye(std::stringstream& ss, SrsSipRequest *req);
|
virtual void req_bye(std::stringstream& ss, SrsSipRequest *req);
|
||||||
|
virtual void req_401_unauthorized(std::stringstream& ss, SrsSipRequest *req);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue