|
@@ -1,6 +1,9 @@
|
|
|
package com.fs.aiSipCall.service.impl;
|
|
package com.fs.aiSipCall.service.impl;
|
|
|
|
|
|
|
|
|
|
+import cn.hutool.core.util.ObjectUtil;
|
|
|
|
|
+import cn.hutool.json.JSONUtil;
|
|
|
import com.alibaba.fastjson.JSONObject;
|
|
import com.alibaba.fastjson.JSONObject;
|
|
|
|
|
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
|
|
import com.fasterxml.jackson.core.type.TypeReference;
|
|
import com.fasterxml.jackson.core.type.TypeReference;
|
|
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
|
@@ -16,18 +19,27 @@ import com.fs.aiSipCall.vo.ApiCallRecordQueryVo;
|
|
|
import com.fs.common.core.domain.AjaxResult;
|
|
import com.fs.common.core.domain.AjaxResult;
|
|
|
import com.fs.common.core.redis.RedisCache;
|
|
import com.fs.common.core.redis.RedisCache;
|
|
|
import com.fs.company.domain.CompanyAiWorkflowExec;
|
|
import com.fs.company.domain.CompanyAiWorkflowExec;
|
|
|
|
|
+import com.fs.company.domain.CompanyVoiceRoboticBusiness;
|
|
|
import com.fs.company.domain.CompanyVoiceRoboticCallLogCallphone;
|
|
import com.fs.company.domain.CompanyVoiceRoboticCallLogCallphone;
|
|
|
import com.fs.company.mapper.CompanyAiWorkflowExecMapper;
|
|
import com.fs.company.mapper.CompanyAiWorkflowExecMapper;
|
|
|
|
|
+import com.fs.company.mapper.CompanyVoiceRoboticBusinessMapper;
|
|
|
import com.fs.company.mapper.CompanyVoiceRoboticCallLogCallphoneMapper;
|
|
import com.fs.company.mapper.CompanyVoiceRoboticCallLogCallphoneMapper;
|
|
|
import com.fs.company.mapper.EasyCallMapper;
|
|
import com.fs.company.mapper.EasyCallMapper;
|
|
|
import com.fs.company.param.ExecutionContext;
|
|
import com.fs.company.param.ExecutionContext;
|
|
|
|
|
+import com.fs.company.service.CompanyWorkflowEngine;
|
|
|
|
|
+import com.fs.company.service.impl.call.node.AiCallTaskNode;
|
|
|
|
|
+import com.fs.company.vo.CidConfigVO;
|
|
|
import com.fs.company.vo.easycall.EasyCallCallPhoneVO;
|
|
import com.fs.company.vo.easycall.EasyCallCallPhoneVO;
|
|
|
|
|
+import com.fs.company.vo.easycall.EasyCallOutBoundVO;
|
|
|
|
|
+import com.fs.system.service.ISysConfigService;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
import org.apache.commons.lang3.StringUtils;
|
|
import org.apache.commons.lang3.StringUtils;
|
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
import org.springframework.scheduling.annotation.Async;
|
|
import org.springframework.scheduling.annotation.Async;
|
|
|
import org.springframework.stereotype.Service;
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
|
|
|
|
|
+import java.math.BigDecimal;
|
|
|
|
|
+import java.math.RoundingMode;
|
|
|
import java.net.URLEncoder;
|
|
import java.net.URLEncoder;
|
|
|
import java.time.LocalDateTime;
|
|
import java.time.LocalDateTime;
|
|
|
import java.util.*;
|
|
import java.util.*;
|
|
@@ -35,6 +47,8 @@ import java.util.concurrent.CompletableFuture;
|
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
|
import java.util.stream.Collectors;
|
|
import java.util.stream.Collectors;
|
|
|
|
|
|
|
|
|
|
+import static com.fs.company.service.impl.call.node.AiCallTaskNode.EASYCALL_WORKFLOW_REDIS_KEY;
|
|
|
|
|
+
|
|
|
/**
|
|
/**
|
|
|
* aiSIP手动外呼通话记录Service业务层处理
|
|
* aiSIP手动外呼通话记录Service业务层处理
|
|
|
*
|
|
*
|
|
@@ -58,6 +72,17 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
|
|
|
|
|
@Autowired
|
|
@Autowired
|
|
|
private ObjectMapper objectMapper;
|
|
private ObjectMapper objectMapper;
|
|
|
|
|
+ @Autowired
|
|
|
|
|
+ private CompanyVoiceRoboticBusinessMapper companyVoiceRoboticBusinessMapper;
|
|
|
|
|
+
|
|
|
|
|
+ @Autowired
|
|
|
|
|
+ private CompanyWorkflowEngine companyWorkflowEngine;
|
|
|
|
|
+ @Autowired
|
|
|
|
|
+ private ISysConfigService configService;
|
|
|
|
|
+
|
|
|
|
|
+ private static final BigDecimal DEFAULT_CALL_CHARGE = new BigDecimal("0.12");
|
|
|
|
|
+ private static final BigDecimal ONE_MINUTES_SECOND = new BigDecimal("60");
|
|
|
|
|
+
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
@Override
|
|
@@ -187,7 +212,7 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
@Async
|
|
@Async
|
|
|
public CompletableFuture<String> scheduledGetCallRecord() {
|
|
public CompletableFuture<String> scheduledGetCallRecord() {
|
|
|
if (!isRunning.compareAndSet(false, true)) {
|
|
if (!isRunning.compareAndSet(false, true)) {
|
|
|
- log.warn("scheduledGetCallRecord 任务正在执行中,请稍后再试");
|
|
|
|
|
|
|
+ log.error("scheduledGetCallRecord 任务正在执行中,请稍后再试");
|
|
|
return CompletableFuture.completedFuture("任务正在执行中,请稍后再试");
|
|
return CompletableFuture.completedFuture("任务正在执行中,请稍后再试");
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -276,19 +301,18 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
params.setPageNum(currentPage);
|
|
params.setPageNum(currentPage);
|
|
|
List<AiSipCallOutboundCdr> pageData = fetchSinglePageRecords(params);
|
|
List<AiSipCallOutboundCdr> pageData = fetchSinglePageRecords(params);
|
|
|
|
|
|
|
|
- // 失败时跳过该页,继续下一页
|
|
|
|
|
|
|
+ // 失败时直接结束
|
|
|
if (pageData == null) {
|
|
if (pageData == null) {
|
|
|
- log.warn("分页查询第{}页失败,已丢弃该页数据", currentPage);
|
|
|
|
|
- currentPage++;
|
|
|
|
|
- continue;
|
|
|
|
|
|
|
+ log.error("sip手动外呼同步电话 分页查询第{}页失败,已丢弃该页数据,停止查询", currentPage);
|
|
|
|
|
+ break;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
if (pageData.isEmpty()) {
|
|
if (pageData.isEmpty()) {
|
|
|
- log.info("第{}页无数据,查询结束", currentPage);
|
|
|
|
|
|
|
+ log.error("sip手动外呼同步电话 第{}页无数据,查询结束", currentPage);
|
|
|
hasMore = false;
|
|
hasMore = false;
|
|
|
} else {
|
|
} else {
|
|
|
allRecords.addAll(pageData);
|
|
allRecords.addAll(pageData);
|
|
|
- log.debug("第{}页数据:{},累计总数:{}", currentPage, pageData.size(), allRecords.size());
|
|
|
|
|
|
|
+ log.error("sip手动外呼同步电话 第{}页数据:{},累计总数:{}", currentPage, pageData.size(), allRecords.size());
|
|
|
|
|
|
|
|
// 如果返回数据少于页大小,说明已是最后一页
|
|
// 如果返回数据少于页大小,说明已是最后一页
|
|
|
if (pageData.size() < params.getPageSize()) {
|
|
if (pageData.size() < params.getPageSize()) {
|
|
@@ -299,13 +323,13 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
|
|
|
|
|
// 安全限制:最多拉取 50 页
|
|
// 安全限制:最多拉取 50 页
|
|
|
if (currentPage > 50) {
|
|
if (currentPage > 50) {
|
|
|
- log.warn("已达到最大页数限制 50 页,停止查询。已获取数据量:{}", allRecords.size());
|
|
|
|
|
|
|
+ log.error("sip手动外呼同步电话 已达到最大页数限制 50 页,停止查询。已获取数据量:{}", allRecords.size());
|
|
|
hasMore = false;
|
|
hasMore = false;
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- log.info("远程数据获取完成,总计:{} 条", allRecords.size());
|
|
|
|
|
|
|
+ log.error("sip手动外呼同步电话 远程数据获取完成,总计:{} 条", allRecords.size());
|
|
|
return allRecords.isEmpty() ? Collections.emptyList() : allRecords;
|
|
return allRecords.isEmpty() ? Collections.emptyList() : allRecords;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -472,9 +496,12 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
|
|
|
|
|
String callType = StringUtils.isBlank(req.getCallType()) ? "03" : req.getCallType();
|
|
String callType = StringUtils.isBlank(req.getCallType()) ? "03" : req.getCallType();
|
|
|
|
|
|
|
|
- EasyCallCallPhoneVO callPhoneRes = easyCallMapper.getCallPhoneInfoByUuid(req.getUuid());
|
|
|
|
|
|
|
+// EasyCallCallPhoneVO callPhoneRes = easyCallMapper.getCallPhoneInfoByUuid(req.getUuid());
|
|
|
|
|
+ EasyCallOutBoundVO callPhoneRes = easyCallMapper.getOutBoundInfoByUuid(req.getUuid());
|
|
|
String callBackUuid = UUID.randomUUID().toString();
|
|
String callBackUuid = UUID.randomUUID().toString();
|
|
|
CompanyAiWorkflowExec record = currentExecutionMapper.selectByWorkflowInstanceId(req.getWorkflowInstanceId());
|
|
CompanyAiWorkflowExec record = currentExecutionMapper.selectByWorkflowInstanceId(req.getWorkflowInstanceId());
|
|
|
|
|
+ CompanyVoiceRoboticBusiness business = companyVoiceRoboticBusinessMapper.selectOne(new LambdaQueryWrapper<CompanyVoiceRoboticBusiness>()
|
|
|
|
|
+ .eq(CompanyVoiceRoboticBusiness::getId, record.getBusinessKey()));
|
|
|
try {
|
|
try {
|
|
|
Map<String, Object> variablesMap;
|
|
Map<String, Object> variablesMap;
|
|
|
if (record.getVariables() == null || record.getVariables().isEmpty()) {
|
|
if (record.getVariables() == null || record.getVariables().isEmpty()) {
|
|
@@ -493,27 +520,50 @@ public class AiSipCallOutboundCdrServiceImpl extends ServiceImpl<AiSipCallOutbou
|
|
|
CompanyVoiceRoboticCallLogCallphone companyVoiceRoboticCallLogCallphone = new CompanyVoiceRoboticCallLogCallphone();
|
|
CompanyVoiceRoboticCallLogCallphone companyVoiceRoboticCallLogCallphone = new CompanyVoiceRoboticCallLogCallphone();
|
|
|
companyVoiceRoboticCallLogCallphone.setCallbackUuid(callBackUuid);
|
|
companyVoiceRoboticCallLogCallphone.setCallbackUuid(callBackUuid);
|
|
|
companyVoiceRoboticCallLogCallphone.setRoboticId(req.getRoboticId());
|
|
companyVoiceRoboticCallLogCallphone.setRoboticId(req.getRoboticId());
|
|
|
- companyVoiceRoboticCallLogCallphone.setCallerId(null);
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setRunTime(null);
|
|
|
|
|
|
|
+ if (ObjectUtil.isNotEmpty(business)) {
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCallerId(business.getCalleeId());
|
|
|
|
|
+ }
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setRunTime(new Date(callPhoneRes.getStartTime()));
|
|
|
companyVoiceRoboticCallLogCallphone.setRunParam(null);
|
|
companyVoiceRoboticCallLogCallphone.setRunParam(null);
|
|
|
companyVoiceRoboticCallLogCallphone.setResult(null);
|
|
companyVoiceRoboticCallLogCallphone.setResult(null);
|
|
|
companyVoiceRoboticCallLogCallphone.setStatus(req.getStatus());
|
|
companyVoiceRoboticCallLogCallphone.setStatus(req.getStatus());
|
|
|
- companyVoiceRoboticCallLogCallphone.setRecordPath(callPhoneRes.getRecordServerUrl());
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setContentList(callPhoneRes.getDialogue());
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setCallerNum(callPhoneRes.getTelephone());
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setCalleeNum(callPhoneRes.getCallerNumber());
|
|
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setRecordPath(callPhoneRes.getRecordFilename());
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setContentList(callPhoneRes.getChatContent());
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCallerNum(callPhoneRes.getCallee());
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCalleeNum(callPhoneRes.getCaller());
|
|
|
companyVoiceRoboticCallLogCallphone.setUuid(req.getUuid());
|
|
companyVoiceRoboticCallLogCallphone.setUuid(req.getUuid());
|
|
|
- companyVoiceRoboticCallLogCallphone.setCallCreateTime(callPhoneRes.getCalloutTime());
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setCallAnswerTime(callPhoneRes.getConnectedTime());
|
|
|
|
|
- companyVoiceRoboticCallLogCallphone.setIntention(callPhoneRes.getIntent());
|
|
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCallCreateTime(callPhoneRes.getStartTime());
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCallAnswerTime(callPhoneRes.getAnsweredTime());
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setIntention(req.getIntent());
|
|
|
companyVoiceRoboticCallLogCallphone.setCompanyId(req.getCompanyId());
|
|
companyVoiceRoboticCallLogCallphone.setCompanyId(req.getCompanyId());
|
|
|
companyVoiceRoboticCallLogCallphone.setCompanyUserId(req.getCompanyUserId());
|
|
companyVoiceRoboticCallLogCallphone.setCompanyUserId(req.getCompanyUserId());
|
|
|
companyVoiceRoboticCallLogCallphone.setCallTime(Long.valueOf(callPhoneRes.getTimeLen()));
|
|
companyVoiceRoboticCallLogCallphone.setCallTime(Long.valueOf(callPhoneRes.getTimeLen()));
|
|
|
- companyVoiceRoboticCallLogCallphone.setCost(callPhoneRes.getTotalCost());
|
|
|
|
|
companyVoiceRoboticCallLogCallphone.setCallType(Integer.valueOf(callType));
|
|
companyVoiceRoboticCallLogCallphone.setCallType(Integer.valueOf(callType));
|
|
|
|
|
|
|
|
|
|
+ String json = configService.selectConfigByKey("cid.config");
|
|
|
|
|
+ CidConfigVO cidConfigVO = JSONUtil.toBean(json, CidConfigVO.class);
|
|
|
|
|
+ BigDecimal callCharge = cidConfigVO.getCallCharge();
|
|
|
|
|
+ //
|
|
|
|
|
+ if (null == callCharge) {
|
|
|
|
|
+ callCharge = DEFAULT_CALL_CHARGE;
|
|
|
|
|
+ }
|
|
|
|
|
+ //向上取整分钟数
|
|
|
|
|
+ BigDecimal divide = new BigDecimal(companyVoiceRoboticCallLogCallphone.getCallTime()).divide(ONE_MINUTES_SECOND, 0, RoundingMode.CEILING);
|
|
|
|
|
+ BigDecimal multiply = divide.multiply(callCharge);
|
|
|
|
|
+ companyVoiceRoboticCallLogCallphone.setCost(multiply);
|
|
|
|
|
+
|
|
|
|
|
+
|
|
|
|
|
+ int i = companyVoiceRoboticCallLogCallphoneMapper.insertCompanyVoiceRoboticCallLogCallphone(companyVoiceRoboticCallLogCallphone);
|
|
|
|
|
+
|
|
|
|
|
+ Map<String, Object> param = new HashMap<>();
|
|
|
|
|
+ param.put("callBackUuid", callBackUuid);
|
|
|
|
|
+ param.put("callSource", "callBack");
|
|
|
|
|
+ companyWorkflowEngine.resumeFromBlockingNode(req.getWorkflowInstanceId(),record.getCurrentNodeKey(),param);
|
|
|
|
|
+
|
|
|
|
|
+ redisCache.deleteObject(EASYCALL_WORKFLOW_REDIS_KEY + callBackUuid);
|
|
|
|
|
+
|
|
|
|
|
|
|
|
- return companyVoiceRoboticCallLogCallphoneMapper.insertCompanyVoiceRoboticCallLogCallphone(companyVoiceRoboticCallLogCallphone);
|
|
|
|
|
|
|
+ return i;
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|