freeswitch新增模块APP
概述
之前的文章中,我们讲解了freeswitch的源码基本结构,如何新增一个插件式模块,以及如何在模块中新增一个命令式API接口。
freeswitch的架构非常适合这种业务开发模式,即以freeswitch的基本功能为开发平台,新增插件式模块来适配各种不同的业务场景,开发效率很高,学习成本相对较低。
对于APP接口,我们可以把它理解为业务接口,每一个APP接口完成一项具体的任务,多个APP接口组合起来完成复杂的呼叫流程,后续我们会在dialplan拨号计划中详细的讲述如何通过组合多个APP接口实现定制化的呼叫流程。
对于freeswitch中的API和APP接口,我们要理解它们的区别:
1, 场景的区别,API是外部命令式接口,APP是内部业务接口。
2, 参数的区别,API接口参数中的session一般是空值,因为外部调用API时是没有呼叫会话的,而APP则不同,APP一般在呼叫流程中调用,参数session表示当前的会话,可以获取到当前呼叫的所有参数。
3, 粒度的区别,API接口一般用于完成一整套完整的流程,而APP接口的粒度更小,更多用于完成具体的一项任务。
4, 使用的区别,API接口可以使用命令行、脚本或事件套接字发起调用,而APP接口多在dialplan拨号计划或脚本中调用。
本节我们来介绍如何在模块中新增加APP接口,提供给呼叫流程使用。
开发环境
centos:CentOS release 7.0 (Final)或以上版本
freeswitch:v1.8.7
GCC:4.8.5
新增模块APP
新增模块的方法请参考之前的内容,本节内容在模块mod_task的基础上修改。
mod_task.c内容如下:
#include <switch.h>
SWITCH_MODULE_LOAD_FUNCTION(mod_task_load);
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_task_shutdown);
SWITCH_MODULE_DEFINITION(mod_task, mod_task_load, mod_task_shutdown, NULL);
SWITCH_STANDARD_API(task_api_function)
{
//SWITCH_STANDARD_API have three args: (cmd, session, stream)
char *mycmd = NULL;
int argc = 0;
char *argv[16];
bzero(argv, sizeof(argv));
//split cmd and parse
if (cmd)
{
mycmd = strdup(cmd);
if (!mycmd)
{
stream->write_function(stream, "Out of memory\n");
return SWITCH_STATUS_FALSE;
}
if (!(argc = switch_split(mycmd, ' ', argv)) || !argv[0])
{
argc = 0;
switch_safe_free(mycmd);
return SWITCH_STATUS_FALSE;
}
}
//parse cmd, brach process
if(0 == strcmp("test1", argv[0]))
{
stream->write_function(stream, "task api test1, cmd:%s, session:%p", cmd, session);
}
else if(0 == strcmp("test2", argv[0]))
{
stream->write_function(stream, "task api test2, cmd:%s, session:%p", cmd, session);
}
else
{
stream->write_function(stream, "unknown cmd, cmd:%s, session:%p", cmd, session);
}
switch_safe_free(mycmd);
return SWITCH_STATUS_SUCCESS;
}
SWITCH_STANDARD_APP(task_app_function)
{
//task_app(session, data);
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO,
"task_app_function start, session=%p, data=%s\n", (void*)session, data);
}
SWITCH_MODULE_LOAD_FUNCTION(mod_task_load)
{
switch_api_interface_t* api_interface = NULL;
switch_application_interface_t* app_interface = NULL;
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO,
"mod_task_load start\n");
// register APP
SWITCH_ADD_APP(app_interface,
"task_app",
"task_app",
"task_app",
task_app_function,
"NULL",
SAF_SUPPORT_NOMEDIA | SAF_ROUTING_EXEC);
// register API
SWITCH_ADD_API( api_interface,
"task",
"task api",
task_api_function,
"<cmd> <args>");
// 注册终端命令自动补全
switch_console_set_complete("add task test1 [args]");
switch_console_set_complete("add task test2 [args]");
return SWITCH_STATUS_SUCCESS;
}
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_task_shutdown)
{
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO,
"mod_task_shutdown stop\n");
return SWITCH_STATUS_SUCCESS;
}
编译安装
进入task模块目录,编译安装,在Makefile.am文件未变化的情况下,不需要重新config。
cd $(top_srcdir)/src/mod/applications/mod_task make install
配置
修改dialplan拨号计划
cd /usr/local/freeswitch/conf/dialplan
vi public.xml
…
<include>
<context name="public">
<extension name="test">
<condition>
<action application="task_app" data="${destination_number}"/>
</condition>
</extension>
…
加载测试
cd /usr/local/freeswitch/bin/ ./freeswitch –nonat
freeswitch启动成功后,在freeswitch命令行中输入API命令加载mod_task模块:
freeswitch@localhost.localdomain> load mod_task 2021-08-26 09:58:06.056989 [INFO] mod_task.c:96 mod_task_load start 2021-08-26 09:58:06.056989 [CONSOLE] switch_loadable_module.c:1540 Successfully Loaded [mod_task] +OK Reloading XML +OK 2021-08-26 09:58:06.056989 [NOTICE] switch_loadable_module.c:292 Adding Application 'task_app' 2021-08-26 09:58:06.056989 [NOTICE] switch_loadable_module.c:338 Adding API Function 'task'
通过其他sip服务器发起invite呼叫到本机的5080端口,在日志中可以查看到:
2021-08-26 09:59:10.417036 [NOTICE] switch_channel.c:1114 New Channel sofia/external/10011@192.168.0.110 [daeaaf00-5705-428f-a4a3-17b94a3678d1] 2021-08-26 09:59:10.417036 [INFO] mod_dialplan_xml.c:637 Processing 10011 <10011>->10012 in context public 2021-08-26 09:59:10.417036 [INFO] mod_task.c:86 task_app_function start, session=0x7fada402fba8, data=10012 2021-08-26 09:59:10.417036 [NOTICE] switch_core_state_machine.c:385 sofia/external/10011@192.168.0.110 has executed the last dialplan instruction, hanging up. 2021-08-26 09:59:10.417036 [NOTICE] switch_core_state_machine.c:387 Hangup sofia/external/10011@192.168.0.110 [CS_EXECUTE] [NORMAL_CLEARING] 2021-08-26 09:59:10.417036 [NOTICE] switch_core_session.c:1744 Session 2 (sofia/external/10011@192.168.0.110) Ended 2021-08-26 09:59:10.417036 [NOTICE] switch_core_session.c:1748 Close Channel sofia/external/10011@192.168.0.110 [CS_DESTROY]
OK,今天我们这一节的新增模块APP的说明就完成了。
空空如常
求真得真
freeswitch新增模块APP的更多相关文章
- freeswitch新增模块
概述 freeswitch的架构由稳定的核心模块和大量的外围插件式模块组成.核心模块保持稳定,外围模块可以动态的加载/卸载,非常灵活方便. 外围模块通过核心提供的 Public API与核心进行通信, ...
- freeswitch新增模块API
概述 上一章我们讲解了freeswitch的源码基本结构,以及如何新增一个插件式模块. freeswitch的架构非常适合这种业务开发模式,即以freeswitch的基本功能为开发平台,新增插件式模块 ...
- RookeyFrame在线新增模块
今天给大家演示下在线新增模块的功能,在线新增模块跟在vs中写model实体类区别不大,线上新增少了手动初始化的过程,新增后模块同样具备新增.修改.删除.查看.导入.导出.复制.批量编辑.回收站.草稿箱 ...
- Nginx在线服务状态下平滑升级或新增模块
nginx在使用过程中,有时需要在不影响当前业务的情况下,进行升级或新增模块.nginx的升级有两种方法:1.半自动化升级:2.手动升级 不过都需要先查看安装的nginx版本和配置信息,然后前往官网下 ...
- AngularJS之手动加载模块app和controller
使用ng的页面中一般都是使用模块自动加载,页面的结构一般是这样的 加载angularjs脚本 加载业务代码脚本(或者写在script标签中) html结构代码(带有ng指令) 就像这样 app.htm ...
- 微信小程序(有始有终,全部代码)开发--- 新增模块: 图片选取以及拍照功能
开篇语 前几天发了一篇: <简年15: 微信小程序(有始有终,全部代码)开发---跑步App+音乐播放器 > 后来又发了BUG修复的版本: 简年18: 微信小程序(有始有终,全部代码)开发 ...
- RDIFramework.NET ━ .NET快速信息化系统开发框架 V3.2->Web版本模块管理界面新增模块排序功能
模块(菜单)的排序是每个系统都必须要有的功能,我们框架模块的排序在业务逻辑中已经体现. WinForm版本可以直接在界面上对模块进行排序以控制模块展示的顺序.Web版本在3.2版本中也新增了直接可以模 ...
- ABP新增模块可能遇到的问题
当我们新增一个模块时: public class SSORedisModule: AbpModule { //public override void PreInitialize() //{ // b ...
- 调试经验分享-让自己的电脑充当WI-Fi模块,用来抓取连接Wi-Fi模块APP上的通信数据
需求 手头有了厂家的APP和Wi-Fi模块 在已经知道APP是通过TCP连接Wi-Fi模块(8266), 同时也知道了连接的端口号的 情况下如何知道厂家的APP发送给Wi-Fi模块的数据 打开自己的笔 ...
- freeswitch python模块
概述 freeswitch支持多种语言的业务开发,包括C/C++,java,python,js,lua,Golang等等.freeswitch在使用python做业务开发时,有俩种接入方式,一种是ES ...
随机推荐
- 衡兰芷若成绝响,人间不见周海媚(4k修复基于PaddleGan)
一代人有一代人的经典回忆,1994年由周海媚.马景涛.叶童主演的<神雕侠侣>曾经风靡一时,周海媚所诠释的周芷若凝聚了汉水之钟灵,峨嵋之毓秀,遇雪尤清,经霜更艳,俘获万千观众,成为了一代人的 ...
- ubuntu20.04安装goaccess实时对nginx日志进行分析
ubuntu20.04安装goaccess实时对nginx日志进行分析 goaccess可以对nginx日志进行分析,生成实时动态页面,同时通过nginx反向代理来解决WebSocket数据传输问题. ...
- Java在指定路径下执行cmd命令的方法
目前状态:毕业设计ing 背景: 做毕设时,由于需要将python的运行效果展示出来,所以使用了Java写了一个前端的界面.但是在使用Java对python的脚本进行调用时就尴尬了,出错-- 这里也许 ...
- LeetCode227:基本计算器|| (栈、模拟)
解题思路:两个双端队列模拟,一个存放操作数 a,另一个存放操作符 op,如果找到另一个操作数b,判断操作队列队尾是否是*/,是的话执行 a(*or/)b.遍历完字符串,如果操作符队列非空,说明还有+- ...
- All in One, 快速搭建端到端可观测体系
本文分享自华为云社区<All in One, 快速搭建端到端可观测体系>,作者:王磊. 随着云原生技术的应用,可观测成为云服务的主角,应用程序的部署密度及变化频率较传统环境有着巨大的变化, ...
- 数字孪生和GIS融合为智慧水利带来了什么?
数字孪生和GIS的融合为智慧水利领域带来了巨大的变革和机遇.数字孪生技术的出现,使得水利管理者和决策者能够更好地理解和管理水资源,提高水利设施的效率和可持续性.而GIS作为空间数据管理和分析工具,为数 ...
- java注释、变量、数据类型和运算符
注释 单行注释:// 多行注释:/*开头,*/结尾 JavaDoc注释:/**开头,*/结尾 快捷键:ctrl + ? 变量 第一步:声明变量.即根据数据类型在内存分配空间. 第二步:赋值.即将数据的 ...
- 文心一言 VS 讯飞星火 VS chatgpt (167)-- 算法导论13.1 7题
七.用go语言,试描述一棵含有 n 个关键字的红黑树,使其红色内部结点个数与黑色内部结点个数的比值最大.这个比值是多少? 该比值最小的树又是怎样呢?比值是多少? 文心一言: 红黑树是一种自平衡的二叉查 ...
- 在eclipse中拖动项目到Tomcat服务器中报错:Project facet Java version 16 is not supported.解决办法
补充,还有一种情况:拖不进来,但是根本不报错,解决办法:
- Python——第三章:函数的返回值
函数的返回值: 函数执行之后. 会给调用方一个结果. 这个结果就是返回值 关于return: 函数只要执行到了return. 函数就会立即停止并返回内容. 函数内的return的后续的代 ...
