草庐IT

微信小程序向公众号推送模板消息(根据用户登录小程序openid实现向同一主体下对应公众号推送模板消息)

小刘爱搬砖 2023-04-05 原文

最近文章更新没有那么频繁,可能每隔一个月左右会更一篇文章,主要就是我正在学习的内容demo或者工作中刚好用到的内容,有的篇幅内容相较于之前的文章会略长,可以根据目录检测自己需要查看的段落。

这里只针对第二种方式写了demo,后期可能对第一种方式进行补充。

方式一:
向用户推送消息需要用到用户的openid,当用户关注公众号后就会生产一个唯一不变的openid,小程序是在登录注册时就可以获取到openid,但是小程序的openid和公众号的openid是各自独立的,我们通过返回的unionid来对二者进行关联。
方式二【本文演示的】
通过微信提供的接口也可以实现,通过用户登录小程序产生的openid来实现消息的推送(用户如果未登录小程序不能实现消息的推送,也必须对公众号进行关注)
官方文档地址:https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/mp-message-management/uniform-message/sendUniformMessage.html

注意:当前小程序和公众号必须在同一主体下

请求的参数主要包含

  • access_token 可以通过get请求来进行获取
  • touser 用户的openid,小程序返回的openid
  • mp_template_msg 推送公众号消息的参数
    • appid 公众号的id
    • template_id 模板的id
    • url 模板所要跳转的内容
    • miniprogram 模板索要跳转的小程序
    • data 小程序要发送的消息内容

参数示例

一个消息发送的实例

{
  "touser":"OPENID",
  "mp_template_msg":{
        "appid":"APPID ",
        "template_id":"TEMPLATE_ID",
        "url":"http://weixin.qq.com/download",
        "miniprogram":{
            "appid":"xiaochengxuappid12345",
            "pagepath":"index?foo=bar"
        },
        "data":{
            "first":{
                "value":"恭喜你购买成功!",
                "color":"#173177"
            },
            "keyword1":{
                "value":"巧克力",
                "color":"#173177"
            },
            "keyword2":{
                "value":"39.8元",
                "color":"#173177"
            },
            "keyword3":{
                "value":"2014年9月22日",
                "color":"#173177"
            },
            "remark":{
                "value":"欢迎再次购买!",
                "color":"#173177"
            }
        }
    }
}

返回成功数据的实例

{
  "errCode": 0,
  "errMsg": "openapi.uniformMessage.send:ok"
} 

整合Springboot

用到的依赖

 <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.73</version>
        </dependency>

        <!-- 缓存 用来保存access_token-->
        <dependency>
            <groupId>com.github.ben-manes.caffeine</groupId>
            <artifactId>caffeine</artifactId>
        </dependency>
        
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>

        <!-- lombok包中携带Slf4j日志 -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
   
        <!-- 定时任务 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-quartz</artifactId>
        </dependency>

    </dependencies>
  1. 首先你要获取到access_token

官方文档地址:http://caibaojian.com/wxwiki/9a186d136f1e9dce26e9593cadbc7130083b48d0.html
公众号在进行接口调用时都会需要access_token,这相当于我们和微信服务器进行交互的凭证,access_token每两个小时会刷新一次,新的token生效后前面的token会失效,两个token进行交替时5min内新老token都可以使用。我们每天对接口的调用是有次数限制的,每天最多调用2000次。



1.1 创建一个实体类,用来保存微信相关的配置信息

@Data
@Component
@PropertySource(value = "classpath:/application.yml")
@ConfigurationProperties(prefix = "wechat")
public class WeChatProperties {
    /**
     * 微信公众号的appid
     */
    @Value("${appId}")
    private String appId;
    /**
     * 微信小程序的secret
     */
    @Value("${secret}")
    private String secret;
    /**
     * 获取access_token的url
     */
    @Value("${accessUrl}")
    private String accessUrl;
    /**
     * 消息发送的id
     */
    @Value("${uniformSend}")
    private String uniformSend;
    /**
     * 微信小程序的appid
     */
    @Value("${miniAppId}")
    private String miniAppId;
    /**
     * 微信小程序跳转页面路径
     */
    @Value("${pagepath}")
    private String pagepath;

    /**
     * 获取token的定时任务corn
     */
    @Value("${corn}")
    private String corn;

}

1.2 这里我直接使用的restTemplate接口接口的调用,也可以使用httpclient,自由选择
所以这里需要对restTemplate进行配置

/**
 * @desc:
 * @author: LiuChang
 * @since: 2022/7/22
 */
@Configuration
public class RestTemplateConfig {

    /**
     * rest 模板
     *
     * @param clientHttpRequestFactory
     * @return
     */
    @Bean
    public RestTemplate restTemplate(ClientHttpRequestFactory clientHttpRequestFactory) {
        return new RestTemplate(clientHttpRequestFactory);
    }

    /**
     * 请求连接池的配置信息
     *
     * @return
     */
    @Bean
    public ClientHttpRequestFactory simpleClientHttpRequestFactory() {
        SimpleClientHttpRequestFactory factory = new SimpleClientHttpRequestFactory();
        factory.setConnectTimeout(15000);
        factory.setReadTimeout(5000);
        return factory;
    }

}
  1. 在yaml配置文件中指定配置信息
server:
  port: 7777
wechat:
  #公众号的appid
  appId: 
  #小程序的appid
  miniAppId: 
  #小程序的secret
  secret: 
  accessUrl: https://api.weixin.qq.com/cgi-bin/token
  uniformSend: https://api.weixin.qq.com/cgi-bin/message/wxopen/template/uniform_send?access_token=
  #小程序跳转的url
  pagepath: pages/index
  #定时任务执行
  corn: 0 0 0/2 * * ?
  1. 调用接口获取token
@Resource
RestTemplate restTemplate;
@Resource
WeChatProperties weChatProperties;

public String getAccessToken() {
        String url = weChatProperties.getAccessUrl();
        // 这里的参数要和下面的Map Key值对应
        String path = "?grant_type={grant_type}&appid={appid}&secret={secret}";
        Map<String, String> params = new HashMap<>(3);
        params.put("grant_type", "client_credential");
        params.put("appid", weChatProperties.getMiniAppId());
        params.put("secret", weChatProperties.getSecret());
        ResponseEntity<String> forObject = restTemplate.getForEntity(url + path, String.class, params);
        JSONObject jsonObject = JSONObject.parseObject(forObject.getBody());
        String accessToken = jsonObject.getString("access_token");
        if (null == accessToken) {
            log.error("获取access_token失败");
        } else {
            //将token保存到缓存中
            //caffeineCache.put("access_token", accessToken);
        }
        return accessToken;
    }
  1. 引入缓存(可忽略)

2.1 缓存的配置文件 我这里缓存用的caffeine,主要是读的性能比较好,由于项目只有token用到了缓存就引入整个,也可以使用redis/memoryCache

/**
 * @desc: token的缓存配置
 * @author: LiuChang
 * @since: 2022/7/22
 */
@Configuration
public class TokenCacheConfig {

    @Bean
    public Cache caffeineCache() {
        return Caffeine.newBuilder()
                // 设置最后一次写入或访问后经过固定时间过期
//                .expireAfterWrite(90, TimeUnit.MINUTES)
                // 初始的缓存空间大小
                .initialCapacity(100)
                // 缓存的最大条数
                .maximumSize(1000)
                .build();
    }

}

2.2 使用cache

@Resource
Cache<String, Object> caffeineCache;


//获取
 caffeineCache.getIfPresent("access_token")
//放入数据
caffeineCache.put("access_token", accessToken);
  1. 引入定时任务(可忽略)

3.1 需要执行的定时任务

@Slf4j
@Component
public class QuartzJob extends QuartzJobBean {
    @Resource
    WxMessageSendService wxMessageSendService;

    @Override
    protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
        try {
            // 定时刷新token
            String token = wxMessageSendService.getAccessToken();
            log.info(token);
        } catch (Exception e) {
            log.error("任务失败 error:{}", e.getMessage());
        }
    }
}


3.2 定时任务的配置文件

@Configuration
public class QuartzConfig {
    /**
     * corn表达式
     */
    @Value("${wechat.corn}")
    private String restartCron;

    @Bean
    public JobDetail restartJob() {
        return JobBuilder.newJob(QuartzJob.class).withIdentity("QuartzJob").storeDurably().build();
    }

    @Bean
    public Trigger restartTrigger() {
        CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(restartCron);
        return TriggerBuilder.newTrigger().forJob(restartJob())
                .withIdentity("QuartzJob").withSchedule(scheduleBuilder).build();
    }

}

3.3 启动类增加注解

@SpringBootApplication
@EnableScheduling
  1. 调用官方接口,推送消息测试

消息内容的实体类

/**
 * @desc: 微信推送消息类
 * @author: LiuChang
 * @since: 2022/7/25
 */
@Data
@AllArgsConstructor
@NoArgsConstructor
public class WxMessageBean {
    private MessageBean first;

    private MessageBean keyword1;

    private MessageBean keyword2;

    private MessageBean keyword3;

    private MessageBean keyword4;

    private MessageBean remark;

    /**
     * 自定义构造,first和remark必传,keyword根据需要自定义个数
     *
     * @param first
     * @param remark
     * @param keyword
     */
    public WxMessageBean(MessageBean first, MessageBean remark, MessageBean... keyword) {
        this.first = first;
        int count = 1;
        for (MessageBean keyword1 : keyword) {
            if (count == 1) {
                this.keyword1 = keyword1;
            } else if (count == 2) {
                this.keyword2 = keyword1;
            } else if (count == 3) {
                this.keyword3 = keyword1;
            } else if (count == 4) {
                this.keyword4 = keyword1;
            }
            count++;
        }

        this.remark = remark;
    }

}
@Data
@AllArgsConstructor
@NoArgsConstructor
public class MessageBean {
    /**
     * 消息内容
     */
    private String value;
    /**
     * color
     */
    private String color;
}

接口编写

@Service
@Slf4j
public class WxMessageSendService {
    @Resource
    RestTemplate restTemplate;
    @Resource
    WeChatProperties weChatProperties;
    @Resource
    Cache<String, Object> caffeineCache;

    /**
     * 推送消息
     *
     * @param openId
     * @param message
     * @return
     */
    public String messageSend(String openId, Object message) {
        String token = (String) caffeineCache.getIfPresent(ConstantEnum.ACCESS_TOKEN.getCode());
        if (token == null) {
            token = this.getAccessToken();
        }
        String url = weChatProperties.getUniformSend() + token;
        // 这里的参数要和下面的Map Key值对应
        JSONObject obj = new JSONObject();
        JSONObject mpTemplateMsg = new JSONObject();
        JSONObject mini = new JSONObject();
        mini.put("appid", weChatProperties.getMiniAppId());
        mini.put("pagepath", weChatProperties.getPagepath());
        mpTemplateMsg.put("data", message);
        mpTemplateMsg.put("miniprogram", mini);
        ///微信公众号appid
        mpTemplateMsg.put("appid", weChatProperties.getAppId());
        ///微信公众号模板id
        mpTemplateMsg.put("template_id", template_id);
        obj.put("touser", openId);
        obj.put("mp_template_msg", mpTemplateMsg);
        ResponseEntity<String> forObject = restTemplate.postForEntity(url, obj, String.class);
        JSONObject object = JSON.parseObject(forObject.getBody());
        Integer errcode = (Integer) object.get("errcode");
        if (errcode == 0) {
            log.info("消息推送成功");
            return null;
        } else if (errcode == 40003) {
            log.error("推送消息的openid错误,openid:{},消息内容:{}", openId, message);
            return "推送消息的openid错误";
        } else if (errcode == 43004) {
            log.error("该用户未关注公众号,openid:{},消息内容:{}", openId, message);
            return "该用户未关注公众号";
        } else {
            return null;
        }
    }
    
    //下面还有个方法是获取access_token的 这里省略没写,在上面获取token已经将方法写了
}

调用接口

@RestController
@RequestMapping("/message_send")
public class WxMessageSendController {
    @Resource
    WxMessageSendService wxMessageSendService;

    /**
     * 通过openid和消息内容向指定用户推送公众号模板消息
     *
     * @return
     */
    @PostMapping()
    public String messageSend() {
        WxMessageBean messageBean = new WxMessageBean(new MessageBean("你有新的待审批消息,请点击查看", "#173177"),
                                                      new MessageBean("footer", "#173177"),
                                                      new MessageBean("keyword1", "#173177"));
        return wxMessageSendService.messageSend(openid, messageBean);
    }

}

调用接口后的结果

注意事项:

  1. 两个appid和secret要对应填写正确别写反
  2. openid是用户小程序的id
  3. access_token 参数填写正确

有关微信小程序向公众号推送模板消息(根据用户登录小程序openid实现向同一主体下对应公众号推送模板消息)的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  3. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  4. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  5. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  6. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  7. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  8. ruby-on-rails - 使用 rails 4 设计而不更新用户 - 2

    我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它​​不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数

  9. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  10. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

随机推荐