全局唯一接口调用凭据 access_token

用于接口调用的一个必要参数

有了 access_token 就能实现所有的接口

  • 特点: 

1. 有效期为 2 小时,所以 2 小时要更新一次,提前 5 分钟更新(确保后续正常使用)

2. 如果重复获取,会导致上一次失效需要 appid 和 appsecret 来获取)

3. access_token 存储至少要保留 512 个字符空间

4. 接口调用有限制,普通 2000次/天,测试号200次/天

  • 请求方式  https GET

https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET

正常情况下,微信会返回下述JSON数据包给公众号:

{"access_token":"ACCESS_TOKEN", "expires_in":7200}

全局返回码,参见

  • 设计思路

1. 第一次发送请求,获取 access_token,保存在将来 2 小时内使用

2. 以后发送请求,读取上一次保存的 access_token,判断是否过期

过期了,重新发送请求获取 access_token

没有过期,直接使用

优化为 getValidAccessToken()

直接 redAccessToken() 读取 access_token,判断是否过期 isValidAccessToken()

读取成功:

没过期,直接用

过期,发送请求 requestAccessToken() 获取 access_token,saveAccessToken()

读取失败

发送请求 requestAccessToken() 获取 access_token,saveAccessToken()

class WeChat {

async requestAccessToken(){

// 定义请求地址和参数

const url = `https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=${???}&secret=${???}`;

// 导入发送请求的库 request-promise-native

// request  无需导入

// npm install request  request-promise-native 发送请求(服务器端不能用 ajax)

// 请求方式,请求地址,如果响应回来的数据是JSON,则自动转化为 js 对象

// {"access_token":"ACCESS_TOKEN", "expires_in":7200}

const access_token= await rp({method:'GET', url, json: true});

// 重写过期时间,提前 5 分钟刷新

accessToken

return accessToken;

}    // 返回一个 Promise 对象,其中有 access_token 对象

saveAccessToken(accessToken){    // 为了不被修改,使用 fs 模块将 access_token 写入 txt 文件

// 写入文件时,无法写 数组、函数、对象 类型的数据 [object Object]

new Promise((resolve, reject)=>{

writeFile('./access_token.txt', JSON.stringify(accessToken), err=>{

if(err){

reject(err);

}else{

resolve('保存 access_token 成功');

}

});

});

}

readAccessToken(){

new Promise((resolve, reject)=>{

readFile('./access_token.txt', (err, data)=>{

if(err){

reject(err);

}else{

resolve('读取 access_token 成功');

}

});

});

}

isValidAccessToken({expires_in}){

return (expires_in > Date.now);

}

}

// 直接测试:

(async ()=>{

const w = new WeChat();

w.getAccessToken().then(async result=>{

if(w.isValidAccessToken(result)){

return result;

}else{

result = await w.getAccessToken();

await w.saveAccessToken();

return result;

};

}).catch(err=>{    // 第一次读取,会失败

result = await w.getAccessToken();

await w.saveAccessToken();

return result;

}).then(result=>{

console.log(result);

});

})();

接口编程(方法需要参数 access_token)

  • 自定义菜单(创建菜单,可能不会马上生效,微信服务器需要时间更新)

自定义菜单最多包括 3 个一级菜单

每个一级菜单最多包含 5 个二级菜单

菜单有 10 中类型

凡是 POST 请求都有 请求体数据

只要没有 请求体 数据,就一定是 GET 请求

  • 创建菜单(要先删除老菜单,才能创建新菜单)

const body = {

"button":[
{
"type":"click",
"name":"一级菜单微信表情",
"key":"click"
},
{
"name":"二级菜单",
"sub_button":[
{
"type":"view",
"name":"百度",
"url":"http://www.baidu.com/"
},
{
"type":"miniprogram",
"name":"wxa",
"url":"http://mp.weixin.qq.com",
"appid":"wx286b93c14bbf93aa",
"pagepath":"pages/lunar/index"
},
{
"type":"click",
"name":"赞一下我们",
"key":"V1001_GOOD"
}]
}]
"button":[
{
"type":"click",
"name":"一级菜单微信表情",
"key":"click"
},
{
"name":"二级菜单",
"sub_button":[
{
"type":"view",
"name":"百度",
"url":"http://www.baidu.com/"
},
{
"type":"miniprogram",
"name":"wxa",
"url":"http://mp.weixin.qq.com",
"appid":"wx286b93c14bbf93aa",
"pagepath":"pages/lunar/index"
},
{
"type":"click",
"name":"赞一下我们",
"key":"V1001_GOOD"
}]
}]

};

creaetMenu(body){

// 获取到 access_token

const {access_token} = await this.fetchAccessToken();

// 2. 定义请求体地址

const url = `https://api.weixin.qq.com/cgi-bin/menu/create?access_token=${access_token}`;

// 3. 发送请求

const result = await rp({method:'POST',  url, json:true, body});

return result;

}

  • 尝试: 事件推送_接口
  • 尝试: 查询_接口

源代码:

index.js

  • const express = require('express');
    const {interfaceInit} = require('./interfaceInit'); const app = express(); interfaceInit(); // 中控服务器 初始化 app.listen(
    3000,
    err=>console.log(err?err:'\n\n服务器已启动\n\t\tHunting Happy!')
    );

interfaceInit/index.js

  • /****
    * access_token 对象____中控服务器----公众号的全局唯一接口调用凭据
    *
    * {
    * access_token: '17_Nq3M5HMdnX3Xwkbi48uPEaVZ4qnh_H5B8HOzBy-DnXqLz6s9h3ALAPd6sk11K0zclzu0Ap3cZciBVp2aml9EuJGmSZ-iGKe7gFOwVUEYGhOB70Il9GeCMWtppgpXcdMzm7YaqVE_W55L1bgfBEQcAHAGJV',
    * expires_in: 7200
    * }
    ****/
    const promiseRequest = require('request-promise-native');
    const {APPID, APPSECRET, accessToken} = require('../config');
    const {writeFile, readFile} = require('fs'); const {menu, deleteMenu, createMenu} = require('./menu'); class WeChat{
    getValidAccessToken(){
    if(this.access_token && this.isValidAccessToken(this)){
    return Promise.resolve({
    access_token: this.access_token,
    expires_in: this.expires_in
    });
    }else{
    return this.readAccessToken().then(async objAccessToken=>{
    if (this.isValidAccessToken(objAccessToken)){
    return objAccessToken;
    }else{
    const newObjAccessToken = await this.requestAccessToken();
    await this.saveAccessToken(newObjAccessToken);
    return newObjAccessToken;
    }
    }).catch(async err=>{
    const newObjAccessToken = await this.requestAccessToken();
    await this.saveAccessToken(newObjAccessToken);
    return newObjAccessToken;
    }).then(objAccessToken=>{
    // 更新 WeChat
    this.access_token = objAccessToken.access_token;
    this.expires_in = objAccessToken.expires_in; // 返回 Promise 的 access_token
    return Promise.resolve(objAccessToken);
    });
    };
    } readAccessToken(){ // 一、读取access_token的方法
    return new Promise((resolve, reject)=>{
    readFile('./access_token.txt', (err, buffer)=>{
    if(err){
    reject('Read ./access_token.txt' + err);
    }else{
    resolve(JSON.parse(buffer.toString()));
    }
    });
    });
    } isValidAccessToken({expires_in}){ // 二、判断 access_token 是可用的吗?
    return expires_in > Date.now();
    }; async requestAccessToken(){ // 三、发送请求 getAccessToken() 获取 access_token
    // 1. access_token 请求 url
    const url = `${accessToken}appid=${APPID}&secret=${APPSECRET}`; // 2. POST 请求 access_token 对象
    const objAccessToken = await promiseRequest({
    method: 'POST',
    url,
    json: true
    }); // 重写过期时间,提前 5 分钟刷新
    objAccessToken.expires_in = Date.now() - (7200 - 300)*1000;
    return objAccessToken;
    } saveAccessToken(objAccessToken){ // 四、保存 access_token 到文件
    return new Promise((resolve, reject)=>{ // 异步执行文件写完
    writeFile('./access_token.txt', JSON.stringify(objAccessToken), err=>{
    if(err){
    reject("Write Success.");
    }else{
    resolve('access_token 最新已保存');
    };
    });
    });
    }
    } module.exports = {
    async interfaceInit(){
    const wechat = new WeChat(); console.log('---- 先删除菜单 ----');
    const deleteRet = await deleteMenu(wechat);
    console.log(deleteRet); console.log('---- 再创建菜单 ----');
    const createRet = await createMenu(wechat, menu);
    console.log(createRet);
    }
    };

interfaceInit/menu.js