diff --git a/contact-center/app/src/main/java/com/chatopera/cc/acd/ACDPolicyService.java b/contact-center/app/src/main/java/com/chatopera/cc/acd/ACDPolicyService.java index 39f163b2b43bda37f0b0bda7e0180f3769793f11..7bdde377f3e197432e33e4c2e93b799419033512 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/acd/ACDPolicyService.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/acd/ACDPolicyService.java @@ -22,9 +22,11 @@ import com.chatopera.cc.basic.MainUtils; import com.chatopera.cc.cache.Cache; import com.chatopera.cc.model.AgentStatus; import com.chatopera.cc.model.AgentUser; +import com.chatopera.cc.model.SNSAccount; import com.chatopera.cc.model.SessionConfig; import com.chatopera.cc.persistence.repository.AgentUserRepository; import com.chatopera.cc.persistence.repository.OnlineUserRepository; +import com.chatopera.cc.persistence.repository.SNSAccountRepository; import com.chatopera.cc.persistence.repository.SessionConfigRepository; import com.chatopera.cc.util.HashMapUtils; import com.chatopera.cc.util.WebIMReport; @@ -58,6 +60,9 @@ public class ACDPolicyService { @Autowired private AgentUserRepository agentUserRes; + @Autowired + private SNSAccountRepository snsAccountRes; + /** * 载入坐席 ACD策略配置 * @@ -229,25 +234,28 @@ public class ACDPolicyService { * * TODO 指定技能组无用户,停止分配 */ + + SNSAccount snsAccount = snsAccountRes.findBySnsidAndOrgi(agentUser.getAppid(), orgi); + // 对于该租户的所有客服 - // for (final Map.Entry entry : map.entrySet()) { - // if ((!entry.getValue().isBusy()) && (entry.getValue().getUsers() < sessionConfig.getMaxuser())) { - // agentStatuses.add(entry.getValue()); - // logger.info( - // "[filterOutAvailableAgentStatus] find ready agent {}, agentname {}, status {}, service {}/{}, skills {}", - // entry.getValue().getAgentno(), entry.getValue().getUsername(), entry.getValue().getStatus(), - // entry.getValue().getUsers(), - // entry.getValue().getMaxusers(), - // HashMapUtils.concatKeys(entry.getValue().getSkills(), "|")); - // } else { - // logger.info( - // "[filterOutAvailableAgentStatus] skip ready agent {}, name {}, status {}, service {}/{}, skills {}", - // entry.getValue().getAgentno(), entry.getValue().getUsername(), entry.getValue().getStatus(), - // entry.getValue().getUsers(), - // entry.getValue().getMaxusers(), - // HashMapUtils.concatKeys(entry.getValue().getSkills(), "|")); - // } - // } + for (final Map.Entry entry : map.entrySet()) { + if ((!entry.getValue().isBusy()) && (entry.getValue().getUsers() < sessionConfig.getMaxuser()) && entry.getValue().getSkills().containsKey(snsAccount.getOrgan())) { + agentStatuses.add(entry.getValue()); + logger.info( + "[filterOutAvailableAgentStatus] find ready agent {}, agentname {}, status {}, service {}/{}, skills {}", + entry.getValue().getAgentno(), entry.getValue().getUsername(), entry.getValue().getStatus(), + entry.getValue().getUsers(), + entry.getValue().getMaxusers(), + HashMapUtils.concatKeys(entry.getValue().getSkills(), "|")); + } else { + logger.info( + "[filterOutAvailableAgentStatus] skip ready agent {}, name {}, status {}, service {}/{}, skills {}", + entry.getValue().getAgentno(), entry.getValue().getUsername(), entry.getValue().getStatus(), + entry.getValue().getUsers(), + entry.getValue().getMaxusers(), + HashMapUtils.concatKeys(entry.getValue().getSkills(), "|")); + } + } } logger.info("[filterOutAvailableAgentStatus] agent status list size: {}", agentStatuses.size()); diff --git a/contact-center/app/src/main/java/com/chatopera/cc/acd/middleware/visitor/ACDVisSessionCfgMw.java b/contact-center/app/src/main/java/com/chatopera/cc/acd/middleware/visitor/ACDVisSessionCfgMw.java index 355d70d709b00b5f2c63b22aa397eac0429ac82d..3ad4a7e5c873d031a90380a6df01852d2adeb2c6 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/acd/middleware/visitor/ACDVisSessionCfgMw.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/acd/middleware/visitor/ACDVisSessionCfgMw.java @@ -19,6 +19,7 @@ package com.chatopera.cc.acd.middleware.visitor; import com.chatopera.cc.acd.ACDPolicyService; import com.chatopera.cc.acd.ACDWorkMonitor; import com.chatopera.cc.acd.basic.ACDComposeContext; +import com.chatopera.cc.basic.MainContext; import com.chatopera.cc.basic.MainUtils; import com.chatopera.cc.model.AgentReport; import com.chatopera.cc.model.SessionConfig; @@ -66,8 +67,12 @@ public class ACDVisSessionCfgMw implements Middleware { ctx.setMessage(sessionConfig.getNotinwhmsg()); } else if (report.getAgents() == 0) { // 没有就绪的坐席 - logger.info("[apply] find no agents, redirect to leave a message."); - ctx.setNoagent(true); + if (ctx.getChannel().equals(MainContext.ChannelType.MESSENGER.toString())) { + next.apply(); + } else { + logger.info("[apply] find no agents, redirect to leave a message."); + ctx.setNoagent(true); + } } else { logger.info("[apply] find agents size {}, allocate agent in next.", report.getAgents()); // 具备工作中的就绪坐席,进入筛选坐席 diff --git a/contact-center/app/src/main/java/com/chatopera/cc/controller/Handler.java b/contact-center/app/src/main/java/com/chatopera/cc/controller/Handler.java index 9ff235e1ffb8afdc23c68012505e3e3d64a26a2f..494a9788e9355c092082916451ba1faf7c16f865 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/controller/Handler.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/controller/Handler.java @@ -48,6 +48,7 @@ import javax.servlet.http.HttpServletRequest; import java.io.IOException; import java.text.ParseException; import java.util.ArrayList; +import java.util.HashMap; import java.util.Map; import static org.elasticsearch.index.query.QueryBuilders.termQuery; @@ -107,6 +108,7 @@ public class Handler { /** * 获得登录账号的当前导航的组织机构 + * * @param request * @return */ @@ -122,7 +124,19 @@ public class Handler { Organ organ = (Organ) request.getSession(true).getAttribute(Constants.ORGAN_SESSION_NAME); if (organ == null) { if (organs.size() > 0) { - organ = organs.get(0); + ArrayList organTree = new ArrayList(); + organs.stream().forEach(o -> { + if (organTree.stream().filter(p -> StringUtils.equals(o.getParent(), p)).findFirst().isPresent()) { + int index = organTree.indexOf(o.getParent()); + organTree.add(index + 1, o.getId()); + } else { + organTree.add(0, o.getId()); + } + }); + + organ = organs.stream().filter(o -> + StringUtils.equals(o.getId(), organTree.get(0)) + ).findFirst().orElse(organs.get(0)); request.getSession(true).setAttribute(Constants.ORGAN_SESSION_NAME, organ); } } diff --git a/contact-center/app/src/main/java/com/chatopera/cc/controller/apps/AgentAuditController.java b/contact-center/app/src/main/java/com/chatopera/cc/controller/apps/AgentAuditController.java index c6a3d3f66a4884bff711f565deda82ef5603c35e..773a83dc60e5831e028a3421c3f8a49190873982 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/controller/apps/AgentAuditController.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/controller/apps/AgentAuditController.java @@ -166,7 +166,7 @@ public class AgentAuditController extends Handler { if (StringUtils.isBlank(skill) && StringUtils.isBlank(agentno)) { if (organs.size() > 0) { - agentUsers = agentUserRes.findByOrgiAndStatusAndSkillInAndAgentnoIsNot( + agentUsers = agentUserRes.findByOrgiAndStatusAndSkillInAndAgentnoIsNotAndChatbotopsIsFalse( orgi, MainContext.AgentUserStatusEnum.INSERVICE.toString(), organs.keySet(), logined.getId(), defaultSort); } } else if (StringUtils.isNotBlank(skill) && StringUtils.isNotBlank(agentno)) { diff --git a/contact-center/app/src/main/java/com/chatopera/cc/model/FbOTN.java b/contact-center/app/src/main/java/com/chatopera/cc/model/FbOTN.java index c98cc13fc2c099a8f07292aef55e121e3807aca0..0b3251e2edb330a44dc1936a67179d8886681b9a 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/model/FbOTN.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/model/FbOTN.java @@ -40,6 +40,7 @@ public class FbOTN implements Serializable { private Date createtime; private Date updatetime; private Date sendtime; + private Integer melinkNum; private Integer subNum; private FbMessenger fbMessenger; @@ -152,6 +153,14 @@ public class FbOTN implements Serializable { this.subNum = subNum; } + public Integer getMelinkNum() { + return melinkNum; + } + + public void setMelinkNum(Integer melinkNum) { + this.melinkNum = melinkNum; + } + @ManyToOne(fetch = FetchType.EAGER) @JoinColumn(name = "pageId", referencedColumnName = "pageId", insertable = false, updatable = false) @NotFound(action = NotFoundAction.IGNORE) diff --git a/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/AgentUserRepository.java b/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/AgentUserRepository.java index ec0d400e4de75df0ca602f881ca37996d5637438..1122f64b61b5d69a739f6a967a2e5cad622dc695 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/AgentUserRepository.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/AgentUserRepository.java @@ -62,7 +62,7 @@ public interface AgentUserRepository extends JpaRepository { List findByOrgiAndStatusAndSkillAndAgentnoIsNot(final String orgi, final String status, final String skill, final String agentno, final Sort sort); - List findByOrgiAndStatusAndSkillInAndAgentnoIsNot(final String orgi, final String status, final Collection skill, final String agentno, final Sort sort); + List findByOrgiAndStatusAndSkillInAndAgentnoIsNotAndChatbotopsIsFalse(final String orgi, final String status, final Collection skill, final String agentno, final Sort sort); List findByOrgiAndStatusAndAgentno(final String orgi, final String status, final String agentno, final Sort defaultSort); diff --git a/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/FbOTNRepository.java b/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/FbOTNRepository.java index b24309331d16bef18a34cafd450c6f117f3bbb6d..8a78e695bd30229dd03e882df5dd08a27e011737 100644 --- a/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/FbOTNRepository.java +++ b/contact-center/app/src/main/java/com/chatopera/cc/persistence/repository/FbOTNRepository.java @@ -41,4 +41,9 @@ public interface FbOTNRepository extends JpaRepository { @Modifying @Query(nativeQuery = true, value = "update cs_fb_otn set sub_num = sub_num +1 where id = ?1") void incOneSubNumById(final String id); + + @Transactional + @Modifying + @Query(nativeQuery = true, value = "update cs_fb_otn set melink_num = melink_num +1 where id = ?1") + void incOneMelinkNumById(final String id); } diff --git a/contact-center/app/src/main/resources/static/js/otnContent.js b/contact-center/app/src/main/resources/static/js/otnContent.js index e0d999f3a597299fa5fea87a36d5004f1bbb108c..00b8a42dcb6338b315e51d76512fcf79cdd1b00d 100644 --- a/contact-center/app/src/main/resources/static/js/otnContent.js +++ b/contact-center/app/src/main/resources/static/js/otnContent.js @@ -48,12 +48,14 @@ uploadText.hide(); json.url = data.url; jsonInput.val(JSON.stringify(json)); + jsonInput.trigger("change"); img.css('height', '130px'); }); }); textInput.bind("input propertychange", function (event) { json.content = textInput.val(); jsonInput.val(JSON.stringify(json)); + jsonInput.trigger("change"); }); select.change(function () { if (select.val() == 'image') { @@ -66,8 +68,10 @@ json.type = 'text' } jsonInput.val(JSON.stringify(json)); + jsonInput.trigger("change"); }) jsonInput.val(JSON.stringify(json)); + jsonInput.trigger("change"); $this.append(jsonInput) $this.append(select) $this.append(box) diff --git a/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser.pug b/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser.pug index bad4ce5c496b9033f96b27ed32b097d9903fe0b0..f4a2ca1e87d32b95127de8aa9073c6d3dc70b4f1 100644 --- a/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser.pug +++ b/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser.pug @@ -682,10 +682,10 @@ script. }); $('#quickReplyAgentBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); $('#quickReplyBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); diff --git a/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser_messenger.pug b/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser_messenger.pug index 9388ca1359cf3e8e42c5f9899f70d62fcd95bc51..8e7b17198810e28ba4163b39bb28524e5cbe14c1 100644 --- a/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser_messenger.pug +++ b/contact-center/app/src/main/resources/templates/apps/agent/mainagentuser_messenger.pug @@ -685,10 +685,10 @@ script. }); $('#quickReplyAgentBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); $('#quickReplyBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); diff --git a/contact-center/app/src/main/resources/templates/apps/cca/mainagentuser.pug b/contact-center/app/src/main/resources/templates/apps/cca/mainagentuser.pug index f29a6907bc8f3a396799b013a1c3157488ed79db..8f821900aef40f42bf6b0cb56f4be1aa688fccdb 100644 --- a/contact-center/app/src/main/resources/templates/apps/cca/mainagentuser.pug +++ b/contact-center/app/src/main/resources/templates/apps/cca/mainagentuser.pug @@ -681,10 +681,10 @@ script. }); $('#ccaQuickReplyAgentBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); $('#ccaQuickReplyBox').on('click', '.chooseAnswer', function () { - editor.html($(this).data('title')) + editor.html($(this).data('title') + "") }); diff --git a/contact-center/app/src/main/resources/templates/apps/im/chatbot/index.pug b/contact-center/app/src/main/resources/templates/apps/im/chatbot/index.pug index 6804dd9293882d8d70c2ea1395b876ab818cb78d..0c283effec67f6fc05fd8fd21a14d96bbc7156d8 100644 --- a/contact-center/app/src/main/resources/templates/apps/im/chatbot/index.pug +++ b/contact-center/app/src/main/resources/templates/apps/im/chatbot/index.pug @@ -311,7 +311,7 @@ html .ukefu-func-tab ul if models.contains("chatbot") && inviteData.ai && aiid - if !exchange || exchange == "true" + if exchange || exchange == "true" li a(href="/im/index.html?appid=" + appid + "&orgi=" + orgi + (aiid ? "&aiid=" + aiid : "") + "&ai=false" + (client ? "&client=" + client : '') + (type ? "&type=text" : "") + (skill ? "&skill=" + skill : '') + (agent ? "&agent=" + agent : '') + (title ? "&title=" + title : '') + (url ? "&url=" + url : '') + (traceid ? "&traceid=" + traceid : '') + "&userid=" + userid + "&sessionid=" + sessionid + "&t=" + pugHelper.currentTimeMillis()) 人工坐席 li.cur diff --git a/contact-center/app/src/main/resources/templates/apps/im/chatbot/mobile.pug b/contact-center/app/src/main/resources/templates/apps/im/chatbot/mobile.pug index 3c0830d4417cdac5c727cdfb12494aa87b6e1baa..0908eff052e37361479a282a110ce75c9d73b511 100644 --- a/contact-center/app/src/main/resources/templates/apps/im/chatbot/mobile.pug +++ b/contact-center/app/src/main/resources/templates/apps/im/chatbot/mobile.pug @@ -213,7 +213,7 @@ html .ukefu-func-tab ul if models.contains("chatbot") && inviteData.ai && inviteData.ai == true && aiid - if !exchange || exchange == "true" + if exchange || exchange == "true" li a(href="/im/index.html?appid=" + appid + "&orgi=" + orgi + (aiid ? '&aiid=' + aiid : '') + "&ai=false" + (client ? '&client=' + client : '') + (type ? '&type=' + type : '') + (skill ? '&skill=' + skill : '') + (agent ? '&agent=' + agent : '') + "&userid=" + userid + "&sessionid=" + sessionid + "&t=" + pugHelper.currentTimeMillis()) 人工客服 li.cur diff --git a/contact-center/app/src/main/resources/templates/apps/im/point.pug b/contact-center/app/src/main/resources/templates/apps/im/point.pug index 2f94cd490d78bb589772cbeb6da309a49fddfcc3..b8ce2e28e0d5633d2cd831f01b69ac1199305982 100644 --- a/contact-center/app/src/main/resources/templates/apps/im/point.pug +++ b/contact-center/app/src/main/resources/templates/apps/im/point.pug @@ -54,7 +54,6 @@ else if inviteData.consult_invite_color == "6" #ukefu-point-theme.ukefu-theme-color.theme1(class="ukefu-point-theme" + (inviteData.consult_vsitorbtn_model ? inviteData.consult_vsitorbtn_model : '1'), style=style + theme) #ukefu-im-point-text.ukefu-im-point-text(style='cursor: pointer;' + text + ';line-height: 23px;font-size: 15px;text-align: center;margin: 0 auto;') i(style="width:24px;height:24px;display: inline-block;font: normal normal normal 14px/1 FontAwesome;font-size: inherit;text-rendering: auto;-webkit-font-smoothing: antialiased;background-image:url();") - br | #{inviteData.consult_vsitorbtn_content ? inviteData.consult_vsitorbtn_content : '在线客服'} if inviteData.skill == true && inviteData.consult_skill_fixed == false #ichatContent.ichat-content.ichat-float-left(style='width:152px;display: block;font-size:14px;' + theme + ';box-shadow: 0 0 10px #DDDDDD;border: 1px solid #DDDDDD;overflow: hidden;text-align: left;') @@ -72,12 +71,17 @@ else if inviteData.consult_invite_color == "6" else img(src=baseUrl + '/images/title.jpg', style='width:140px;margin:0 5px;vertical-align: middle;') div(style="padding:10px 5px;text-align:center;color:#555;")= inviteData.skill == true && inviteData.consult_skill_fixed == false && inviteData.consult_skill_msg ? inviteData.consult_skill_msg : '' + if skillGroups for skill in skillGroups - var skillNum = 0 for agent in agentList - - skillNum = skillNum + 1 - | #{skillNum} + if agent.skills[skill.id] + - skillNum = skillNum + 1 + + div(style='padding:5px;color:#555;border:1px solid #dddddd;background-color:#efefef;') + a(href=baseUrl + '/im/text/' + appid + '.html?skill=' + skill.id + '&orgi=' + orgi, onclick='return openAgentChatDialog(this.href)', style='color: #777; text-decoration: none;') #{skill.name}(#{skillNum}) + if inviteData.consult_skill_agent && inviteData.consult_skill_agent == true ul(class="ichat-list-ul", style="border: 0 none;list-style: none outside none;margin: 0;padding: 0;width: 150px;") if agentList diff --git a/contact-center/config/sql/cosinee-MySQL-slim.sql b/contact-center/config/sql/cosinee-MySQL-slim.sql index bd77b9c0827f375d4a1b2ecf1775661970abd2ab..18c00c627618b17e12e628344497417cb4bdb772 100644 --- a/contact-center/config/sql/cosinee-MySQL-slim.sql +++ b/contact-center/config/sql/cosinee-MySQL-slim.sql @@ -794,6 +794,7 @@ CREATE TABLE `cs_fb_otn` ( `updatetime` DATETIME NOT NULL, `sendtime` DATETIME NULL DEFAULT NULL, `sub_num` INT(11) NOT NULL, + `melink_num` INT(11) NOT NULL, PRIMARY KEY (`id`) USING BTREE ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='FB OTN'; diff --git a/contact-center/config/sql/upgrade/008.alert_cs_fb_otn_melink.sql b/contact-center/config/sql/upgrade/008.alert_cs_fb_otn_melink.sql new file mode 100644 index 0000000000000000000000000000000000000000..e571023d62329d508d0173a2729bc54723bfeb08 --- /dev/null +++ b/contact-center/config/sql/upgrade/008.alert_cs_fb_otn_melink.sql @@ -0,0 +1,32 @@ +USE `cosinee`; +-- ----------------- +-- prepare variables +-- ----------------- + +SET @dbname = DATABASE ( ); +SET @tablename = "cs_fb_otn"; +SET @columnname = "melink_num"; + +SET @preparedStatement = ( + SELECT + IF + ( + ( + SELECT + COUNT( * ) + FROM + INFORMATION_SCHEMA.COLUMNS + WHERE + ( table_name = @tablename ) + AND ( table_schema = @dbname ) + AND ( column_name = @columnname ) + ) > 0, + "SELECT 1", + CONCAT( "ALTER TABLE ", @tablename, " ADD ", @columnname, " INT(11) DEFAULT 0 COMMENT 'Melink 点击数';" ) + ) +); +PREPARE alterIfNotExists +FROM + @preparedStatement; +EXECUTE alterIfNotExists; +DEALLOCATE PREPARE alterIfNotExists; \ No newline at end of file diff --git a/contact-center/root/pom.xml b/contact-center/root/pom.xml index e910d27a58b53ea816f9e5be055a9c9541b6b638..fc226ed830576042579f8142a3790d9172002fed 100644 --- a/contact-center/root/pom.xml +++ b/contact-center/root/pom.xml @@ -55,12 +55,10 @@ - - de.neuland-bfi spring-pug4j - 2.0.0-alpha-4-SNAPSHOT + 2.0.0-alpha-2 diff --git a/public/plugins/chatbot/classes/ApiChatbotController.java b/public/plugins/chatbot/classes/ApiChatbotController.java index 5a222e0e1fedaf1ca02d2a15c9cb2ea05b33e4ab..6f8212d4c43de5259e2094b44073dd20d8b06bda 100644 --- a/public/plugins/chatbot/classes/ApiChatbotController.java +++ b/public/plugins/chatbot/classes/ApiChatbotController.java @@ -497,6 +497,7 @@ public class ApiChatbotController extends Handler { } else if (c.getChannel().equals(Constants.CHANNEL_TYPE_MESSENGER)) { FbMessenger fbMessenger = fbMessengerRepository.findOneByPageId(c.getSnsAccountIdentifier()); fbMessenger.setAiid(null); + fbMessenger.setAi(false); fbMessengerRepository.save(fbMessenger); }