preface
We know that the H5 page often needs to redirect users to the APP by downloading the installation package or jumping to the APP Store/APP Store. However, when the applet nested webView, it needed to verify the domain name, so jumping to the third-party app market and Appstroe could not achieve the diversion. So what to do? It can only be said that the devil is higher than a foot, look at the micro blog small program is how to guide the flow:
Curve save the country way, the use of small program online function can be opened H5 way, to download guidance. So, it leads to the topic of this document, small program online customer service automatic reply function. 😆
Before reading this document, it is best to know the official documents related to small program customer service information:
1. Customer service message usage guide
2. Small program customer service message server interface
3. Customer service message development documents
This development to do online customer service functions also stepped on a lot of pits, online access to a lot of information, but most of the background is based on PHP or Python, Java development, node.js development is less, so the development of the process record, for everyone’s reference, to avoid everyone stepped on pits. There may be some mistakes welcome correction communication. In addition, the Node framework we use is based on the KOA self-encapsulation, some details will be different from other frameworks, don’t bother.
Requirements describe
Click the button in the mini program to jump to the online customer service interface, and automatically reply according to the keyword. CMS can be configured with key and respond.
type | content |
---|---|
text | Text = Text reply content |
link | Title = Title description= Description URL = jump link thumb_url= image address |
image | Imageurl = Image address |
- After the configuration, the user can receive an automatic reply only when the reply condition is exactly matched
- Multiple keys can be configured and respond
- The default automatic reply can be configured for replies other than the configured key
The development process
Write a button to redirect customer service
index.wxml
<button open-type="contact"</button>Copy the code
The background configuration
After logging in to the background of the small program, in the “Development” – “Development Settings” – “Message Push” area, the administrator scans the code to enable the message service and fills in the server address (URL), Token (Token), and message encryption key (EncodingAESKey).
- URL Server address
URL: The URL of the interface used by the developer to receive wechat messages and events. The URL entered by the developer must start with http:// or https://, supporting port 80 and port 443 respectively.
Be sure to remember that the server address must be online, because wechat server is required to access. No localhost, no IP, no Intranet address.
Otherwise, the system displays “Parsing failed. Please check whether the information is correct”.
So the problem comes, different companies have a set of on-line process, always can not be in order to debug whether the URL is available to go up to the online test, the cost is too large, also not convenient.
This leads to Intranet penetration, which is simply to configure an online domain name, but this domain name can be penetrated to your configured local development address, so that you can easily debug to see the log. A tool that can penetrate the Intranet is recommended. (Non-advertising 😆)
NATAPP is not detailed in order to avoid advertising suspicion.
To put it simply, NATAPP has two modes: free and paid. The free one is the domain name changing from time to time. It is a bit extravagant for wechat’s push message configuration, which only has three chances to change in a month. The domain name may not be accessible at any time and must be reconfigured. The paid domain name is fixed, and the mapped Intranet address can be changed at any time. The owner of the building from free to paid mode, a month of VIP use about ten dollars.
2.Token
Token can be written by itself, but remember it because you will use it in the interface.
3.EncodingAESKey
Random generation.
4. Encryption mode and data format
Choose the safe mode and JSON format according to your preference. Different schemas and data formats will vary in development, so measure for yourself. Now that these configurations are clear, the start code.
Verify that the message is indeed from the wechat server
Before submitting the configuration, you need to write the interface for verifying messages from the wechat server.
server.js
https://developers.weixin.qq.com/miniprogram/dev/framework/server-ability/message-push.html / * * * * validation message from WeChat indeed server The developer verifies the request with the validation signature (see below). * If you confirm that the GET request is from the wechat server, please return the echostr parameter content as is. * Then the access takes effect and the developer succeeds. Otherwise, the access fails. The encryption/verification process is as follows: * Lexicographical sorting of token, timestamp and nonce parameters * sha1 encryption is performed by stitching the three parameter strings together * the encrypted string can be compared with signature. */ const crypto = require('crypto');
async wxCallbackAction(){ const ctx = this.ctx; const method = ctx.method; // verify the signature of the wechat server to confirm that the request is from wechatif(method === 'GET'{signature, timestamp, nonce, echostr const {signature, timestamp, nonce, echostr} = ctx.query; // 2. Sort the token, timestamp, and nonce parameters lexicographicallylet array = ['yourToken', timestamp, nonce]; array.sort(); Const tempStr = array.join(const tempStr = array.join(const tempStr = array.join(' ');
const hashCode = crypto.createHash('sha1'); // Create an encryption type const resultCode =hashCode.update(tempStr, 'utf8').digest('hex'); // 4. The developer obtains an encrypted string that can be compared with signature to indicate that the request is from wechatif (resultCode === signature) {
console.log('Verification successful, message forwarded from wechat server');
return this.json(echostr);
}else {
console.log('Verification failed!! ');
return this.json({
status: -1,
message: "Verification failed"}); }}}Copy the code
Verify interface development is complete, background configuration can go to click submit. The following message is displayed if the configuration is successful:
Receive and push messages
When users send messages during customer service sessions or events are pushed due to certain user actions, the wechat server will send the packets of messages or events to the URL filled in by the developer. Developers can respond asynchronously to requests using the send customer message interface.
This paper takes receiving text messages as an example to develop:
server.js
const WXDecryptContact = require('./WXDecryptContact');
async wxCallbackAction(){ const ctx = this.ctx; const method = ctx.method; // The message is received as a POST request; (The complete code is merged with the validation above)if(method === 'POST'){ const { Encrypt } = ctx.request.body; // The security mode selected during configuration therefore needs to be decryptedif(! Encrypt){return this.json('success');
}
const decryptData = WXDecryptContact(Encrypt);
await this._handleWxMsg(decryptData);
return this.json('success'); }} async _handleWxMsg(msgJson){if(! msgJson){return ;
}
const { MsgType } = msgJson;
if(MsgType === 'text'){ await this._sendTextMessage(msgJson); }} // async _sendTextMessage(msgJson){const result = await this.callService('cms.getDataByName'.'wxApplet.contact');
letkeyWordObj = result.data || {}; // Return to default by defaultlet options = keyWordObj.default;
for(let key inKeyWordObj){// Check if the configured keyword is matchedif(msgjson. Content === key){//CMS configuration item options = keyWordObj[key]; Access_token const accessToken = await this._getAccesstoken (); /* * Check whether the message type is image * if so, you need to upload the image file through the new material interface to obtain media_id */let media_id = ' ';
if(options.type === 'image'){// Get the image address (relative path)leturl = options.url; const file = fs.createReadStream(url); Const mediaResult = await this.callService()'wxApplet.uploadTempMedia',
{
access_token: accessToken,
type: 'image'
},
{
media: file
}
);
if(mediaResult.status === 0){
media_id = mediaResult.data.media_id;
}else{// If the image id fails to get, the default processing is used. }} // reply message to user const sendMsgResult = await this.callService('wxApplet.sendMessageToCustomer',
{
access_token: accessToken,
touser: msgJson.FromUserName,
msgtype: options.type || 'text',
text: {
content: options.description || ' ',
},
link: options.type === "link" ?
{
title: options.title,
description: options.description,
url: options.url,
thumb_url: options.thumb_url
}
:
{},
image: {
media_id
}
}
);
}
Copy the code
service.js
const request = require('request'); /* * Get CMS CMS keyword reply configuration * This interface is only for returning CMS configuration fields reply keyword configuration return data structure as follows */ asynccontact() {return {
data: {
"1": {
"type": "link"."title": "Click to download [****]APP"."description": "Register and receive $*** Registered Red Envelope Gift"."url": "https://m.renrendai.com/mo/***.html"."thumb_url": "https://m.we.com/***/test.png"
},
"2": {
"url": "http://m.renrendai.com/cms/****/test.jpg"."type": "image"
},
"3": {
"url": "/cms/***/test02.png"."type": "image"
},
"default": {
"type": "text"."description": "Goodbye"}}}} /* * Upload media files to wechat server. Currently, only images are supported. Used to send customer service messages or passively reply to user messages. * https://developers.weixin.qq.com/miniprogram/dev/api-backend/open-api/customer-message/customerServiceMessage.uploadTemp Media.html */ async uploadTempMedia(data,formData){ const url = `https://api.weixin.qq.com/cgi-bin/media/upload?access_token=${data.access_token}&type=${data.type}`;
return new Promise((resolve, reject) => {
request.post({url, formData: formData}, (err, response, body) => {
try{
const out = JSON.parse(body);
let result = {
data: out,
status: 0,
message: "ok"
}
return resolve(result);
}catch(err){
returnreject({ status: -1, message: err.message }); }}); }} /* * Send a customer service message to the user * https://developers.weixin.qq.com/miniprogram/dev/api-backend/open-api/customer-message/customerServiceMessage.send.html */ async sendMessageToCustomer(data){ const url = `https://api.weixin.qq.com/cgi-bin/message/custom/send?access_token=${data.access_token}`;
returnnew Promise((resolve, reject) => { request.post({url, data}, (err, response, body) => { ... }); }}Copy the code
WXDecryptContact.js
Message encryption decrypts documents
const crypto = require('crypto'); // Const decodePKCS7 =function (buff) {
let pad = buff[buff.length - 1];
if (pad < 1 || pad > 32) {
pad = 0;
}
returnbuff.slice(0, buff.length - pad); }; Const decryptContact = (key, iv, crypted) => {const aesCipher = crypto.createDecipheriv('aes-256-cbc', key, iv);
aesCipher.setAutoPadding(false);
let decipheredBuff = Buffer.concat([aesCipher.update(crypted, 'base64'), aesCipher.final()]);
decipheredBuff = decodePKCS7(decipheredBuff);
const lenNetOrderCorpid = decipheredBuff.slice(16);
const msgLen = lenNetOrderCorpid.slice(0, 4).readUInt32BE(0);
const result = lenNetOrderCorpid.slice(4, msgLen + 4).toString();
returnresult; }; // Decrypt the message returned by wechat to the configured message server const decryptWXContact = (wechatData) => {if(! wechatData){ wechatData =' '; } //EncodingAESKey is a const key generated randomly during background configuration = buffer. from(EncodingAESKey +)'='.'base64');
const iv = key.slice(0, 16);
const result = decryptContact(key, iv, wechatData);
const decryptedResult = JSON.parse(result);
console.log(decryptedResult);
return decryptedResult;
};
module.exports = decryptWXContact;
Copy the code
Finally the code is finished, let’s see the effect:
conclusion
The development has not been smooth sailing, and there have been some notable potholes, to emphasize:
- Configure the URL in the background. The URL can be accessed from the Internet (through Intranet penetration).
- File upload interface
uploadTempMedia
media
The parameters are in FormData format (node’srequest
Libraries are easy to implement.urllib
T_T: This library is full of tears. - Remember to return the received message with or without success
success
Otherwise, IOS displays a message indicating that the service provided by the small program is faulty. Please try again later.
The resources
Koa is connected to wechat mini program customer service message
The request documents
Renrendai Big Front-end Technology Blog Center
And finally broadcast it. Welcome to renrendai Front End Technology Blog Center
Nodejs React reactNative applet front-end engineering and other related technical articles
Wechat small program to step on the pit guide