All Downloads are FREE. Search and download functionalities are using the official Maven repository.

me.chanjar.weixin.cp.tp.message.WxCpTpMessageRouter Maven / Gradle / Ivy

There is a newer version: 4.6.7.B
Show newest version
package me.chanjar.weixin.cp.tp.message;

import com.google.common.util.concurrent.ThreadFactoryBuilder;
import lombok.extern.slf4j.Slf4j;
import me.chanjar.weixin.common.api.WxErrorExceptionHandler;
import me.chanjar.weixin.common.api.WxMessageDuplicateChecker;
import me.chanjar.weixin.common.api.WxMessageInMemoryDuplicateChecker;
import me.chanjar.weixin.common.api.WxMessageInMemoryDuplicateCheckerSingleton;
import me.chanjar.weixin.common.session.InternalSession;
import me.chanjar.weixin.common.session.InternalSessionManager;
import me.chanjar.weixin.common.session.WxSessionManager;
import me.chanjar.weixin.common.util.LogExceptionHandler;
import me.chanjar.weixin.cp.bean.message.WxCpTpXmlMessage;
import me.chanjar.weixin.cp.bean.message.WxCpXmlOutMessage;
import me.chanjar.weixin.cp.tp.service.WxCpTpService;
import org.apache.commons.lang3.StringUtils;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.*;

/**
 * 
 * 微信消息路由器,通过代码化的配置,把来自微信的消息交给handler处理
 * 和WxCpMessageRouter的rule相比,多了infoType和changeType维度的匹配
 *
 * 说明:
 * 1. 配置路由规则时要按照从细到粗的原则,否则可能消息可能会被提前处理
 * 2. 默认情况下消息只会被处理一次,除非使用 {@link WxCpTpMessageRouterRule#next()}
 * 3. 规则的结束必须用{@link WxCpTpMessageRouterRule#end()}或者{@link WxCpTpMessageRouterRule#next()},否则不会生效
 *
 * 使用方法:
 * WxCpTpMessageRouter router = new WxCpTpMessageRouter();
 * router
 *   .rule()
 *       .msgType("MSG_TYPE").event("EVENT").eventKey("EVENT_KEY").content("CONTENT")
 *       .interceptor(interceptor, ...).handler(handler, ...)
 *   .end()
 *   .rule()
 *       .infoType("INFO_TYPE").changeType("CHANGE_TYPE")
 *       // 另外一个匹配规则
 *   .end()
 * ;
 *
 * // 将WxXmlMessage交给消息路由器
 * router.route(message);
 *
 * 
* * @author Daniel Qian */ @Slf4j public class WxCpTpMessageRouter { private static final int DEFAULT_THREAD_POOL_SIZE = 100; private final List rules = new ArrayList<>(); private final WxCpTpService wxCpTpService; private ExecutorService executorService; private WxMessageDuplicateChecker messageDuplicateChecker; private WxSessionManager sessionManager; private WxErrorExceptionHandler exceptionHandler; /** * 构造方法. * * @param wxCpTpService the wx cp tp service */ public WxCpTpMessageRouter(WxCpTpService wxCpTpService) { this.wxCpTpService = wxCpTpService; ThreadFactory namedThreadFactory = new ThreadFactoryBuilder().setNameFormat("WxCpTpMessageRouter-pool-%d").build(); this.executorService = new ThreadPoolExecutor(DEFAULT_THREAD_POOL_SIZE, DEFAULT_THREAD_POOL_SIZE, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(), namedThreadFactory); this.messageDuplicateChecker = WxMessageInMemoryDuplicateCheckerSingleton.getInstance(); this.sessionManager = wxCpTpService.getSessionManager(); this.exceptionHandler = new LogExceptionHandler(); } /** * 使用自定义的 {@link ExecutorService}. * * @param wxCpTpService the wx cp tp service * @param executorService the executor service */ public WxCpTpMessageRouter(WxCpTpService wxCpTpService, ExecutorService executorService) { this.wxCpTpService = wxCpTpService; this.executorService = executorService; this.messageDuplicateChecker = WxMessageInMemoryDuplicateCheckerSingleton.getInstance(); this.sessionManager = wxCpTpService.getSessionManager(); this.exceptionHandler = new LogExceptionHandler(); } /** * 系统退出前,应该调用该方法 */ public void shutDownExecutorService() { this.executorService.shutdown(); } /** * 系统退出前,应该调用该方法,增加了超时时间检测 * * @param second the second */ public void shutDownExecutorService(Integer second) { this.executorService.shutdown(); try { if (!this.executorService.awaitTermination(second, TimeUnit.SECONDS)) { this.executorService.shutdownNow(); if (!this.executorService.awaitTermination(second, TimeUnit.SECONDS)) log.error("线程池未关闭!"); } } catch (InterruptedException ie) { this.executorService.shutdownNow(); Thread.currentThread().interrupt(); } } /** *
   * 设置自定义的 {@link ExecutorService}
   * 如果不调用该方法,默认使用 Executors.newFixedThreadPool(100)
   * 
* * @param executorService the executor service */ public void setExecutorService(ExecutorService executorService) { this.executorService = executorService; } /** *
   * 设置自定义的 {@link WxMessageDuplicateChecker}
   * 如果不调用该方法,默认使用 {@link WxMessageInMemoryDuplicateChecker}
   * 
* * @param messageDuplicateChecker the message duplicate checker */ public void setMessageDuplicateChecker(WxMessageDuplicateChecker messageDuplicateChecker) { this.messageDuplicateChecker = messageDuplicateChecker; } /** *
   * 设置自定义的{@link WxSessionManager}
   * 如果不调用该方法,默认使用 {@link me.chanjar.weixin.common.session.StandardSessionManager}
   * 
* * @param sessionManager the session manager */ public void setSessionManager(WxSessionManager sessionManager) { this.sessionManager = sessionManager; } /** *
   * 设置自定义的{@link WxErrorExceptionHandler}
   * 如果不调用该方法,默认使用 {@link LogExceptionHandler}
   * 
* * @param exceptionHandler the exception handler */ public void setExceptionHandler(WxErrorExceptionHandler exceptionHandler) { this.exceptionHandler = exceptionHandler; } /** * Gets rules. * * @return the rules */ List getRules() { return this.rules; } /** * 开始一个新的Route规则. * * @return the wx cp tp message router rule */ public WxCpTpMessageRouterRule rule() { return new WxCpTpMessageRouterRule(this); } /** * 处理微信消息. * * @param suiteId the suiteId * @param wxMessage the wx message * @param context the context * @return the wx cp xml out message */ public WxCpXmlOutMessage route(final String suiteId, final WxCpTpXmlMessage wxMessage, final Map context) { if (isMsgDuplicated(suiteId, wxMessage)) { // 如果是重复消息,那么就不做处理 return null; } final List matchRules = new ArrayList<>(); // 收集匹配的规则 for (final WxCpTpMessageRouterRule rule : this.rules) { if (rule.test(wxMessage)) { matchRules.add(rule); if (!rule.isReEnter()) { break; } } } if (matchRules.size() == 0) { return null; } WxCpXmlOutMessage res = null; final List futures = new ArrayList<>(); for (final WxCpTpMessageRouterRule rule : matchRules) { // 返回最后一个非异步的rule的执行结果 if (rule.isAsync()) { futures.add( this.executorService.submit(() -> { rule.service(wxMessage, context, WxCpTpMessageRouter.this.wxCpTpService, WxCpTpMessageRouter.this.sessionManager, WxCpTpMessageRouter.this.exceptionHandler); }) ); } else { res = rule.service(wxMessage, context, this.wxCpTpService, this.sessionManager, this.exceptionHandler); // 在同步操作结束,session访问结束 log.debug("End session access: async=false, sessionId={}", wxMessage.getSuiteId()); sessionEndAccess(wxMessage); } } if (futures.size() > 0) { this.executorService.submit(() -> { for (Future future : futures) { try { future.get(); log.debug("End session access: async=true, sessionId={}", wxMessage.getSuiteId()); // 异步操作结束,session访问结束 sessionEndAccess(wxMessage); } catch (InterruptedException e) { log.error("Error happened when wait task finish", e); Thread.currentThread().interrupt(); } catch (ExecutionException e) { log.error("Error happened when wait task finish", e); } } }); } return res; } /** * 处理微信消息. * * @param wxMessage the wx message * @param context the context * @return the wx cp xml out message */ public WxCpXmlOutMessage route(final WxCpTpXmlMessage wxMessage, final Map context) { return this.route(null, wxMessage, context); } /** * 处理微信消息. * * @param wxMessage the wx message * @return the wx cp xml out message */ public WxCpXmlOutMessage route(final WxCpTpXmlMessage wxMessage) { return this.route(wxMessage, new HashMap<>(2)); } private boolean isMsgDuplicated(final String suiteId, WxCpTpXmlMessage wxMessage) { StringBuilder messageId = new StringBuilder(); messageId.append(wxMessage.getToUserName()); if (wxMessage.getInfoType() != null) { messageId.append(wxMessage.getInfoType()) .append("-").append(StringUtils.trimToEmpty(wxMessage.getSuiteId())) .append("-").append(wxMessage.getTimeStamp()) .append("-").append(StringUtils.trimToEmpty(wxMessage.getAuthCorpId())) .append("-").append(StringUtils.trimToEmpty(wxMessage.getUserID())) .append("-").append(StringUtils.trimToEmpty(wxMessage.getChangeType())) .append("-").append(StringUtils.trimToEmpty(wxMessage.getServiceCorpId())) .append("-").append(StringUtils.trimToEmpty(wxMessage.getExternalUserID())); } else { if (StringUtils.isNotBlank(suiteId)) { messageId.append(suiteId); } } if (wxMessage.getMsgType() != null) { if (wxMessage.getMsgId() != null) { messageId.append(wxMessage.getMsgId()) .append("-").append(wxMessage.getCreateTime()) .append("-").append(wxMessage.getFromUserName()); } else { messageId.append(wxMessage.getMsgType()) .append("-").append(wxMessage.getCreateTime()) .append("-").append(wxMessage.getFromUserName()) .append("-").append(StringUtils.trimToEmpty(wxMessage.getEvent())) .append("-").append(StringUtils.trimToEmpty(wxMessage.getEventKey())); } } return this.messageDuplicateChecker.isDuplicate(messageId.toString()); } /** * 对session的访问结束. */ private void sessionEndAccess(WxCpTpXmlMessage wxMessage) { InternalSession session = ((InternalSessionManager) this.sessionManager).findSession(wxMessage.getSuiteId()); if (session != null) { session.endAccess(); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy