概述

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

fs中有非常多的接口和通道变量,使用方式多变。

官方文档有时候也仅仅是介绍了最基本的使用方法和格式。

环境

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

freeswitch:v1.6

GCC:4.8.5

execute_on_xxxx

fs提供了一系列的execute_on_xxxx通道变量,供使用者设置,常见的如下。

execute_on_answer

execute_on_pre_answer

execute_on_media

execute_on_ring

execute_on_tone_detect

execute_on_originate

execute_on_post_originate

execute_on_pre_originate

execute_on_pre_bridge

execute_on_post_bridge

每一个通道变量对应呼叫流程中的状态,符合呼叫状态即会调用。

比如execute_on_media,当收到183或200之后,有媒体通道建立成功,则调用对应的APP。

用法,execute_on_media特别适合做录音的启动点,可以把早期媒体和正常通话都录下来。

<action application="export" data="execute_on_media=app + params"/>

但是某些场景下,我们需要在一个状态时调用多个APP,比如当有媒体进入的时候,我们希望同时开启录音和音频检测。

<action application="export" data="execute_on_media=record_session ${record_file}.wav"/>

<action application="export" data="execute_on_media=spandsp_start_dtmf"/>

这样的配置在实际业务流程中,由于execute_on_media是通道变量,后一个会覆盖前一个,最后只会执行一个“spandsp_start_dtmf”的APP。

源码

查看switch_channel.c的源码,execute_on的函数如下,参数2“variable_prefix”为“execute_on_media”。

SWITCH_DECLARE(switch_status_t) switch_channel_execute_on(switch_channel_t *channel, const char *variable_prefix)

{

switch_event_header_t *hp;

switch_event_t *event, *cevent;

int x = 0;

switch_core_get_variables(&event);

switch_channel_get_variables(channel, &cevent);

switch_event_merge(event, cevent);

for (hp = event->headers; hp; hp = hp->next) {

char *var = hp->name;

char *val = hp->value;

if (!strncasecmp(var, variable_prefix, strlen(variable_prefix))) {

if (hp->idx) {

int i;

for (i = 0; i < hp->idx; i++) {

x++;

do_execute_on(channel, hp->array[i]);

}

} else {

x++;

do_execute_on(channel, val);

}

}

}

switch_event_destroy(&event);

switch_event_destroy(&cevent);

return x ? SWITCH_STATUS_SUCCESS : SWITCH_STATUS_FALSE;

}

可以看出,execute_on的处理流程中,会把event的全部headers遍历一遍,对于每个header,对headername和variable_prefix进行比较。

匹配则检查header的idx,idx不为0的时候,对header下的array字段轮询调用do_execute_on。

这里有两种配置方案可以实现我们需要的多次execute_on_media。

第一种,比较明显,利用header的idx,对execute_on_media设置多个值。

<action application="export" data="execute_on_media[0]=log::execute_on_media[0]"/>

<action application="export" data="execute_on_media[1]=log execute_on_media[1]"/>

第二种,隐藏方案,strncasecmp只比较固定长度。

<action application="export" data="execute_on_media-1=log::execute_on_media[0]"/>

<action application="export" data="execute_on_media-2345=log execute_on_media[1]"/>

测试方案

测试方案使用第一种配置方案,dialplan如下。

<action application="export" data="execute_on_media[0]=log::execute_on_media[0]"/>

<action application="export" data="execute_on_media[1]=log execute_on_media[1]"/>

测试结果

发起呼叫,dialplan的流程日志如下。

2023-11-22 17:54:46.345184 [NOTICE] sofia_media.c:92 Pre-Answer sofia/external/10011@10.55.55.138!

EXECUTE sofia/external/10011@10.55.55.138 log(execute_on_media[1])

2023-11-22 17:54:46.345184 [DEBUG] mod_dptools.c:1742 execute_on_media[1]

...

2023-11-22 17:54:46.345184 [DEBUG] sofia.c:7084 Channel sofia/external/10011@10.55.55.138 entering state [early][183]

2023-11-22 17:54:46.345184 [DEBUG] switch_ivr.c:623 sofia/external/10011@10.55.55.138 Command Execute log(execute_on_media[0])

EXECUTE sofia/external/10011@10.55.55.138 log(execute_on_media[0])

2023-11-22 17:54:46.345184 [DEBUG] mod_dptools.c:1742 execute_on_media[0]

总结

execute_on大部分源码在switch_channel.c中。

思考为什么测试结果中,先执行log1,再执行log0。

空空如常

求真得真

freeswitch设置多个execute_on_media的更多相关文章

  1. freeswitch设置支持视频语音编码

    1.修改FreeSWITCH安装路径下/conf/var.xml文件中,增加: <X-PRE-PROCESS cmd=="set" data="proxy_medi ...

  2. FreeSWITCH技巧:notify与message-waiting

    FreeSWITCH技巧:notify与message-waiting @(Freeswitch经验点滴) 现象描述 在客户端登陆抓包时,发现了FreeSWITCH发来的包: NOTIFY sip:9 ...

  3. 线程池c3p0和dbcp2的配置初始化实例

    一.c3p0 public class ConnectionManager { public static ComboPooledDataSource dataSource; static { try ...

  4. Linux命令总结大全,包含所有linux命令

    使用说明:此文档包含所有的Linux命令,只有你想不到的没有你看不到的,此文档共计10万余字,有8400多行,预计阅读时间差不多需要3个小时左右,所以要给大家说一说如何阅读此文档 为了方便大家阅读,我 ...

  5. FreeSWITCH黑名单功能设置

    功能描述:对呼叫的号码进行过滤 步骤: 1.编译mod_blacklist 模块:进入源目录/usr/local/src/freeswitch  --> make mod_blacklist-i ...

  6. 使用FreeSWITCH做电话自动回访设置

    一.背景介绍: 目前公司在处理客户回访方面,需要人工进行电话回访,尤其是逢年过节的时候,电话问候更能体现服务的品质: 在某些公司,电话销售员需要给大批量的陌生用户打电话,如果能过滤掉不关心的用户,销售 ...

  7. FreeSWITCH 客户端fs_cli连接设置(acl)

    FreeSWITCH 默认配置只能 在本机连接, 要从 外面连接, 就要配置: acl.conf.xml::network-lists/list event_socket.conf.xml::appl ...

  8. freeswitch控制台日志级别设置以及存储

    1.在管理控制台上设置 console loglevel (0~7)越往上级别越大 2.打开sip详细日志 sofia profile internal siptrace on 3.关闭sip详细日志 ...

  9. freeswitch模块之event_socket

    这是我之前整理的关于freeswitch mod_event_socket的相关内容,这里记录下,也方便我以后查阅. mod_event_socket以socket的形式,对外提供控制FS一种途径, ...

  10. freeswitch呼叫流程分析

    今天翻文档时发现之前整理的关于freeswitch呼叫相关的内容,写成博文分享出来也方便我以后查阅. 整体结构图 FreeswitchCore 模块加载过程 freeswitch主程序初始化时会从mo ...

随机推荐

  1. LangChain:打造自己的LLM应用

    1.LangChain是什么 LangChain是一个框架,用于开发由LLM驱动的应用程序.可以简单认为是LLM领域的Spring,以及开源版的ChatGPT插件系统.核心的2个功能为: 1)可以将 ...

  2. Hi3798MV200 恩兔N2 NS-1 (三): 制作 Ubuntu rootfs

    目录 Hi3798MV200 恩兔N2 NS-1 (一): 设备介绍和刷机说明 Hi3798MV200 恩兔N2 NS-1 (二): HiNAS海纳思使用和修改 Hi3798MV200 恩兔N2 NS ...

  3. Vu3+Element-Plus根据路由配置生成菜单导航栏

    先看效果,整体界面结构如下 点击左侧菜单栏,右侧切换显示不同页面内容. Vue3使用路由–南河小站 1 路由配置 路由配置如下: const routes = [ { path: "&quo ...

  4. 自定义组件使用v-model

    场景描述 我们在一个系统中,会出现这样的情况, 有一个联系人的下拉框,这个下拉框中的数据是从服务端获取的. 在很多页面都需要使用这个联系人(下拉框). 我们通常是这样做的: 写一个下拉框组件然后调用接 ...

  5. 【Azure App Service】为部署在App Service上的PHP应用开启JIT编译器

    问题描述 在App Service for linux上创建一个PHP应用,通过 phpinfo() 查看PHP的扩展设置,发现JIT没有被开启, jit_buffer_size 大小为0. 那么,在 ...

  6. 头疼!卷积神经网络是什么?CNN结构、训练与优化一文全解

    本文全面探讨了卷积神经网络CNN,深入分析了背景和重要性.定义与层次介绍.训练与优化,详细分析了其卷积层.激活函数.池化层.归一化层,最后列出其训练与优化的多项关键技术:训练集准备与增强.损失函数.优 ...

  7. Codeforces 1257D - Yet Another Monster Killing Problem

    题意: 有\(n\)个怪物,每个怪物有攻击力\(a_i\)点:有\(m\)个英雄,每个英雄有攻击力\(p_i\)点,耐力\(s_{i}\)点. 怪物需要被依次杀死(按输入顺序). 每一天可以挑选一个英 ...

  8. 用Rust手把手编写一个Proxy(代理), 动工

    用Rust手把手编写一个Proxy(代理), 动工 项目 ++wmproxy++ gitee 传送门 github 传送门 设计流程图 flowchart LR A[客户端] -->|Http| ...

  9. ora2pg使用记录

    ora2pg使用记录 前言 这篇文章是我在学习使用ora2pg过程中的学习记录,以便日后遗忘查阅: 诸君也可跟随我的步伐了解一下ora2pg,或可移步如下官方文档参考学习:Ora2Pg : Migra ...

  10. oracle clob字段信息读取,不限字符长度

    在项目中,经常需要在plsql中查询clob的内容,提供以下两种方法: 第一种,oracle常规函数,限制文本在4000以内 select dbms_lob.substr(sql_text) from ...