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

me.chanjar.weixin.channel.message.WxChannelMessageRouter Maven / Gradle / Ivy

The newest version!
package me.chanjar.weixin.channel.message;

import com.google.common.util.concurrent.ThreadFactoryBuilder;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import me.chanjar.weixin.channel.api.WxChannelService;
import me.chanjar.weixin.common.api.WxErrorExceptionHandler;
import me.chanjar.weixin.common.api.WxMessageDuplicateChecker;
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.StandardSessionManager;
import me.chanjar.weixin.common.session.WxSessionManager;
import me.chanjar.weixin.common.util.LogExceptionHandler;
import org.apache.commons.lang3.StringUtils;

/**
 * 消息路由器
 *
 * @author Zeyes
 */
@Data
@Slf4j
public class WxChannelMessageRouter {
  /** 规则列表 */
  private final List> rules = new ArrayList<>();
  /** 线程池 */
  private ExecutorService executorService;
  /** 异常处理器 */
  private WxErrorExceptionHandler exceptionHandler;
  /** 消息重复检查器 */
  private WxMessageDuplicateChecker messageDuplicateChecker;

  public WxChannelMessageRouter() {
    ThreadFactory namedThreadFactory = new ThreadFactoryBuilder().setNameFormat("WxChMsgRouter-pool-%d").build();
    this.executorService = new ThreadPoolExecutor(2, 100,
      0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(), namedThreadFactory);
    //this.sessionManager = new StandardSessionManager();
    this.exceptionHandler = new LogExceptionHandler();
    this.messageDuplicateChecker = WxMessageInMemoryDuplicateCheckerSingleton.getInstance();
  }

  /**
   * 使用自定义的 {@link ExecutorService}.
   */
  public WxChannelMessageRouter(ExecutorService executorService) {
    this.executorService = executorService;
    this.exceptionHandler = new LogExceptionHandler();
    this.messageDuplicateChecker = WxMessageInMemoryDuplicateCheckerSingleton.getInstance();
  }

  /**
   * 系统退出前,应该调用该方法
   */
  public void shutDownExecutorService() {
    this.executorService.shutdown();
  }

  /**
   * 系统退出前,应该调用该方法,增加了超时时间检测
   */
  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}
   * 如果不调用该方法,默认使用内置的
   * 
*/ public void setExecutorService(ExecutorService executorService) { this.executorService = executorService; } /** * 消息路由入口 * * @param message 消息 * @param content 原始消息(解密之后的) * @param appId appId * @param service 服务实例 * @return 返回值 */ public Object route(final WxChannelMessage message, final String content, final String appId, final WxChannelService service) { return this.route(message, content, appId, new HashMap<>(2), service, new StandardSessionManager()); } /** * 路由微信消息 * * @param message 消息 * @param content 消息原始内容 * @param context 上下文 * @return Object */ public Object route(final WxChannelMessage message, final String content, final String appId, final Map context, final WxChannelService service, final WxSessionManager sessionManager) { // 如果是重复消息,那么就不做处理 if (isMsgDuplicated(message)) { log.info("收到重复消息,{}", content); return null; } final List> matchRules = new ArrayList<>(); // 收集匹配的规则 for (final WxChannelMessageRouterRule rule : this.rules) { if (rule.isMatch(message)) { matchRules.add(rule); if (!rule.isNext()) { break; } } } if (matchRules.size() == 0) { return null; } final List> futures = new ArrayList<>(); Object result = null; for (final WxChannelMessageRouterRule rule : matchRules) { // 返回最后一个非异步的rule的执行结果 if (rule.isAsync()) { futures.add( this.executorService.submit(() -> { rule.process(message, content, appId, context, service, sessionManager, exceptionHandler); }) ); } else { result = rule.process(message, content, appId, context, service, sessionManager, exceptionHandler); // 在同步操作结束,session访问结束 sessionEndAccess(sessionManager, message, false); } } if (futures.size() > 0) { this.executorService.submit(() -> { for (Future future : futures) { try { future.get(); // 异步操作结束,session访问结束 sessionEndAccess(sessionManager, message, true); } catch (InterruptedException | ExecutionException e) { log.error("Error happened when wait task finish", e); } } }); } return result; } /** * 判断消息是否重复 * * @param wxMessage 消息 * @return 是否重复 */ protected boolean isMsgDuplicated(WxChannelMessage wxMessage) { String messageId = this.generateMessageId(wxMessage); return this.messageDuplicateChecker.isDuplicate(messageId); } /** * 生成消息id * * @return 消息id */ protected String generateMessageId(WxChannelMessage wxMessage) { StringBuilder sb = new StringBuilder(); if (wxMessage.getMsgId() == null) { sb.append(wxMessage.getCreateTime()) .append("-").append(wxMessage.getFromUser()) .append("-").append(StringUtils.trimToEmpty(wxMessage.getEvent())); } else { sb.append(wxMessage.getMsgId()) .append("-").append(wxMessage.getCreateTime()) .append("-").append(wxMessage.getFromUser()); } if (StringUtils.isNotEmpty(wxMessage.getToUser())) { sb.append("-").append(wxMessage.getToUser()); } return sb.toString(); } /** * 对session的访问结束 * * @param sessionManager session管理器 * @param message 消息 * @param async 是否异步 打印log用 */ private void sessionEndAccess(WxSessionManager sessionManager, WxChannelMessage message, boolean async) { log.debug("End session access: async={}, sessionId={}", async, message.getFromUser()); InternalSession session = ((InternalSessionManager) sessionManager).findSession(message.getFromUser()); if (session != null) { session.endAccess(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy