介绍
顶顶通呼叫中心中间件(mod_cti)实现了一套http api 接口,方便web前端(浏览器直接和mod_cti通讯)直接调用,用来管理和配置系统。后端开发(java、php、c#)等建议直接操作redis,实现系统管理和配置。
基于这个接口开发的管理后台演示地址 http://ddcti.com:88 密码 abc
获取token
为了安全性,建议后端先获取token,然后传递给前端。如果300秒内不使用会自动过期,如果使用这个token调用了任意API,过期时间会延期300秒。
接口
/api?action=GetToken&key=${key}&organization=租户&acl=
- key 的获取请看 httpcli
- organization 关联的租户,如果organization为空,就是不使用多租户功能。如果使用了多租户功能,分机名字,线路名字等都会自动添加@${organization} 的后缀,并且只能访问这个租户的下的信息。系统级别的配置通过这个token是没权限访问的。
- acl 访问控制,就是哪些IP可以使用这个token,不设置,就是任意IP都可以。acl格式,支持fs的acl.conf.xml预定的,也可以 “192.168.1.1/32,192.168.0.0/24” ip/掩码这样的格式。
- inactivity 最大不活动时间(单位秒),多久不使用这个token就自动过期,默认300秒。
- expire 最大有效期 (单位秒),超过这个时间 token强制过期,默认0永不强制过期。
- 响应
{"token":"C797707B-2A70-44cb-BA3D-29EFA6C946C7"} |
提示:后端直接调用本接口的API也可以直接用key认证,不用获取token,前段调用,强烈建议用token认证,key认证的方法就是把 token=${token} 改成key=${key}
错误信息
如果请求接口错误,返回json的错误信息,并且http的响应代码不是200。http错误代码 400:请求参数不对, 401:认证失败。
{“error”:"错误信息"} |
获取cti模块信息
/api?action=GetCtiState&token=${token}
获取系统函数列表
/api?action=GetAssist&token=${token}&name=api|application
获取分机名列表(获取配置名称)
/api?action=GetExtenNames&token=${token}
action 请求的动作,其他类似动作有以下这些:
- GetExtenNames 获取分机列表
- GetLineNames 获取线路列表
- GetLineGroupNames 获取线路组列表
- GetQueueDialerTaskNames 获队列外呼任务列表
- GetScheduleDialerConfigNames 获取定时外呼配置列表
- GetAcdNames 获取ACD排队组列表
- GetGatewayNames 获取网关配置列表
- GetTemplateNames 获取配置模板列表
- GetDialplanExtensionNames 获取拨号方案列表
- GetDialplanContextNames 获取路由列表
- GetSipProfileNames 获取sip配置列表
- GetConfigFileNames 获取配置文件列表
响应
{ |
- action 执行的动作
- result 动作返回的信息,本例子就是分机列表。
获取分机信息(获取配置详细信息)
/api?action=GetExtenInfos&token=${token}&name=分机名
name 要查询的分机名字
多个分机名可以用逗号隔开比如name=120,121。也可以用post 方式提交请求的分机名列表,可以同时请求多个分机信息,如果只请求一个也需要用数组的格式。
["120","121"]
action 请求的动作,其他类似动作有以下这些:
- GetExtenInfos 获取分机信息
- GetLineInfos 获取线路信息
- GetLineGroupInfos 获取线路组信息
- GetQueueDialerTaskInfos 获取队列外呼信息
- GetScheduleDialerConfigInfos 获取定时外呼配置信息
- GetAcdInfos 获取ACD排队信息
- GetGatewayInfos 获取网关配置信息
- GetTemplateInfos 获取配置模板信息
- GetDialplanExtensionInfos 获取拨号方案信息
- GetDialplanContextInfos 获取拨路由信息
- GetSipProfileInfos 获取sip配置信息
- GetConfigFileInfos 获取配置文件信息
响应
{ |
action 执行的动作
result 动作返回的信息,本例子就是分机信息。返回的格式是 “名字”:”json对象的信息”。
设置分机信息(设置配置信息)
/api?token=${token}&action=setexteninfo&name=分机名
name 要设置的分机名字
post 内容是分机信息的JSON格式数据。
{ |
action 请求的动作,其他类似动作有以下这些:
- SetExtenInfo 设置分机信息
- SetLineInfo 设置线路信息
- SetLinePause设置线路暂停(示忙)
- SetLineGroupInfo 设置线路组信息
- SetQueueDialerTaskInfo 设置队列外呼信息
- SetScheduleDialerConfigInfo 设置定时外呼配置信息
- SetAcdInfo 设置ACD排队信息
- SetGatewayInfo 设置网关配置信息
- SetTemplateInfo 设置配置模板信息
- SetDialplanExtensionInfo 设置拨号方案信息
- SetDialplanContextInfo 设置路由信息
- SetSipProfileInfo 设置sip配置信息
- SetConfigFileInfo 设置配置文件信息
响应
{ |
- action 执行的命令
- message 如果分机之前已经存在,是更新信息 message是 “update succeed”,如果之前不存在是增加分级 message是”added succeed”。
删除分机(删除配置)
/api?action=DelExtens&token=${token}&name=分机名
name 要查询的分机名字
多个分机名可以用逗号隔开比如name=120,121。也可以用post 方式提交删除的分机名列表,可以同时删除多个分机,如果只删除一个也需要用数组的格式。
["120","121"]
action 请求的动作,其他类似动作有以下这些:
- DelExtens 删除分机
- DelLines 删除线路
- DelLineGroups 删除线路组
- DelQueueDialerTasks 删除列外呼任务
- DelScheduleDialerConfigs 删除定时外呼配置
- DelAcds 删除ACD排队组
- DelGateways 删除网关配置
- DelTemplates 删除配置模板
- DelDialplanExtensions 删除拨号方案
- DelDialplanContexts 删除路由
- DelSipProfiles 删除sip配置
- DelConfigFiles 删除配置文件
响应
{ |
- action 执行的动作
- result 成功删除的个数。
获取线路状态(获取状态)
/api?token=${token}&action=LineState&name=分机名
name 要查询的线路名
action 请求的动作,其他类似动作有以下这些:
- ExtenState 查询分机注册状态
- LineState 查询线路状态
- LineGroupState 查询线路组状态
- AcdState 查询排队状态
- QueueDialerState 查询队列外呼状态
- ScheduleDialerState 查询定时外呼状态
- QcState 查询质检状态
- GatewayState 查询网关注册状态
- SipProfileState 查询sip状态
- SipStatus 查询所有sip信息
响应 内容是 线路配置信息和线路状态信息的集合
{
"params": {
"count": 2,
"use": 0,
"rest": 0,
"fault_threshold": 0,
"fault_try_interval": 0,
"fault_count": 0,
"fault_condition": {
"connect_time": 0
},
"available": false,
"dialstring": "user/120",
"cluster_dialstring": "",
"active_time": "-23.89",
"areacode": [],
"type": "user",
"device": "120"
},
"variables": {
"originate_timeout": "60",
"employee_id": "120",
"cti_line_name": "120"
},
"owner_groups": {
"2000": {
"priority": 0
}
}
}
- owner_groups 加入的线路组
- priority 在线路组的优先级
获取队列外呼任务状态
/api?token=${token}&action=QueueDialerState &name=任务名
name 要查询的任务名
响应 内容是 线路配置信息和线路状态信息的集合
{
"description": "接通后转接到8001排队",
"systeminfo": { //系统信息
"datatime": "2024-12-23 19:16:10",//系统时间
"cps": 100, //最大CPS
"auth_limit": 1000, //授权并发
"pause": false, //暂停外呼
"idle_cpu": 0, //空闲CPU
"min_idle_cpu": 10, //最小空闲CPU
"loadavg": 0, //系统负载
"max_loadavg": 1.5 //最大负载
},
"params": {
"enable": true, //启用状态
"wait_destroy": false, //等待删除
"cps": 100, //最大CPS
"actual_cps": 100, //实际CPS
"current_cps": 0, //当前CPS
"limit": 1, //最大并发
"call_count": 0, //当前并发
"answer_count": 0, //应答并发
"bridge_count": 0, //桥接并发
"start_time": "2021-05-17 18:51:41", //开始时间
"stop_time": "2030-05-17 18:51:42", //停止时间
"work_hour": [ //呼叫时间
{
"wday": [
0 //星期几
],
"begin": "08:00:00", //上班时间
"end": "23:00:00" //下班时间
}
],
"work_week": [
0 //工作日
],
"holiday": [], //节假日
"last_check_allow_call_time": "2024-12-23 17:50:27", //检测时间(任务是否在允许呼叫时间范围内的最后检测时间)
"last_check_allow_call_result": false, //允许呼叫(任务是否在允许呼叫时间范围内的最后检测结果)
"last_check_is_service_block_result": false, //闭塞状态(根据线路并发、空闲坐席、排队限制、CPS检测是否允许呼叫)
"line_group": [
"trunk" //外呼中继
],
"line_group_idle_count": 1000, //空闲中继
"number_queue": "number_3000", //号码队列
"number_cacheable": 0, //号码缓存设置
"number_cacheable_count": 0, //已缓存号码
"number_cacheable_request": 0, //请求中号码
"number_queue_ready": false, //取号状态
"destination_extension": "8001", //目的地
"destination_dialplan": "XML", //拨号方案
"destination_context": "acd", //路由
"service_line_group": "user", //坐席组
"service_line_group_idle_count": 0, //空闲坐席
"call_multiple": 2, //呼叫倍数
"queue_limit": 0.5, //排队限制
"cps_multiple": 0, //CPS限制
"blacklist": "", //黑名单
"ttsconfig": "" //TTS设置
},
"variables": {
"call_source": "queuedialer", //呼叫来源
"source_name": "3000", //来源名字
"origination_nested_vars": "false", //嵌套变量
"no_wait_answer": "true" //不等待应答
}
}
线路示忙
/api?token=${token}&action=SetLinePause&name=线路名
name 要设置的线路名字
post 内容 0 :示闲 1:示忙。
1
声音文件管理
storage 有 redis和db两种类型,建议使用db存储。
存储在redis的文件 细节可以看 放音文件说明 ,上传会自动从redis同步到硬盘,存储目录为 cti.json配置 sounds_dir 指定。
存储在db的文件 通过放音路径 db://文件路径/文件名
的方式访问。放音的时候会缓存到 fs的storage_dir/db_file_cache 目录。
列出声音文件
/api?token=${token}&action=ListAudiofilenames&dir=目录&type=类型&storage=存储
也支持post发送请求参数{
"dir":"",
"type":"",
"storage":"db"
}
- dir 如果不设置就是列出根目录
- type 列出文件还是文件夹,不设置就是全部
- file 文件
- folder 文件夹
storage
- redis 从redis读取数据
- db 从数据库读取数据
响应
[
{
"name": "2.wav",
"type": "file",
"filemtime": "2023-08-27T15:33:20",
"filesize": "97324",
"description": ""
},
{
"name": "a2",
"type": "folder",
"filemtime": "2023-08-27T15:39:40",
"filesize": "",
"description": ""
}
]name 名字
filemtime 修改时间
filesize 文件大小(type:folder 是目录的意思,没有大小)
description 描述
创建声音文件目录
/api?token=${token}&action=MakeAudiofileDir&dir=父目录&storage=db&description=描述
上传声音文件
表单方式上传 Content-Type 为 multipart/form-data
/api?token=${token}&action=UploadAudiofile&dir=父目录&storage=db&description=描述
post内容为表格格式的文件数据
二进制上传 Content-Type 为 application/binary
/api?token=${token}&action=UploadAudiofile&dir=父目录&storage=db&description=描述&filename=文件名
post内容为二进制的文件内容
下载声音文件
/api?token=${token}&action=DownloadAudiofile&dir=父目录&name=文件名&storage=db
删除声音文件
/api?token=${token}&action=DeleteAudiofile&dir=父目录&name=文件名&storage=db&type=file|folder
改名声音文件,(storage=redis,不支持改名操作)
/api?token=${token}&action=RenamedAudiofile&dir=父目录&name=文件名&newdir=新目录&newname=新名字&storage=db&type=file|folder
修改备注
/api?token=${token}&action=ChangeDescriptionAudiofile&dir=父目录&name=文件或者目录名&storage=db&description=描述
列出redis全部文件
/api?token=${token}&action=ListAllAudiofilenamesFromRedis
列出db全部文件
/api?token=${token}&action=ListAllAudiofilenamesFromDB
CDR查询接口
获取CDR字段
/api?token=${token}&action=listCdrField
响应{
"field": "uuid",
"type":"number|string"
}
- type
- numbr 对应 int,double,float
- string 对应 char,varchar
- datatime 支持的比较方法和number一样
查询CDR记录
/api?token=${token}&action=listCdrRecord
post查询条件
{ |
- mode 查询模式
- 查询单条记录
- 按主叫查询(主叫记录和查询条件匹配就输出记录)
- 按被叫查询(被叫记录和查询条件匹配就输出记录)
- 按主叫或被叫查询(主叫或被叫和查询条件匹配就输出记录)
- 按被叫和被叫查询(主叫和被叫都和查询条件匹配就输出记录)
- field 要显示的字段,不设置就显示所有字段
- offset 分页开始记录
- limit 最大返回多少套记录
- order_field 排序字段
- order_method 排序方法
- asc 顺序
- desc 逆序
- filter_mode 多个filter_condition条件的组合方式。
- and 所有条件满足
- or 任意一个条件满足
- xor 所有条件都满足或者所有条件都不满足
- filter_condition 条件,就是sql的where
- field 要匹配的数据库字段
- value 比较的值,如果字段是int或者浮点型,value要用number类型,日期时间和字符串,都是字符串类型。
- method 比较方法
=
等于number和string都支持!=
不等于number和string都支持>=
只能用于number<=
只能用于numbercontain
只能用于string,filed的包含valueprefix
只能用于string,filed的前缀等于valuesuffix
只能用于string,filed的后缀等于valueregexp
只能用于string,filed和value符合正则表达式,注意不是perl正则语法是POSIX标准,不支持\d,改用[0-9]或者[:digit:],- [:alpha:] 代表任意英文大小写字符 a-z A-Z
- \w – [:alnum:] 代表所有的大小写英文字符和数字0-9 A-Z a-z
- \W – ^[:alnum:]
- \s – [:space:]
- \S – ^[:space:]
- null 为空 number和string都支持
- notnull 不为空 number和string都支持
返回{
"message": "succeed",
"action": "ListCdrRecord",
"result": [
{
"uuid": "cf121f93-d510-460a-b995-b70463f777fa",
"channel": "sofia/internal/121@192.168.31.57",
//mode 1或者2的时候_brother是另外一段的记录
"_brother":{
"uuid":"",
"channel":""
}
},
{
"uuid": "3e6dbf04-caa1-4a1a-8ff1-dcf8d113253b",
"channel": "sofia/internal/121@192.168.31.57"
}
],
"count": 592
}
- result CDR记录
- count 符合条件总的记录数,用于分页。
保存CDR查询条件记录
/api?token=${token}&action=SetCdrHistory&name=记录名字
获取CDR查询条件记录
/api?token=${token}&action=GetCdrHistory
下载通话录音文件
/api?token=${token}&action=DownloadRecordfile&name=cdr查询到的uuid
导入号码到外呼队列
/api?token=${token}&action=ImportQueueDialerNumber&name=队列名字
post内容 要导入的号码数组。
[ |
响应
{ |
- result 导入成功的号码总数
终端监控
需要ws连接才支持
/api?token=${token}&action=FsTerminal&inactivity =600
inactivity ws连接不活动时间,就是多久不执行命令连接就断开,默认600秒。
打开日志
发送
{ |
- level
- “DISABLE”,
- “CONSOLE”,
- “ALERT”,
- “CRIT”,
- “ERR”,
- “WARNING”,
- “NOTICE”,
- “INFO”,
- “DEBUG”,
- filter 过滤关键词,就是日志中包含了指定关键词才输出
- expire 订阅日志最大时间,时间到了自动停止日志,单位秒
关闭日志
发送
{ |
执行FreeSWITCH 命令
发送
{ |
常用FreeSWITCH命令
- 挂机
uuid_kill uuid
- 更多命令可以看 https://developer.signalwire.com/freeswitch/FreeSWITCH-Explained/Modules/mod_commands_1966741/
发起呼叫
/api?token=${token}&action=call
例子1. 点击拨号,推荐使用例子4,接通后转接到拨号方案,可以自动处理录音
{ |
例子2. 语音通知
{ |
例子3. 语音通知,开启录音和限制最大通话时间
{ |
例子4. 呼叫分机接通后执行拨号方案
{ |
- dial
- dialstring 拨号串,如果设置了,line和number参数就无效,如果不设置会使用line和number参数。类似这样的格式:user/120,sofia/external/abc@ip
- line 发起呼叫使用的线路或者线路中 line/线路名 linegroup/线路组名
- number 被叫号码
- variables 变量列表
- bridge 用于先呼叫一个号码A,A接通后呼叫号码B。
- actions 电话接通后要执行的命令列表,比如放音。如果设置了bridge,actions会被忽略。
- 动作名字:动作参数,多个参数空格隔开,如果参数中包含逗号,可以用 动作名:^^|参数1|参数2 这样的方法转义。
- notifyurl 呼叫进度通知回掉
- 呼叫失败,失败原因的含义看CDR文档,呼叫失败没办法返回UUID,可以通过设置变量 origination_uuid=自定义UUID,来自己定义UUID,然后把UUID放到回掉URL的参数里面来解决呼叫失败通知无UUID。
{"call":"failed","cause":"呼叫失败原因"}
- 收到183
{"call":"early","uuid":"4c613373-aeda-484a-8ce3-11f1ed6e8b96"}
- 接通
{"call":"answer","uuid":"6ab1403d-de06-4ac3-8a9c-6e5960c0aa6b"}
- bridge端收到183
{"call":"bridge","uuid":"6ab1403d-de06-4ac3-8a9c-6e5960c0aa6b","peer_uuid":"cb089768-a192-40af-89af-8d9c37b75344"}
- uuid dial端的UUID
- peer_uuid bridge端的UUID
- bridge端接通
{"call":"peer_anser","uuid":"6ab1403d-de06-4ac3-8a9c-6e5960c0aa6b","peer_uuid":"cb089768-a192-40af-89af-8d9c37b75344"}
- 挂断
{"call":"hangup","uuid":"6ab1403d-de06-4ac3-8a9c-6e5960c0aa6b","hangup_cause":"NORMAL_CLEARING","peer_hangup_code":"200","hangup_code":"","time":{"created":1695717657898439,"progress":1695717658416816,"progress_media":1695717657996699,"answered":1695717666178472,"hungup":1695717675921398}}
- hangup_code dial端的挂机代码,如果挂机代码为空,就是fs主动发起的挂机
- peer_hangup_code bridge端的挂机代码,如果挂机代码为空,就是fs主动发起的挂机
- 常用变量说明
- origination_caller_id_number 设置主叫号码
- origination_uuid 如果需要使用自定义的呼叫UUID,通过这个参数传给FS。
- record_filename 写入CDR表的录音路径
- execute_on_answer=’record_session $${recordings_dir}/20210303/${uuid}.wav’ 接通后开始录音,$${recordings_dir}/日期目录/${uuid}.wav 这个是录音文件路径
- $${recordings_dir} vars.xml 配置的录音文件路径 默认是 fs安装目录 recording
- ${caller_id_number} 来电号码
- ${uuid} 通话callid
- ${strftime(%Y-%m-%d)} 日期
- ignore_early_media=true 忽略早期媒体,如果坐席使用手机接听,需要这个参数,也就是先呼叫的号码,如果是手机,需要加上这个参数,如果需要录制早期媒体,需要改成 ignore_early_media=consume。
- absolute_codec_string=g729 指定g729编码,如果需要同时指定多个编码可以设置为
^^:G729:PCMA:PCMU
^^:意思是把逗号用:代替,类似转义符。 - originate_timeout 呼叫超时(比如分机30秒不接听就停止呼叫),需要和ignore_early_media=true 一起使用。
- instant_ringback=true 安装自定义彩铃
- transfer_ringback=$${cn-ring} 设置自定义彩铃声音,可以指定具体的文件名,$${cn-ring}使用嘟嘟声音,如果不和ignore_early_media=true一起使用,线路响应了183,就会播放线路的提示音。
- origination_nested_vars=true 允许origiante参数不解析变量,变量格式\${变量名}。
- export_vars 把变量传递到后续桥接的通道,比如export_vars=‘record_filename,其他变量名’ ,如果只需要设置变量到后续桥接的通道,用前缀
nolocal:
例子:{nolocal:sip_h_X-AutoAccept=true,export_vars=’nolocal:sip_h_X-AutoAccept’} sip_h_*
添加自定义sip头到 INVITE ;如果值里面又逗号用\转义,例子 sip_h_X-My-Header=one\,two\,three。sip_rh_*
添加自定义sip头到 200sip_ph_*
添加自定义sip头到 180 183sip_bye_h_*
添加自定义sip头到 bye
获取帮助信息
/api?token=${token}&action=GetAssist
获取系统信息
/api?token=${token}&action=SystemStatus
响应{
"recordings_dir": "D:/src/utility/freeswitch/x64/Debug/recordings",
"sounds_dir": "D:/src/utility/freeswitch/x64/Debug/sounds",
"sessions_total": 0,
"sessions_count": 0,
"sessions_peak": 0,
"sessions_peak_fivemin": 0,
"sessions_limit": 10000,
"sps": 0,
"sps_limit": 1000,
"sps_speak": 0,
"sps_peak_fivemin": 0,
"loadavg": 0,
"idle_cpu": 82.25260416666667,
"uptime": 169382,
"total_memroy": 34211385344,
"avail_memory": 15465250816
}
- recordings_dir 录音路径,默认录音文件路径
- sounds_dir 放音路径,默认放音文件路径
- sessions_total 累计呼叫,启动以来总的呼叫
- sessions_count 当前并发,当前总呼叫
- sessions_peak 并发峰值,最高呼叫并发
- sessions_peak_fivemin 最近并发峰值, 最近5分钟最高呼叫并发
- sessions_limit 最大并发,最大允许的呼叫并发
- sps 当前CPS,最后1秒呼叫数
- sps_speak CPS峰值, 最高每秒呼叫数
- sps_peak_fivemin 最近CPS峰值,最近5分钟 最高每秒呼叫数
- sps_limit 最大CPS,最大允许每秒呼叫数
- uptime 运行时间,系统运行时间(秒)
- loadavg 负载,系统最近1分钟负载
- idle_cpu 空闲CPU
- total_memroy 系统总内存
- avail_memory 系统可用内存
列出所有API
/api?token=${token}&action=GetAssist&name=api
列出所有APP
/api?token=${token}&action=GetAssist&name=application
获取cti配置
/api?token=${token}&action=GetCtiConfig
返回cti.json配置,会自动去了注释。
设置cti配置
/api?token=${token}&action=SetCtiConfig
post内容就是cti.json的内容。
设置cti.json配置,配置会自动保存到fs安装目录conf/cti.json。
导出redis配置备份
/api?token=${token}&action=ExportRedisConfig&audiofile=no&sip=yes
可以通过传入参数控制导出那些配置 yes导出no不导出,没设置的参数,默认no。
audiofile 声音文件
sip sip配置
exten 分机配置
gateway 网关配置
dialplan 拨号方案和路由
line 线路和线路组
acd 排队
queuedialer 队列外呼
scheduledialer 定时外呼
robotflow 话术流程
callqc 质检配置
configfile 配置文件
template 模板文件
导入之前备份的配置
/api?token=${token}&action=ImportRedisConfig&reset=false&audiofile=no&sip=yes
可以通过传入参数控制导出那些配置 yes导出no不导出,没设置的参数,默认no。
- reset 导入配置的时候是否先清空之前的配置项目。yes 先清空在写入,no 覆盖同名和插入新的项目。
声音转换接口
表单方式上传 Content-Type 为 multipart/form-data
post内容为表格格式的文件数据
二进制上传 Content-Type 为 application/binary
post内容为二进制的文件内容URL中指定文件路径 filename=文件路径
/api?token=${token}&action=AudioConvert&method=format|cut
上传声音文件去除首尾静音并且转换成单声道wav格式
/api?token=${token}&action=AudioConvert&method=format&reserve_silence=20&threshold_volume=200&minimum_voice=10&sample_rate=8000
method必须是format
reserve_silence 声音首尾保留的静音时间单位毫秒,默认20
threshold_volume 静音阈值,默认200
minimum_voice 声音时间大于多少认为声音开始 默认10
sample_rate 转换后输出的声音频率 默认8000
上传声音文件返回截取后的声音
/api?token=${token}&action=AudioConvert&method=cut&begintime=1000&endtime=8000&channel=0
method必须是cut
begintime 截取声音的开始时间
- endtime 截取声音的结束时间
- channel 截取哪个声道 第一声道 0,第二声道1