首页 > 其他分享 >Spring Cloud Admin添加微信通知

Spring Cloud Admin添加微信通知

时间:2023-04-09 14:00:56浏览次数:47  
标签:jsonObject String accessToken Admin Spring token 微信 null public

SpringCloud发送微信消息推送参考https://blog.csdn.net/qq_44697754/article/details/128035736

 
 

SpringCloud Admin要增加微信通知,需要继承AbstractStatusChangeNotifier类,在doNotify方法按照模板发送消息。
 
AdminServe添加依赖:

<dependency>
        <groupId>commons-lang</groupId>
        <artifactId>commons-lang</artifactId>
        <version>2.3</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>fastjson</artifactId>
        <version>2.0.21.graal</version>
    </dependency>

    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
    </dependency>

 
文件application.properties增加配置:

wechat.originalId=gh_dca23axxx
wechat.appId=wx360b2xxxxx
wechat.appsecret=1a7876a7bxxxxxxx
wechat.templateId=-Z6u1ASSqsstKrzxxxxxxxxxxxxx
wechat.accessTokenUrl=https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential
wechat.userGetUrl=https://api.weixin.qq.com/cgi-bin/user/get
wechat.messageSendUrl=https://api.weixin.qq.com/cgi-bin/message/template/send
wechat.templateMassage={\"touser\":\"%s\",\"template_id\":\"%s\",\
             \"data\":{\"serviceName\": {\"value\":\"%s\",\"color\":\"#00BFFF\"},\"serviceUrl\":{\"value\":\"%s\",\"color\":\"#00FFFF\"},\
             \"status\": {\"value\":\"%s\",\"color\":\"#173177\"},\"details\": {\"value\":\"%s\",\"color\":\"#EE212D\"}}}

 
 

微信接口配置类WechatServiceProperties:

@Getter
@Setter
@Component
@ConfigurationProperties(prefix = "wechat")
public class WechatServiceProperties {


   /** 获取access_token的接口地址(GET) 限2000(次/天) */
    private  String  accessTokenUrl = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential";

    /** 获取关注用户id */
    private  String userGetUrl  = "https://api.weixin.qq.com/cgi-bin/user/get";

    /**
     * 发送消息url
     */
    private String messageSendUrl = "https://api.weixin.qq.com/cgi-bin/message/template/send";

    /**
     * 模板id
     */
    private String  templateId = "91921dL1CxrEUms2rnIlKi9_rfmwmiICaiaXxAbhsc8";

    /**
     * 模板消息
     */
    private String templateMassage;

    /**
     * appId
     */
    private String appId;

    private String appsecret;

    private String originalId;


    public String getAccessTokenUrl() {
        if (StringUtils.isBlank(accessTokenUrl)) {
            throw new IllegalArgumentException("获取access_token的接口地址不能为空");
        }

        if (StringUtils.isBlank(appId)) {
            throw new IllegalArgumentException("appId不能为空");
        }

        if (StringUtils.isBlank(appsecret)) {
            throw new IllegalArgumentException("appsecret不能为空");
        }
        return String.format(accessTokenUrl+"&appid=%s&secret=%s",appId, appsecret);
    }




    public String formatGetUserUrl(String accessToken) {
        if (StringUtils.isBlank(accessToken)) {
            throw  new IllegalArgumentException("accessToken不能为空");
        }

        return String.format(userGetUrl+"?access_token=%s", accessToken);
    }

    public String formatMessageSendUrl(String accessToken) {
        if (StringUtils.isBlank(messageSendUrl)) {
            throw  new IllegalArgumentException("messageSendUrl不能为空");
        }

        if (StringUtils.isBlank(accessToken)) {
            throw  new IllegalArgumentException("accessToken不能为空");
        }
        return  String.format(messageSendUrl+"?access_token=%s", accessToken);
    }


    public String formatTemplateMassage(TemplateMassageParam param) {
        if (StringUtils.isBlank(messageSendUrl)) {
            throw  new IllegalArgumentException("messageSendUrl不能为空");
        }

        if (param == null) {
            throw  new IllegalArgumentException("要发送的消息不能为空");
        }
       return String.format(templateMassage,param.getTouser(), param.getTemplateId(), param.getServiceName(), param.getServiceUrl(), param.getStatus(), param.getDetails());
    }
}

 

 

AccessToken

public class AccessToken implements java.io.Serializable{

  
	private static final long serialVersionUID = -4240357901925120079L;

	/** 获取到的凭证 */ 
	private String token;

	/** 凭证有效时间,单位:秒 */ 
	private int expiresIn;

	public AccessToken() {
	}

	public AccessToken(String token, int expiresIn) {
		this.token = token;
		this.expiresIn = expiresIn;
	}

	public String getToken() {
		return token;
	}

	public void setToken(String token) {
		this.token = token;
	}

	public int getExpiresIn() {
		return expiresIn;
	}

	public void setExpiresIn(int expiresIn) {
		this.expiresIn = expiresIn;
	}
}

 

HttpUtils用于发送http请求:

Slf4j
public class HttpUtils {

    public static JSONObject httpsRequest(String requestUrl, String requestMethod, String outputStr) {
        log.debug("[发起https请求]requestUrl=" + requestUrl);
        log.debug("[发起https请求]requestMethod=" + requestMethod);
        log.debug("[发起https请求]outputStr=" + outputStr);
        JSONObject jsonObject = null;
        StringBuffer buffer = new StringBuffer();
        try {
            // 创建SSLContext对象,并使用我们指定的信任管理器初始化
            TrustManager[] tm = {new X509TrustManager() {
                @Override
                public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {

                }

                @Override
                public void checkServerTrusted(X509Certificate[] chain, String authType) {

                }

                @Override
                public X509Certificate[] getAcceptedIssuers() {
                    return new X509Certificate[0];
                }
            }};
            SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");
            sslContext.init(null, tm, new java.security.SecureRandom());
            // 从上述SSLContext对象中得到SSLSocketFactory对象
            SSLSocketFactory ssf = sslContext.getSocketFactory();

            URL url = new URL(requestUrl);
            HttpsURLConnection httpUrlConn = (HttpsURLConnection) url
                    .openConnection();
            httpUrlConn.setSSLSocketFactory(ssf);
            httpUrlConn.setDoOutput(true);
            httpUrlConn.setDoInput(true);
            httpUrlConn.setUseCaches(false);
            // 设置请求方式(GET/POST)
            httpUrlConn.setRequestMethod(requestMethod);

            //if ("GET".equalsIgnoreCase(requestMethod))
            httpUrlConn.connect();

            // 当有数据需要提交时
            if (null != outputStr) {
                OutputStream outputStream = httpUrlConn.getOutputStream();
                // 注意编码格式,防止中文乱码

                outputStream.write(outputStr.getBytes("UTF-8"));
                outputStream.close();

            }

            // 将返回的输入流转换成字符串
            InputStream inputStream = httpUrlConn.getInputStream();
            InputStreamReader inputStreamReader = new InputStreamReader(
                    inputStream, "utf-8");
            BufferedReader bufferedReader = new BufferedReader(
                    inputStreamReader);

            String str = null;
            while ((str = bufferedReader.readLine()) != null) {
                buffer.append(str);
            }
            bufferedReader.close();
            inputStreamReader.close();
            // 释放资源
            inputStream.close();
            httpUrlConn.disconnect();
            jsonObject = JSONObject.parseObject(buffer.toString());
        } catch (ConnectException ce) {
            log.error("Weixin server connection timed out.");
        } catch (Exception e) {
            log.error("https request error:{}", e);
        }
        return jsonObject;
    }

}

 
 

模板消息参数TemplateMassageParam:

@Setter
@Getter
@Builder
public class TemplateMassageParam {

    /**
     * 服务名
     */
    private String serviceName;

    /**
     * 服务url
     */
    private String serviceUrl;

    /**
     * 服务状态
     */
    private String status;

    /**
     * 详情
     */
    private Map<String, Object> details;

    /**
     * 发送到的用户
     */
    private String touser;

    /**
     * 消息模板id
     */
    private String templateId;
}

 
 

通知发送类WechatNotifier:

@Component
@Slf4j
public class WechatNotifier extends AbstractStatusChangeNotifier {

    private Map<String, AccessToken> map = new HashMap<>();

    @Autowired
    private WechatServiceProperties wechatServiceProperties;

    @Autowired
    public WechatNotifier(InstanceRepository repository) {
        super(repository);
    }

    @Override
    protected Mono<Void> doNotify(InstanceEvent event, Instance instance) {
        Registration registration = instance.getRegistration();
        // 获取关注用户
        List<String> users = this.getUsers();
        if (users != null && users.size() > 0) {
            TemplateMassageParam param = TemplateMassageParam.builder().serviceName(registration.getName())
                    .serviceUrl(registration.getServiceUrl())
                    .status(instance.getStatusInfo().getStatus())
                    .details(instance.getStatusInfo().getDetails())
                    .templateId(wechatServiceProperties.getTemplateId())
                    .build();

            String sendUrl = wechatServiceProperties.formatMessageSendUrl(map.get("token").getToken());
            // 并行遍历推送
            users.stream().parallel().forEach(x->{
                if (x != null) {
                    param.setTouser(x);
                    String templateMassage = wechatServiceProperties.formatTemplateMassage(param);
                    log.info("[发送模板信息]sendTemplateMessage:"+templateMassage);
                    JSONObject jsonObject = null;
                    jsonObject = HttpUtils.httpsRequest(sendUrl, "POST", templateMassage);
                    log.info("[发送模板信息] sendTemplateMessage result:"+jsonObject);
                }
            });
        }
        return Mono.empty();
    }


    /**
     * 获取token
     * @return
     */
    public AccessToken getAccessToken() {
        AccessToken accessToken = null;
        String requestUrl = wechatServiceProperties.getAccessTokenUrl();
        JSONObject jsonObject=null;
        jsonObject = HttpUtils.httpsRequest(requestUrl, "GET", null);
        log.info("[(刷新)获取access_token]jsonObject="+jsonObject);
        jsonObject = JSONObject.parseObject((String) jsonObject.toString());

        if (null != jsonObject) {
            try {
                accessToken = new AccessToken(jsonObject.getString("access_token"), jsonObject.getInteger("expires_in"));
            } catch (JSONException e) {
                log.error("获取token失败 errcode:{} errmsg:{}"+jsonObject.getInteger("errcode")+"errmsg:"+jsonObject.getString("errmsg"),e);
            }
        }
        log.info("[获取access_token]jsonObject="+jsonObject);
        log.info("[获取access_token]accessToken="+accessToken.getToken());
        return accessToken;
    }

    /**
     * 获取关注用户
     * @return
     */
    public List<String> getUsers() {
        try {
            log.info("------------ [获取用户信息] ---------------");
            String token = null;
            AccessToken accessToken  = map.get("token");
            if (accessToken == null) {
                accessToken = this.getAccessToken();
                map.put("token",accessToken);
            }

            if (accessToken != null) {
                token = accessToken.getToken();
            }
            if (token != null) {
                String requestUrl = wechatServiceProperties.formatGetUserUrl(token);
                JSONObject jsonObject = HttpUtils.httpsRequest(requestUrl, "GET", null);
                String dataJson = jsonObject.getString("data");
                JSONObject data = JSONObject.parseObject(dataJson);
                JSONArray openid = data.getJSONArray("openid");
                List<String> list = new ArrayList<>();
                if (openid != null && openid.size() > 0) {
                    for (int i = 0; i < openid.size(); i++) {
                        Object obj = openid.get(i);
                        if (obj != null) {
                            list.add(obj.toString());
                        }
                    }
                }
                return list;
            }
        } catch (Exception e) {
            log.error("[获取关注用户信息]数据异常:",e);
        }
        return null;
    }
}

思路是从微信获取token,在获取关注公众号的用户,按照模板构造消息,向微信接口发送消息。如果发送消息出现data format error错误,应该是模板消息不是正确的JSON格式。修改模板消息成正确的JSON格式。
 

从微信测试公众号可以看到:

标签:jsonObject,String,accessToken,Admin,Spring,token,微信,null,public
From: https://www.cnblogs.com/shigongp/p/17300235.html

相关文章

  • Idea点击Run或者Debug无法启动项目_调试按钮按下以后变灰色_一会又恢复成绿色_但项目
    这个现象很烦人,点击了无数次了,就是项目启动不起来,很郁闷后来终于弄明白了,是这里,点击settings,然后找到这个runner这里,然后左上角这个delegateIDE...这个把勾去掉,去掉就可以了. 可以看到去掉以后,然后再点击运行可以看到,就已经显示正在运行中了. 终于弄好~......
  • ava: 程序包com.alibaba.nacos.api.common不存在_RuoYi-Cloud-Plus-master_jar包不存
    来看看原因吧,jar包是存在的,但是就是在idea中引用不到,来看看怎么回事: 原来就是这个包找不到,但是从下面看是有的: 但是注意,这里的com.alibaba.nacos.api...原来可不是这样的,这个是我后来修改过的,原来是只有com.alibaba.nacos.common,而引用的是com.alibaba.nacos.api.commo......
  • ruoyi-cloud微服务版启动过程报错_20230320版_ Verion 9 of Highlight.js has reached
      Verion9ofHighlight.jshasreachedEOL. Itwillnolonger报错: 这里修改成10.7.3版本D:\2023\qdBigData\RuoYi-Cloud-master\ruoyi-ui>npminstall--registry=https://registry.npm.taobao.org然后到对应目录,再去执行编译去看看.不报错了 >npmrundev然后执行看......
  • SpringBoot中日志的使用
    springboot默认就是使用SLF4J作为日志门面,logback作为日志实现来记录日志的文章目录1.SpringBoot中的日志设计2.SpringBoot日志使用1.SpringBoot中的日志设计springboot中的日志<dependency> <artifactId>spring-boot-starter-logging</artifactId> <groupId>org.springfr......
  • Java SpringBoot Bean InitializingBean
    Spring中有两种类型的Bean,一种是普通Bean,另一种是工厂Bean,即FactoryBean。工厂Bean跟普通Bean不同,其返回的对象不是指定类的一个实例,其返回的是该工厂Bean的getObject方法所返回的对象。Spring初始化bean有两种方式:实现InitializingBean接口,实现afterPropertiesSet方法。(比通过反射......
  • Java SpringBoot Test 单元测试中包括多线程时,没跑完就结束了
    如何阻止JavaSpringBootTest单元测试中包括多线程时,没跑完就结束了使用CountDownLatchCountDownLatch、CyclicBarrier使用区别多线程ThreadPoolTaskExecutor应用JavaBasePooledObjectFactory对象池化技术@SpringBootTestpublicclassPoolTest{@Testvoid......
  • Spring 源码解析 --AOP
        ......
  • 不依赖 Spring 的简单 Main 工程
    1.搭建一个没有spring的工程importorg.jetbrains.kotlin.gradle.tasks.KotlinCompileplugins{kotlin("jvm")version"1.8.10"application}group="com.demo"version="1.0-SNAPSHOT"repositories{mavenCent......
  • SpringCloud使用Consul作为配置中心
    Consul提供了一个key/value存储,用于存储配置和其他元数据。SpringCloudConsulConfig是ConfigServer和Client的替代方案。在特殊的“bootstrap”阶段,配置被加载到Spring环境中。默认情况下,配置存储在/config文件夹中。根据应用程序的名称和模拟SpringCloud配置顺序解析财产的......
  • SpringCloud源码学习笔记3——Nacos服务注册源码分析
    系列文章目录和关于我一丶基本概念&Nacos架构1.为什么需要注册中心实现服务治理、服务动态扩容,以及调用时能有负载均衡的效果。如果我们将服务提供方的ip地址配置在服务消费方的配置文件中,当服务提供方实例上线下线,消费方都需要重启服务,导致二者耦合度过高。注册中心就是在......