api.py 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803
  1. #!/usr/bin/env python3
  2. # encoding:utf-8
  3. import json
  4. import sys
  5. from typing import Dict, Any, Optional, List
  6. from src.core.callcenter.constant import get_json_dict
  7. from src.core.callcenter.esl.constant.sip_header_constant import sipHeaderCallId, sipHeaderCaller, sipHeaderCalled, \
  8. sipHeaderDeviceId, sipHeaderDeviceType
  9. from src.core.callcenter.exception import SipUriSyntaxException
  10. from src.core.datasource import SIP_SERVER
  11. from src.core.callcenter.esl.constant.esl_constant import SPLIT, SIP_HEADER
  12. from src.core.callcenter.enumeration import CallType, DeviceType, Direction, CdrType, NextType
  13. class BaseApi:
  14. @classmethod
  15. def from_json(cls, json_string):
  16. data = get_json_dict(json_string)
  17. return cls(**data)
  18. def to_json_string(self):
  19. return json.dumps(self, default=lambda x: x.__dict__, ensure_ascii=False)
  20. def __repr__(self):
  21. return self.to_json_string()
  22. class AgentCallRequest(BaseApi):
  23. """
  24. 呼叫请求对象
  25. """
  26. def __init__(self, saas_id, agent_id, call_type: CallType, caller, called, caller_display="", called_display="",
  27. cti_flow_id=None, uuid1=None, uuid2=None, follow_data: Dict[str, Any] = {}):
  28. # 租户id
  29. self.saas_id = saas_id
  30. # 坐席号
  31. self.agent_id = agent_id
  32. # 呼叫类型
  33. self.call_type: CallType = call_type
  34. # 主叫,如果没有传,则使用坐席号码
  35. self.caller = caller
  36. # 被叫
  37. self.called = called
  38. # 主叫显号,接口没有传就用主技能组配置
  39. self.caller_display = caller_display
  40. # 被叫显号,接口没传就用主技能配置
  41. self.called_display = called_display
  42. self.cti_flow_id = cti_flow_id
  43. # uuid1
  44. self.uuid1 = uuid1
  45. # uuid2
  46. self.uuid2 = uuid2
  47. # 随路数据
  48. self.follow_data = follow_data
  49. class AgentRequest(BaseApi):
  50. def __init__(self, saas_id, agent_number, agent_name, out_id, agent_password, agent_type, phone_number, distribute,
  51. agent_state, identity_type):
  52. # 租户隔离
  53. self.saas_id = saas_id
  54. # 坐席工号, 生成返回
  55. self.agent_number = agent_number
  56. # 坐席名称
  57. self.agent_name = agent_name
  58. # 外部展示Id
  59. self.out_id = out_id
  60. # 坐席密码
  61. self.agent_password = agent_password
  62. # 坐席类型 0:普通坐席 ;1:组长:2:主管
  63. self.agent_type = agent_type
  64. # 分机号
  65. self.phone_number = phone_number
  66. # 分配标志 0:不参与排队;1:参与排队
  67. self.distribute = distribute
  68. # 账号状态 0:可用;1:禁用
  69. self.agent_state = agent_state
  70. # 身份状态
  71. self.identity_type = identity_type
  72. class AgentQueryRequest(BaseApi):
  73. def __init__(self, saas_id, agent_number, out_id, agent_type, agent_name, page=1, size=10):
  74. # 租户隔离
  75. self.saas_id = saas_id
  76. # 坐席工号
  77. self.agent_number = agent_number
  78. # 坐席名称
  79. self.agent_name = agent_name
  80. # 外部展示Id
  81. self.out_id = out_id
  82. # 坐席类型 0:普通坐席 ;1:组长:2:主管
  83. self.agent_type = agent_type
  84. self.page = page,
  85. self.size = size
  86. class AgentActionRequest(BaseApi):
  87. """
  88. 坐席操作
  89. """
  90. def __init__(self, saas_id, agent_id, agent_number=None, out_id=None, identity_type=None, scene='manual', service_id=None, **kwargs):
  91. self.saas_id =saas_id
  92. # 坐席工号
  93. self.agent_id = agent_id
  94. # # 坐席工号
  95. self.agent_number = agent_number
  96. # 外部展示Id
  97. self.out_id = out_id
  98. # 身份状态
  99. self.identity_type = identity_type
  100. # 场景 manual:手动外呼; robot:机器人外呼; monitor:监听
  101. self.scene = scene
  102. self.service_id = service_id
  103. # 处理多余的参数
  104. for key, value in kwargs.items():
  105. setattr(self, key, value)
  106. class HumanServiceQueryRequest(BaseApi):
  107. """人工组查询"""
  108. def __init__(self, saas_id, service_id="00000000000000000", agent_state=None, agent_ids: Dict[str, Any] = {}, **kwargs):
  109. self.saas_id = saas_id
  110. self.serviceId = service_id
  111. self.agentState = agent_state
  112. self.agentIds = agent_ids
  113. # 处理多余的参数
  114. for key, value in kwargs.items():
  115. setattr(self, key, value)
  116. class AgentMonitorData(BaseApi):
  117. """坐席状态信息"""
  118. def __init__(self, saas_id=None, agent_num=None, agent_name=None, out_id=None,
  119. check_state=None, online_state=None, check_in_time=None, check_out_time=None,
  120. service_state=None, busy_time=None, idle_time=None, call_time=None, hang_time=None,
  121. heart_state=None, heart_time=None, update_time=None, check_scene=None):
  122. self.saas_id = saas_id
  123. self.agent_num = agent_num
  124. self.agent_name = agent_name
  125. self.out_id = out_id
  126. self.check_state = check_state
  127. self.online_state = online_state
  128. self.check_in_time = check_in_time
  129. self.check_out_time = check_out_time
  130. self.service_state = service_state
  131. self.busy_time = busy_time
  132. self.idle_time = idle_time
  133. self.call_time = call_time
  134. self.hang_time = hang_time
  135. self.heart_state = heart_state
  136. self.heart_time = heart_time
  137. self.update_time = update_time
  138. self.check_scene = check_scene
  139. class AgentEventData(BaseApi):
  140. def __init__(self, app_code, user_id, data: Dict[str, Any] = {}):
  141. self.appCode = app_code
  142. self.userId = user_id
  143. self.data = data
  144. class AgentStateData(BaseApi):
  145. def __init__(self, _s=None, _t=None, status=None, time=None, assign_time=None, phone_num=None):
  146. self._s = _s
  147. self._t = _t
  148. self.status = status
  149. self.time = time
  150. self.assign_time = assign_time
  151. self.phone_num = phone_num
  152. class HangupCallRequest(BaseApi):
  153. def __init__(self, saas_id, call_id, agent_number):
  154. # saasId(必填)
  155. self.saas_id = saas_id
  156. # 呼叫唯一id(选填)
  157. self.call_id = call_id
  158. # 分机号(必填)
  159. self.agent_number = agent_number
  160. class CheckInCallRequest(BaseApi):
  161. """检查是否在通话中实体"""
  162. def __init__(self, saas_id, agent_number):
  163. self.saas_id = saas_id
  164. self.agent_number = agent_number
  165. class MakeCallContext(BaseApi):
  166. def __init__(self,
  167. sip_server=SIP_SERVER,
  168. route_gateway_name="fxo-gateway",
  169. display: Optional[str] = None,
  170. caller: Optional[str] = None,
  171. called: Optional[str] = None,
  172. call_id: Optional[str] = None,
  173. device_id: Optional[str] = None,
  174. eavesdrop: Optional[str] = None,
  175. device_type: Optional[int] = None,
  176. timeout: Optional[int] = 60,
  177. originate_timeout: Optional[int] = 60,
  178. sip_header_map: Optional[Dict[str, str]] = {},
  179. called_prefix: Optional[str] = "",
  180. service_id: Optional[str] = None,
  181. call_type: Optional[int] = None):
  182. # fs 地址
  183. self.sip_server = sip_server
  184. # 线路名(非必传)
  185. self.route_gateway_name = route_gateway_name
  186. # 外显号(必传)
  187. self.display = display
  188. # 该腿的呼叫者(必传)
  189. self.caller = caller
  190. # 该腿的被呼叫者(必传)
  191. self.called = called
  192. # 该腿的callId (必传)
  193. self.call_id = call_id
  194. # 该腿的uuid(必传)
  195. self.device_id = device_id
  196. # 是否是监听腿 监听腿必填true,其它不填
  197. self.eavesdrop = eavesdrop
  198. # 当前腿的类型(必填 客户 坐席 监听者 机器人)
  199. self.device_type = device_type
  200. # 超时挂断
  201. self.timeout = timeout
  202. # 超时挂断(后面可以考虑删掉)
  203. self.originate_timeout = originate_timeout
  204. # 额外的sip头添加(非必填)
  205. self.sip_header_map = sip_header_map or {}
  206. # 被叫前缀 (用户腿必填)
  207. self.called_prefix = called_prefix
  208. # 任务id(机器人外呼必填)
  209. self.service_id = service_id
  210. # 呼叫类型(必填)
  211. self.call_type = call_type
  212. def get_sip_header(self) -> str:
  213. headers = [
  214. f"{SIP_HEADER}contact_user={self.display}",
  215. "ring_asr=true",
  216. f"origination_caller_id_number={self.display}",
  217. f"origination_caller_id_name={self.display}",
  218. f"origination_uuid={self.device_id}",
  219. "absolute_codec_string=^^:PCMU:PCMA",
  220. ]
  221. if self.originate_timeout is not None:
  222. headers.append(f"originate_timeout={self.originate_timeout}")
  223. headers += [
  224. "RECORD_STEREO=true",
  225. "RECORD_APPEND=true",
  226. f"{SIP_HEADER}{sipHeaderCallId}={self.call_id}",
  227. f"{SIP_HEADER}{sipHeaderCaller}={self.caller}",
  228. f"{SIP_HEADER}{sipHeaderCalled}={self.called}",
  229. f"{SIP_HEADER}{sipHeaderDeviceId}={self.device_id}",
  230. f"{SIP_HEADER}{sipHeaderDeviceType}={self.device_type}",
  231. ]
  232. if self.eavesdrop:
  233. headers.append(f"{SIP_HEADER}eavesdrop={self.eavesdrop}")
  234. if self.service_id:
  235. headers.append(f"{SIP_HEADER}service_id={self.service_id}")
  236. if self.device_type == DeviceType.CUSTOMER.code:
  237. headers += [
  238. "RECORD_STEREO_SWAP=true"
  239. ]
  240. else:
  241. headers += [
  242. "RECORD_STEREO_SWAP=false",
  243. "continue_on_fail=true"
  244. ]
  245. if self.sip_header_map:
  246. headers.extend([f"{SIP_HEADER}{k}={v}" for k, v in self.sip_header_map.items()])
  247. return SPLIT.join(headers)
  248. def get_called(self) -> str:
  249. if self.called_prefix and self.device_type == DeviceType.CUSTOMER.code:
  250. return f"{self.called_prefix}{self.called}"
  251. return self.called
  252. def get_realm(self):
  253. sip_uri = SipURI(self.sip_server)
  254. return sip_uri.get_host()
  255. class SipURI:
  256. DEFAULT_PORT = -1
  257. SIP_SCHEME = "sip"
  258. SCHEME_SEPARATOR = ':'
  259. def __init__(self, sip_uri: str):
  260. self.string_representation = sip_uri
  261. self.userinfo = None
  262. self.host = None
  263. self.port = self.DEFAULT_PORT
  264. self.uri_parameters = {}
  265. scheme = f"{self.SIP_SCHEME}{self.SCHEME_SEPARATOR}"
  266. print('------->', sip_uri, sip_uri.startswith(scheme))
  267. sys.stdout.flush() # 强制刷新输出缓冲区
  268. if not sip_uri.startswith(scheme):
  269. raise SipUriSyntaxException(f"SIP URI must start with {scheme}")
  270. buf = sip_uri[len(scheme):]
  271. at_pos = buf.find("@")
  272. if at_pos == 0:
  273. raise SipUriSyntaxException("userinfo cannot start with a '@'")
  274. if at_pos > 0:
  275. self.userinfo = buf[:at_pos]
  276. buf = buf[at_pos + 1:]
  277. end_hostport = buf.find(";")
  278. if end_hostport == 0:
  279. raise SipUriSyntaxException("hostport not present or it cannot start with ';'")
  280. if end_hostport < 0:
  281. end_hostport = len(buf)
  282. hostport = buf[:end_hostport]
  283. buf = buf[end_hostport:]
  284. colon_pos = hostport.find(":")
  285. if colon_pos > -1:
  286. if colon_pos == len(hostport) - 1:
  287. raise SipUriSyntaxException("hostport cannot terminate with a ':'")
  288. self.port = int(hostport[colon_pos + 1:])
  289. self.host = hostport[:colon_pos]
  290. else:
  291. self.host = hostport
  292. if buf == ";":
  293. buf = ""
  294. while buf:
  295. if buf[0] == ";":
  296. buf = buf[1:]
  297. next_semicolon = buf.find(";")
  298. if next_semicolon < 0:
  299. next_semicolon = len(buf)
  300. next_equals = buf.find("=")
  301. if next_equals < 0 or next_equals > next_semicolon:
  302. next_equals = next_semicolon
  303. key = buf[:next_equals]
  304. value = buf[next_equals + 1:next_semicolon] if next_equals < next_semicolon else ""
  305. self.uri_parameters[key] = value
  306. buf = buf[next_semicolon:]
  307. def __str__(self):
  308. return self.string_representation
  309. def get_host(self):
  310. return self.host
  311. def get_port(self):
  312. return self.port
  313. def get_uri_parameters(self):
  314. return self.uri_parameters
  315. def get_userinfo(self):
  316. return self.userinfo
  317. class AgentInfo(BaseApi):
  318. def __init__(self, saas_id=None, agent_number=None, realm=None, sip_server=None,
  319. call_id=None, device_id=None, real_device_id=None, line_id=None, fs_user=None, domain=None,
  320. service_time=0, max_ready_time=0, total_ready_time=0, ready_times=0, not_ready_times=0,
  321. total_after_time=0, max_talk_time=0, total_talk_time=0, total_ring_times=0, total_answer_times=0):
  322. self.saas_id = saas_id
  323. self.agent_number = agent_number
  324. self.realm = realm
  325. self.sip_server = sip_server
  326. self.call_id = call_id
  327. self.device_id = device_id
  328. # 机器人外呼时真实fs
  329. self.real_device_id = real_device_id
  330. self.line_id = line_id
  331. self.fs_user = fs_user
  332. self.domain = domain
  333. # 坐席最近的一次服务时间,电话则是振铃时间(秒)
  334. self.service_time = service_time
  335. # 最大空闲时长
  336. self.max_ready_time = max_ready_time
  337. # 累计空闲
  338. self.total_ready_time = total_ready_time
  339. # 空闲次数
  340. self.ready_times = ready_times
  341. # 忙碌次数
  342. self.not_ready_times = not_ready_times
  343. # 累计话后时间长
  344. self.total_after_time = total_after_time
  345. # 最大通话时长
  346. self.max_talk_time = max_talk_time
  347. # 当日累计通话时长
  348. self.total_talk_time = total_talk_time
  349. # 振铃次数
  350. self.total_ring_times = total_ring_times
  351. # 当日累计接听次数
  352. self.total_answer_times = total_answer_times
  353. def get_realm(self):
  354. """
  355. Get the realm by extracting the host from the SIP server if available,
  356. otherwise return the current realm.
  357. """
  358. try:
  359. if self.sip_server:
  360. sip_uri = SipURI(self.sip_server)
  361. return sip_uri.get_host() if sip_uri.get_host() else self.realm
  362. except Exception as ex:
  363. print(f"Error parsing SIP server URI: {ex}")
  364. return self.realm
  365. def is_fs_user(self):
  366. """
  367. Determine if the agent is a FreeSWITCH user.
  368. Default is True if fs_user is None or True.
  369. """
  370. return self.fs_user is None or self.fs_user is True
  371. class CallInfo(BaseApi):
  372. def __init__(self, core_uuid=None, cti_flow_id=None, call_id=None, conference=None, saas_id=None, group_id=None,
  373. hidden_customer=0, caller_display=None, caller=None, called_display=None, called=None,
  374. number_location=None, agent_key=None, agent_name=None, login_type=None, ivr_id=None, task_id=None,
  375. media_host=None, sip_server=None, client_host=None, record=None, record2=None, record_time=None,
  376. call_time=None, call_type=None, direction=None, answer_flag=None, wait_time=None, answer_count=0,
  377. hangup_dir=None, sdk_hangup=0, hangup_code=None, answer_time=None, end_time=None, talk_time=None,
  378. first_queue_time=None, queue_start_time=None, queue_end_time=None, overflow_count=0,
  379. uuid1=None, uuid2=None, cdr_notify_url=None, queue_level=None, device_list=[],
  380. device_info_map: Dict[str, Any] = {}, follow_data: Dict[str, Any] = {},
  381. process_data: Dict[str, Any] = {}, next_commands=[], call_details=[]):
  382. self.core_uuid = core_uuid # 通话唯一标识
  383. self.cti_flow_id = cti_flow_id
  384. self.call_id = call_id # 通话唯一标识
  385. self.conference = conference # 会议号
  386. self.saas_id = saas_id # 企业id
  387. self.group_id = group_id # 所在技能组id
  388. self.hidden_customer = hidden_customer # 隐藏客户号码(0:不隐藏;1:隐藏)
  389. self.caller_display = caller_display # 主叫显号
  390. self.caller = caller # 主叫
  391. self.called_display = called_display # 被叫显号
  392. self.called = called # 被叫
  393. self.number_location = number_location # 号码归属地
  394. self.agent_key = agent_key # 坐席
  395. self.agent_name = agent_name # 坐席名称
  396. self.login_type = login_type # 坐席登录类型
  397. self.ivr_id = ivr_id # ivr
  398. self.task_id = task_id # 任务ID
  399. self.media_host = media_host # 媒体
  400. self.sip_server = sip_server # 服务地址
  401. self.client_host = client_host # 客户服务地址
  402. self.record = record # 录音地址
  403. self.record2 = record2 # 第三方存储地址
  404. self.record_time = record_time # 录音开始时间
  405. self.call_time = call_time # 呼叫开始时间
  406. self.call_type = call_type # 呼叫类型 (Enum CallType)
  407. self.direction = direction # 呼叫方向 (Enum Direction)
  408. self.answer_flag = answer_flag # 通话标识
  409. self.wait_time = wait_time # 等待时长
  410. self.answer_count = answer_count # 应答设备数
  411. self.hangup_dir = hangup_dir # 挂机方向 (1主叫挂机, 2:被叫挂机, 3:平台挂机)
  412. self.sdk_hangup = sdk_hangup # sdk挂机标志
  413. self.hangup_code = hangup_code # 挂机原因
  414. self.answer_time = answer_time # 接听时间
  415. self.end_time = end_time # 最后一侧电话挂机时间
  416. self.talk_time = talk_time # 通话时长(秒)
  417. self.first_queue_time = first_queue_time # 第一次进队列时间
  418. self.queue_start_time = queue_start_time # 进入技能组时间
  419. self.queue_end_time = queue_end_time # 出技能组时间
  420. self.overflow_count = overflow_count # 溢出次数
  421. self.uuid1 = uuid1 # uuid1
  422. self.uuid2 = uuid2 # uuid2
  423. self.cdr_notify_url = cdr_notify_url # 话单通知地址
  424. self.queue_level = queue_level # 排队等级,默认是进队列时间
  425. self.device_list = device_list # 当前通话的设备
  426. self.device_info_map = device_info_map
  427. self.follow_data = follow_data # 呼叫随路数据(作为落单数据)
  428. self.process_data = process_data # 模块流程间数据
  429. self.next_commands = next_commands # 执行下一步命令
  430. self.call_details = call_details # 电话流程
  431. @classmethod
  432. def from_json(cls, json_string):
  433. data = get_json_dict(json_string)
  434. device_info_map: Dict[str, Any] = {key: DeviceInfo(**value) for key, value in data.get('device_info_map', {}).items()}
  435. follow_data: Dict[str, Any] = {key: value for key, value in data.get('follow_data', {}).items()}
  436. process_data: Dict[str, Any] = {key: value for key, value in data.get('process_data', {}).items()}
  437. next_commands = [NextCommand(**x) for x in data.get('next_commands', [])]
  438. call_details = [CallDetail(**x) for x in data.get('call_details', [])]
  439. return cls(core_uuid=data.get('core_uuid'), cti_flow_id=data.get('cti_flow_id'), call_id=data.get('call_id'),
  440. conference=data.get('conference'), saas_id=data.get('saas_id'), group_id=data.get('group_id'),
  441. hidden_customer=data.get('hidden_customer'), caller_display=data.get('caller_display'),
  442. caller=data.get('caller'), called_display=data.get('called_display'), called=data.get('called'),
  443. number_location=data.get('number_location'), agent_key=data.get('agent_key'),
  444. agent_name=data.get('agent_name'), login_type=data.get('login_type'), ivr_id=data.get('ivr_id'),
  445. task_id=data.get('task_id'), media_host=data.get('media_host'), sip_server=data.get('sip_server'),
  446. client_host=data.get('client_host'), record=data.get('record'), record2=data.get('record2'),
  447. record_time=data.get('record_time'), call_time=data.get('call_time'),
  448. call_type=data.get('call_type'), direction=data.get('direction'),
  449. answer_flag=data.get('answer_flag'), wait_time=data.get('wait_time'),
  450. answer_count=data.get('answer_count'), hangup_dir=data.get('hangup_dir'),
  451. sdk_hangup=data.get('sdk_hangup'), hangup_code=data.get('hangup_code'),
  452. answer_time=data.get('answer_time'), end_time=data.get('end_time'), talk_time=data.get('talk_time'),
  453. first_queue_time=data.get('first_queue_time'), queue_start_time=data.get('queue_start_time'),
  454. queue_end_time=data.get('queue_end_time'), overflow_count=data.get('overflow_count'),
  455. uuid1=data.get('uuid1'), uuid2=data.get('uuid2'), cdr_notify_url=data.get('cdr_notify_url'),
  456. queue_level=data.get('queue_level'), device_list=data.get('device_list', []),
  457. device_info_map=device_info_map, follow_data=follow_data, process_data=process_data,
  458. next_commands=next_commands, call_details=call_details)
  459. def __str__(self):
  460. return self.to_json_string()
  461. def to_json_string(self):
  462. new_dict = {
  463. "core_uuid": self.core_uuid,
  464. "cti_flow_id": self.cti_flow_id,
  465. "call_id": self.call_id,
  466. "conference": self.conference,
  467. "saas_id": self.saas_id,
  468. "group_id": self.group_id,
  469. "hidden_customer": self.hidden_customer,
  470. "caller_display": self.caller_display,
  471. "caller": self.caller,
  472. "called_display": self.called_display,
  473. "called": self.called,
  474. "number_location": self.number_location,
  475. "agent_key": self.agent_key,
  476. "agent_name": self.agent_name,
  477. "login_type": self.login_type,
  478. "ivr_id": self.ivr_id,
  479. "task_id": self.task_id,
  480. "media_host": self.media_host,
  481. "sip_server": self.sip_server,
  482. "client_host": self.client_host,
  483. "record": self.record,
  484. "record2": self.record2,
  485. "record_time": self.record_time,
  486. "call_time": self.call_time,
  487. "call_type": self.call_type,
  488. "direction": self.direction,
  489. "answer_flag": self.answer_flag,
  490. "wait_time": self.wait_time,
  491. "answer_count": self.answer_count,
  492. "hangup_dir": self.hangup_dir,
  493. "sdk_hangup": self.sdk_hangup,
  494. "hangup_code": self.hangup_code,
  495. "answer_time": self.answer_time,
  496. "end_time": self.end_time,
  497. "talk_time": self.talk_time,
  498. "first_queue_time": self.first_queue_time,
  499. "queue_start_time": self.queue_start_time,
  500. "queue_end_time": self.queue_end_time,
  501. "overflow_count": self.overflow_count,
  502. "uuid1": self.uuid1,
  503. "uuid2": self.uuid2,
  504. "cdr_notify_url": self.cdr_notify_url,
  505. "queue_level": self.queue_level,
  506. "device_list": [x for x in self.device_list],
  507. "device_info_map": {key: vars(value) for key, value in self.device_info_map.items()},
  508. "follow_data": {key: vars(value) for key, value in self.follow_data.items()},
  509. "process_data": {key: vars(value) for key, value in self.process_data.items()},
  510. "next_commands": [vars(x) for x in self.next_commands],
  511. "call_details": [vars(x) for x in self.call_details],
  512. }
  513. return json.dumps(new_dict, default=lambda x: x.__dict__, ensure_ascii=False)
  514. class DeviceInfo(BaseApi):
  515. def __init__(self, cti_flow_id=None, call_id=None, conference=None, device_id=None, agent_key=None, agent_name=None, device_type=None,
  516. cdr_type=0, from_agent=None, caller=None, called=None, display=None, called_location=None,
  517. caller_location=None, call_time=None, ring_start_time=None, ring_end_time=None, answer_time=None,
  518. bridge_time=None, end_time=None, talk_time=None, sip_protocol=None, channel_name=None,
  519. hangup_cause=None, ring_cause=None, sip_status=None, record=None, record_time=None,
  520. record_start_time=None, state=None, apparent_number=None, caller_display=None):
  521. self.cti_flow_id = cti_flow_id
  522. self.call_id = call_id # 通话唯一标识
  523. self.conference = conference # 会议模式
  524. self.device_id = device_id # 设备id
  525. self.agent_key = agent_key # 坐席
  526. self.agent_name = agent_name # 坐席
  527. self.device_type = device_type # 1:坐席,2:客户,3:外线
  528. self.cdr_type = cdr_type # 1:呼入,2:外呼,3:内呼,4:转接,5:咨询,6:监听,7:强插,8:耳语
  529. self.from_agent = from_agent # 咨询或转接来源
  530. self.caller = caller # 主叫
  531. self.called = called # 被叫
  532. self.display = display # 显号
  533. self.called_location = called_location # 被叫归属地
  534. self.caller_location = caller_location # 被叫归属地
  535. self.call_time = call_time # 呼叫开始时间
  536. self.ring_start_time = ring_start_time # 振铃开始时间
  537. self.ring_end_time = ring_end_time # 振铃结束时间
  538. self.answer_time = answer_time # 接通时间
  539. self.bridge_time = bridge_time # 桥接时间
  540. self.end_time = end_time # 结束时间
  541. self.talk_time = talk_time # 通话时长
  542. self.sip_protocol = sip_protocol # 信令协议(tcp/udp)
  543. self.channel_name = channel_name # 呼叫地址
  544. self.hangup_cause = hangup_cause # 挂机原因
  545. self.ring_cause = ring_cause # 回铃音识别
  546. self.sip_status = sip_status # sip状态
  547. self.record = record # 录音地址
  548. self.record_time = record_time # 录音时长
  549. self.record_start_time = record_start_time # 录音开始时间
  550. self.state = state # 当前设备状态
  551. self.apparent_number = apparent_number # 外显号
  552. self.caller_display = caller_display # 主叫号
  553. class NextCommand(BaseApi):
  554. def __init__(self, device_id, next_type, next_value=None):
  555. # 记录执行设备
  556. self.device_id = device_id
  557. # 下一步执行命令
  558. self.next_type = next_type
  559. # 执行参数
  560. self.next_value = next_value
  561. class DelayAction(BaseApi):
  562. def __init__(self, uuid=None, call_id=None, device_id=None):
  563. self.uuid=uuid
  564. self.call_id = call_id
  565. self.device_id = device_id
  566. class CallDetail(BaseApi):
  567. def __init__(self, id=None, cts=None, start_time=None, end_time=None, call_id=None,
  568. detail_index=None, transfer_type=None, transfer_id=None, reason=None,
  569. month=None, ext1=None, ext2=None, status=None):
  570. self.id = id # PK
  571. self.cts = cts # 创建时间
  572. self.start_time = start_time # 开始时间
  573. self.end_time = end_time # 结束时间
  574. self.call_id = call_id # 通话ID
  575. self.detail_index = detail_index # 顺序
  576. self.transfer_type = transfer_type # 类型
  577. self.transfer_id = transfer_id # 转接ID
  578. self.reason = reason # 出队列原因
  579. self.month = month # 月份
  580. self.ext1 = ext1 # 扩展字段1
  581. self.ext2 = ext2 # 扩展字段2
  582. self.status = status # 状态
  583. class RouteGateway(BaseApi):
  584. def __init__(self, id=None, saas_id=None, cts=None, uts=None, name=None, media_host=None,
  585. media_port=None, caller_prefix=None, called_prefix=None,
  586. profile='gateway/gateway-fxo', sip_header1='P-LIBRA-CallId=#{[callId]}',
  587. sip_header2='P-LIBRA-deviceId=#{[deviceId]}', sip_header3=None, status=None):
  588. self.id = id # PK
  589. self.saas_id = saas_id
  590. self.cts = cts # 创建时间
  591. self.uts = uts # 修改时间
  592. self.name = name # 号码
  593. self.media_host = media_host # 媒体地址
  594. self.media_port = media_port # 媒体端口
  595. self.caller_prefix = caller_prefix # 主叫号码前缀
  596. self.called_prefix = called_prefix # 被叫号码前缀
  597. self.profile = profile # 媒体拨号计划文件
  598. self.sip_header1 = sip_header1 # SIP头1
  599. self.sip_header2 = sip_header2 # SIP头2
  600. self.sip_header3 = sip_header3 # SIP头3
  601. self.status = status # 状态
  602. # 机器人外呼
  603. class BotChatRequest:
  604. def __init__(self, nodeId=None, userId=None, sessionId=None, taskId=None, asrText=None, recordId=None,
  605. ext: Optional[str] = None):
  606. self.nodeId = nodeId # 节点id
  607. self.userId = userId # 用户id
  608. self.sessionId = sessionId # 会话id
  609. self.recordId = recordId # 唯一标识
  610. self.taskId = taskId # 机器人任务id
  611. self.asrText = asrText # asr识别的文本
  612. self.ext = ext
  613. def to_json_string(self):
  614. return json.dumps(self.__dict__, ensure_ascii=False)
  615. @classmethod
  616. def from_json(cls, json_string):
  617. data = json.loads(json_string)
  618. return cls(**data)
  619. # def __repr__(self):
  620. # return (f"OutboundRobotRequestParams(node_id={self.node_id}, user_id={self.user_id}, "
  621. # f"session_id={self.session_id}, record_id={self.record_id}, task_id={self.task_id}, "
  622. # f"event_type={self.event_type}, asr_text={self.asr_text}, key_input={self.key_input})")
  623. class ChatAction:
  624. def __init__(self, action_code=None, action_content=None):
  625. self.action_code = action_code # normal:正常通话;hang:挂断;transfer:转人工
  626. self.action_content = action_content # 动作内容
  627. def to_json_string(self):
  628. return json.dumps(self.__dict__, ensure_ascii=False)
  629. @classmethod
  630. def from_json(cls, json_data):
  631. return cls(
  632. action_code=json_data.get("action_code"),
  633. action_content=json_data.get("action_content")
  634. )
  635. class ChatContent:
  636. def __init__(self, content_type=None, content=None, voice_url=None, voice_content=None):
  637. self.content_type = content_type # 播放类型
  638. self.content = content # 播放内容
  639. self.voice_url = voice_url # 语音地址
  640. self.voice_content = voice_content # 语音文本
  641. def to_json_string(self):
  642. return json.dumps(self.__dict__, ensure_ascii=False)
  643. @classmethod
  644. def from_json(cls, json_data):
  645. return cls(
  646. content_type=json_data.get("content_type"),
  647. content=json_data.get("content"),
  648. voice_url=json_data.get("voice_url"),
  649. voice_content=json_data.get("voice_content")
  650. )
  651. class ChatMessage:
  652. def __init__(self, node_id=None, contents=None, wait_time=None,
  653. action=None, inputType=None):
  654. self.node_id = node_id # 节点id
  655. self.contents = contents if contents is not None else [] # 内容列表
  656. self.wait_time = wait_time # 用户静默时长
  657. self.action = action # 动作代码
  658. self.inputType = inputType # dtmf类型
  659. def to_json_string(self):
  660. return json.dumps({
  661. "node_id": self.node_id,
  662. "contents": [content.__dict__ for content in self.contents],
  663. "wait_time": self.wait_time,
  664. "action": self.action.__dict__ if self.action else None,
  665. "inputType": self.inputType
  666. }, ensure_ascii=False)
  667. @classmethod
  668. def from_json(cls, json_data):
  669. contents = [ChatContent.from_json(item) for item in json_data.get("contents", [])]
  670. action = ChatAction.from_json(json_data.get("action", {})) if json_data.get("action") else None
  671. return cls(
  672. node_id=json_data.get("node_id"),
  673. contents=contents,
  674. wait_time=json_data.get("wait_time"),
  675. action=action,
  676. inputType=json_data.get("inputType")
  677. )
  678. class ChatResponse:
  679. def __init__(self, data=None, message=None, code=None):
  680. self.data = data if data is not None else ChatMessage()
  681. self.message = message
  682. self.code = code
  683. def to_json_string(self):
  684. return json.dumps({
  685. "data": json.loads(self.data.to_json_string()),
  686. "message": self.message,
  687. "code": self.code
  688. }, ensure_ascii=False)
  689. @classmethod
  690. def from_json(cls, json_string):
  691. data = json.loads(json_string)
  692. response_data = ChatMessage.from_json(data.get("data", {}))
  693. return cls(
  694. data=response_data,
  695. message=data.get("message"),
  696. code=data.get("code")
  697. )