概述

在之前的文章中,我们介绍了如何使用fs的event事件机制来获取呼叫的各种信息。

这些event事件一般都是底层模块定义好的,其中的各种信息已经很完备了,日常的开发需求都可以满足。

但是,总有一些场景是无法完全满足的,例如,在fs的注册事件中,就没有X-自定义头域的信息。

在定制化的sip交互过程中,freeswitch是支持自定义头域的,头域格式要满足“X-***”的模式。而当我们订阅了"sofia::register"事件,在事件中是无法获得自定义头域的信息的。

本文从fs的核心模块mod_sofia的代码出发,分析如何增加对自定义头域的信息的获取和事件上报。

环境

centos:CentOS  release 7.0 (Final)或以上版本

freeswitch:v1.8.7

GCC:4.8.5

mod_sofia模块

mod_sofia模块是fs在sofia-sip库的基础上实现的SIP终端模块,fs中所有的sip消息处理和上报都要通过mod_sofia模块来对接。

我们可以在 src\mod\endpoints\mod_sofia\sofia_reg.c 文件中,找到注册消息的处理函数“sofia_reg_handle_register_token”,并在函数中找到“MY_EVENT_REGISTER "sofia::register"”事件的创建和上报流程。

if (switch_event_create_subclass(&s_event, SWITCH_EVENT_CUSTOM, MY_EVENT_REGISTER) == SWITCH_STATUS_SUCCESS) {

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "profile-name", profile->name);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "from-user", to_user);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "from-host", reg_host);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "presence-hosts", profile->presence_hosts ? profile->presence_hosts : "n/a");

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "contact", contact_str);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "call-id", call_id);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "rpid", rpid);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "status", reg_desc);

switch_event_add_header(s_event, SWITCH_STACK_BOTTOM, "expires", "%ld", (long) exptime);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "to-user", from_user);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "to-host", from_host);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "network-ip", network_ip);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "network-port", network_port_c);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "username", username);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "realm", realm);

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "user-agent", agent);

if (update_registration) {

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, "update-reg", "true");

}

if (v_event && *v_event) {

switch_event_merge(s_event, *v_event);

}

//add by zr 20220121

for (sip_unknown_t *un = sip->sip_unknown; un; un = un->un_next) {

switch_event_add_header_string(s_event, SWITCH_STACK_BOTTOM, un->un_name, un->un_value);

}

switch_event_fire(&s_event);

}

从上面的代码中,我们可以看到“sofia::register”事件中包含的所有头域内容。

sofia sip协议栈会解析所有的头域,并把非标准的头域都放到“sip->sip_unknown”中。

在红色新增的代码中,我们把所有unknown的头域都放到register上报事件中。

编译

cd freeswitch-1.8.7/src/mod/endpoints/mod_sofia

make

make install

测试

启动freeswitch,使用sip终端注册1002账户。

使用api命令“/event plain ALL”打开fs的事件打印,可以在屏显中看到如下信息:

RECV EVENT

Event-Subclass: sofia::register

Event-Name: CUSTOM

Core-UUID: b95d5721-38f1-406d-b6f1-8c056394d63e

FreeSWITCH-Hostname: localhost.localdomain

FreeSWITCH-Switchname: localhost.localdomain

FreeSWITCH-IPv4: 192.168.0.152

FreeSWITCH-IPv6: ::1

Event-Date-Local: 2022-01-21 17:15:14

Event-Date-GMT: Fri, 21 Jan 2022 09:15:14 GMT

Event-Date-Timestamp: 1642756514420767

Event-Calling-File: sofia_reg.c

Event-Calling-Function: sofia_reg_handle_register_token

Event-Calling-Line-Number: 2011

Event-Sequence: 902

profile-name: internal

from-user: 1002

from-host: 192.168.0.152

presence-hosts: 192.168.0.152,192.168.0.152

contact: "1002" <sip:1002@10.9.136.138:1666;fs_nat=yes;fs_path=sip:1002@10.9.136.138:1666>

call-id: 1378235421@10.9.136.138

rpid: unknown

status: Registered(UDP-NAT)

expires: 3600

to-user: 1002

to-host: 192.168.0.152

network-ip: 10.9.136.138

network-port: 1666

username: 1002

realm: 192.168.0.152

user-agent: FaramAndroid/1.5.9

sip_number_alias: 1002

sip_auth_username: 1002

sip_auth_realm: 192.168.0.152

number_alias: 1002

user_name: 1002

domain_name: 192.168.0.152

record_stereo: true

default_gateway: example.com

default_areacode: 918

transfer_fallback_extension: operator

toll_allow: domestic,international,local

accountcode: 1002

user_context: default

effective_caller_id_name: Extension 1002

effective_caller_id_number: 1002

outbound_caller_id_name: FreeSWITCH

outbound_caller_id_number: 0000000000

callgroup: techsupport

X-Packet: com.sipp

X-Brand: huawei

X-Osver: 29

X-Token: IQAAAACy0eGUAAAWdHyqUVfb0b

X-Jgtoken: 190e35f7e0

X-Olduser: 12345678

我们可以在事件信息的最后看到“X-”开头的自定义头域信息打印。

总结

本文从fs的sip协议栈出发,修改了mod_sofia模块代码,增加自定义消息头域的事件上报,对sip消息的定制有一定参考意义。

但是,在日常的VOIP中,我们应该尽量避免自定义头域的使用,以避免兼容性的问题。


空空如常

求真得真

freeswitch修改mod_sofia模块并上报自定义头域的更多相关文章

  1. Joomla - 自定义(自定义模块、修改原有模块样式、添加全局JS)

    一.自定义模块 自定义模块 参考 Joomla - 模块系统(新建模块.模块类别.自定义模块)第三点 自定义模块部分 自定义模块布局 参考 Joomla - T3模板(非常好用的4屏响应式模板) 的第 ...

  2. APM飞控修改数传模块方法

    APM飞控修改数传模块方法 硬件 ARDUCOPTER第二代 数传模块(USB接口) 数传模块(telem接口) usb-ttl模块 修改方法 注意:APM固件版本和数传模块估计版本是分开的,但有一定 ...

  3. 21、手把手教你Extjs5(二十一)模块Form的自定义的设计

    前面几节完成了模块Grid的自定义,模块Form自定义的过程和Grid的过程类似,但是要更复杂一些.先来设计一下要完成的总体目标. 1、可以有多个Form方案,对应于显示.新增.修改.审核.审批等功能 ...

  4. Django实现自定义template页面并在admin site的app模块中加入自定义跳转链接

    在文章 Django实现自定义template页面并在admin site的app模块中加入自定义跳转链接(一) 中我们成功的为/feedback/feedback_stats/路径自定义了使用tem ...

  5. laravel修改用户模块的密码验证

    做项目的时候,用户认证几乎是必不可少的,如果我们的项目由于一些原因不得不使用 users 之外的用户表进行认证,那么就需要多做一点工作来完成这个功能. 现在假设我们只需要修改登录用户的表,表名和表结构 ...

  6. freeswitch的话单模块

    概述 最近因为业务需要,在看freeswitch中话单相关的一些模块. 在voip的使用过程中,话单是重要的基础模块,涉及到计费和问题查找. 呼叫话单最重要的一点是稳定,不能有错误或遗漏. 本章对fs ...

  7. C#综合揭秘——通过修改注册表建立Windows自定义协议

    引言 本文主要介绍注册表的概念与其相关根项的功能,以及浏览器如何通过连接调用自定义协议并与客户端进行数据通信.文中讲及如何通过C#程序.手动修改.安装项目等不同方式对注册表进行修改.其中通过安装项目对 ...

  8. 【转】C#综合揭秘——通过修改注册表建立Windows自定义协议

    引言 本文主要介绍注册表的概念与其相关根项的功能,以及浏览器如何通过连接调用自定义协议并与客户端进行数据通信.文中讲及如何通过C#程序.手动修改.安装项目等不同方式对注册表进行修改.其中通过安装项目对 ...

  9. 修改VCL源码实现自定义输入对话框

    来自:https://yq.aliyun.com/wenji/88428 通过修改VCL源码实现自定义输入对话框 在BCB中有两个函数可以实现输入对话框:InputBox和InputQuery,其实I ...

  10. Swift自定义头视图和尾视图

    var data: [[String]]! override func viewDidLoad() { super.viewDidLoad() // Do any additional setup a ...

随机推荐

  1. python原生数据类型(上)

    # 查看数据 print('hello world') # 查看数据类型 type('hello world') hello world str 1 数据类型定义 1.1 不可变数据类型 # 数值 # ...

  2. java监听全局组合键

    1. jintellitype pom <!-- 不能注册多个组合键比如alt+abc --> <!-- https://mvnrepository.com/artifact/com ...

  3. 如何根据月份查询每月Xxx的总数

    我以我的项目根据月份查询每月新增会员的总数为例 Controller @GetMapping("/getMemberReport.do") public R getMemberRe ...

  4. CAP 8.0 版本发布通告 - CAP 7岁生日快乐!

    前言 今天,我们很高兴宣布 CAP 发布 8.0 版本正式版,从 2016 年 12 月 14 日CAP立项到 2023 年 12 月14 日发布 8.0 版本刚好满 7 年,祝 CAP 7 岁生日快 ...

  5. keycloak~对接login-status-iframe页面判断用户状态变更

    上次我们说了,keycloak的login-status-iframe页面的作用,并解决了跨域情况下,iframe与主页面数据传递的方法,这一次,我们主要分析login-status-iframe.h ...

  6. [转]NLog学习笔记

    配置文件 NLog所有的配置信息都可以写到一个单独的xml文件中,也可以在程序代码中进行配置. 配置文件位置 启动的时候,NLog会试图查找配置文件完成自动配置,查找的文件依次如下(找到配置信息则结束 ...

  7. Elasticsearch对接MinIO存储Snapshot快照

    服务器配置 MinIO集群节点(4节点) IP 节点1 miniotest1.ip.tp-link.com 172.29.145.80 生产环境需要增加一个Nginx负载均衡前端用于ELK对接,此处直 ...

  8. C#新鲜面试题出炉(2024)

    总所周知  C#这门语言 没有Java的八股文,所以面试题一般都是问的业务, 那么对于新手来讲,最起码也要会一些基础性问题, 以下就是包含C# 和sqlserver几个常见的面试题   1) Dele ...

  9. Golang实现JAVA虚拟机-指令集和解释器

    原文链接:https://gaoyubo.cn/blogs/f57f32cf.html 前置 Golang实现JAVA虚拟机-解析class文件 Golang实现JAVA虚拟机-运行时数据区 一.字节 ...

  10. 如何在IIS上部署docsify以及404问题

    操作步骤 创建一个文件夹,在文件夹中新建2个文件 index.html:入口文件,整个网站只需要这个html文件,其他文件都是md文件 README.md:主页内容,如果没有这个文件,访问时提示404 ...