freeswitch的mod_xml_curl模块动态获取dialplan

news/2024/11/14 13:32:56/

 

概述

freeswitch是一款简单好用的VOIP开源软交换平台。

mod_xml_curl模块支持从web服务获取xml配置,本文介绍如何动态获取dialplan配置。

环境

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

freeswitch:v1.6.20

GCC:4.8.5

dialplan查找流程

我们在“switch_xml_parse_file”函数中设置一个断点,并打印堆栈。

Breakpoint 1, switch_xml_parse_file (file=file@entry=0x7fdda1a015f0 "/tmp/34096cac-311f-11ee-b31e-672fb2a27310.tmp.xml") at src/switch_xml.c:1625

1625                           if ((xml = switch_xml_parse_fd(fd))) {

(gdb) bt

#0  switch_xml_parse_file (file=file@entry=0x7fdda1a015f0 "/tmp/34096cac-311f-11ee-b31e-672fb2a27310.tmp.xml") at src/switch_xml.c:1625

#1  0x00007fddee4d5258 in xml_url_fetch (section=<optimized out>, tag_name=<optimized out>, key_name=<optimized out>, key_value=<optimized out>, params=0x7fdd68004d10, user_data=0x1ad1ae8) at mod_xml_curl.c:311

#2  0x00007fddf71e0ec0 in switch_xml_locate (section=section@entry=0x7fddae9fc7e8 "dialplan", tag_name=tag_name@entry=0x0, key_name=key_name@entry=0x0, key_value=key_value@entry=0x0, root=root@entry=0x7fdda1a01b00,

    node=node@entry=0x7fdda1a01af8, params=0x7fdd68004d10, clone=clone@entry=SWITCH_FALSE) at src/switch_xml.c:1675

#3  0x00007fddae9fc214 in dialplan_xml_locate (node=0x7fdda1a01af8, root=0x7fdda1a01b00, caller_profile=0x7fddd004d690, session=0x7fddd0030328) at mod_dialplan_xml.c:613

#4  dialplan_hunt (session=0x7fddd0030328, arg=<optimized out>, caller_profile=0x7fddd004d690) at mod_dialplan_xml.c:658

#5  0x00007fddf71342ef in switch_core_standard_on_routing (session=0x7fddd0030328) at src/switch_core_state_machine.c:281

#6  switch_core_session_run (session=0x7fddd0030328) at src/switch_core_state_machine.c:643

#7  0x00007fddf712c9fe in switch_core_session_thread (thread=<optimized out>, obj=0x7fddd0030328) at src/switch_core_session.c:1648

#8  0x00007fddf71286f3 in switch_core_session_thread_pool_worker (thread=0x7fddd004f5e0, obj=<optimized out>) at src/switch_core_session.c:1711

#9  0x00007fddf7213e10 in dummy_worker (opaque=0x7fddd004f5e0) at threadproc/unix/thread.c:151

#10 0x00007fddf4a11ea5 in start_thread () from /lib64/libpthread.so.0

#11 0x00007fddf4065b0d in clone () from /lib64/libc.so.6

(gdb)

从堆栈中可以看出,一通呼叫查找动态dialplan的流程如下。

switch_core_session_thread

switch_core_session_run

switch_core_standard_on_routing

dialplan_hunt

dialplan_xml_locate

switch_xml_locate

xml_url_fetch

switch_xml_parse_file

web服务

mod_xml_curl模块依赖于web服务,需要自己创建一个web接口并动态的返回xml配置。

下面是python3.10实现的一个简单的web服务接口函数,从基类“BaseHTTPRequestHandler”继承并实现简单的返回逻辑。

    def fsDialplan(self):

        length = int(self.headers['content-length']) 

        datas = self.rfile.read(length)

        logging.info('/fs/dialplan request, data=%s' % urllib.parse.unquote(datas))

        respcode = '''<document type="freeswitch/xml">

<section name="dialplan" description="dialplan-url">

    <include>

        <context name="public">

            <extension name="test-url" continue="false">

                <condition field="destination_number" expression="^(\d+)$">

                    <action application="answer"/>

                    <action application="playback" data="/usr/local/freeswitch/sounds/dialplan-test-url.wav"/>

                </condition>

            </extension>

        </context>

    </include>

</section>

</document>

'''

        self.send_response(200)

        self.send_header('Content-Type', 'application;charset=utf-8')

        self.end_headers()

        self.wfile.write(respcode.encode('utf-8'))

        return

web服务响应消息格式注意事项,必须有“section”段,xml格式不能使用压缩格式,否则会解析错误。

配置

检查conf/autoload_configs/modules.conf.xml文件,mod_xml_curl模块要放在配置的顶部。

<load module="mod_console"/>

<load module="mod_logfile"/>

<load module="mod_xml_curl"/>

修改conf/autoload_configs/xml_curl.conf.xml文件。

<configuration name="xml_curl.conf" description="cURL XML Gateway">

  <bindings>

    <binding name="dialplan">

      <param name="gateway-url" value="http://10.55.55.137:8080/fs/dialplan" bindings="dialplan"/>

    </binding>

  </bindings>

</configuration>

测试

使用10011发起呼叫,日志如下。

2023-08-03 11:08:27.438976 [INFO] mod_dialplan_xml.c:637 Processing 10011 <10011>->13712345678 in context public

2023-08-03 11:08:27.478976 [CONSOLE] mod_xml_curl.c:323 XML response is in /tmp/042fcd1e-31ab-11ee-b34d-672fb2a27310.tmp.xml

Dialplan: sofia/external/10011@10.55.55.138 parsing [public->test-url] continue=false

Dialplan: sofia/external/10011@10.55.55.138 Regex (PASS) [test-url] destination_number(13712345678) =~ /^(\d+)$/ break=on-false

Dialplan: sofia/external/10011@10.55.55.138 Action answer()

Dialplan: sofia/external/10011@10.55.55.138 Action playback(/usr/local/freeswitch/sounds/dialplan-test-url.wav)

呼叫结果符合预期。

问题

网友的问题:当使用 dialplan时,这个动态获取的dialplan优先级最高,每通电话都会执行到这个dialplan。 而有些电话我不想走xml动态获取的,就想走本地配置的,这个问题怎解决呢?

方案1,mod_xml_curl模块内部已实现的逻辑,当web服务返回“NOT FOUND”时,自动走本地配置。缺点是对web服务依赖,每通呼叫都会到web服务查询。

方案2,修改“switch_xml_locate”函数,增加条件控制,比如caller是“123456”时直接使用本地配置。缺点是不够灵活,配置条件修改较困难。

方案3,修改mod_xml_curl模块的“xml_url_fetch”函数,对web响应的xml结果进行缓存,设置过期时间,减少了对web服务的依赖,又有动态配置的优点。缺点是web配置的生效会增加缓存的延迟,弱一致性。

以上方案仅仅从测试结果推导,并未真正实现,实现细节有不确定性。

总结

mod_xml_curl模块动态获取dialplan数据,控制呼叫流程。

未解决问题。

如何更灵活的控制从web服务和本地xml文件获取配置。

如何缓存web服务的xml配置。

如何解决web服务不可用的时候,呼叫流程问题。

空空如常

求真得真


http://www.ppmy.cn/news/1004440.html

相关文章

2023-08-04 LeetCode每日一题(不同路径 III)

2023-08-04每日一题 一、题目编号 980. 不同路径 III二、题目链接 点击跳转到题目位置 三、题目描述 在二维网格 grid 上&#xff0c;有 4 种类型的方格&#xff1a; 1 表示起始方格。且只有一个起始方格。2 表示结束方格&#xff0c;且只有一个结束方格。0 表示我们可以…

案例|会展大数据服务系统 智能服务体系建设实践

根据相关的市场调查发现在国内进行的2000多个会展项目中&#xff0c;仅有15%的项目能够提供相关主题网站基本业务和服务&#xff0c;且提供的服务并不能够满足会展各方的应用&#xff0c;剩余的大部门会展项目信息系统构建的不够完善、技术性差、应用复杂等&#xff0c;甚至提供…

正则表达式学习记录(Python)

正则表达式学习记录&#xff08;Python&#xff09; 一、特殊符号和字符 多个正则表达式匹配 &#xff08; | ) 用来分隔不同的匹配模式&#xff0c;相当于逻辑或&#xff0c;可以符合其中任何一个正则表达式 at | home # 表示匹配at或者home bat | bet | bit # 表示匹配bat或…

6.如何用CSV文件生成异构图数据集

我们将使用GroupLens研究小组收集的MovieLens数据集。 这个数据集描述了MovieLens的五星评级和标记活动。该数据集包含来自600多名用户的9000多部电影的约10万个评分。我们将使用该数据集生成两种节点类型&#xff0c;分别保存电影和用户的数据&#xff0c;以及一种连接…

Shell脚本学习-Shell脚本框架

我们能写出.sh文件的脚本。已经觉得很好了。但是我们还需要进一步学习脚本框架的概念。 1、Shell脚本&#xff08;模块&#xff09;高级命名规则&#xff1a; 1&#xff09;常规Shell脚本&#xff1a;chang.sh、test.sh等 2&#xff09;模块的启动和停止统一命名为&#xff1…

Qt5.13引入QtWebApp的模块后报错: error C2440: “reinterpret_cast”: 无法从“int”转换为“quintptr”

1、开发环境 Win10-64 qt5.13 msvc2015-64bit-release 2、报错 新建一个demo工程。 引入QtWebApp的httpserver、logging、templateengine三个模块后。 直接运行&#xff0c;&#xff0c;此时报错如下&#xff1a; E:\Qt5.13.1\install\5.13.1\msvc2015_64\include\QtCore…

互联网——根服务器

说明 根服务器是互联网域名系统&#xff08;DNS&#xff09;中最高级别的服务器之一。它们负责管理整个DNS系统的顶级域名空间&#xff0c;例如.com、.org和.net等。 根服务器的主要功能是将用户的DNS查询转发到适当的顶级域名服务器。当用户在浏览器中输入一个域名&#xff…

JVM基础篇-本地方法栈与堆

JVM基础篇-本地方法栈与堆 本地方法栈 什么是本地方法? 本地方法即那些不是由java层面实现的方法&#xff0c;而是由c/c实现交给java层面进行调用&#xff0c;这些方法在java中使用native关键字标识 public native int hashCode()本地方法栈的作用? 为本地方法提供内存空…