Hand-write a Nodejs program that imitates WeChat login

Hand-write a Nodejs program that imitates WeChat login

[[357291]]

Preface

First, let’s take a look at a picture from WeChat’s open documentation:

The picture above clearly introduces the entire WeChat login process. The following is a summary of what is shown in the picture:

1. How to obtain the QR code

  1. After the user opens the login webpage, the login webpage backend requests authorization to log in from the WeChat development platform according to the WeChat OAuth2.0 protocol, and passes parameters such as AppID and AppSecrect that have been reviewed and approved in the WeChat development platform in advance;
  2. The WeChat development platform verifies the AppID and other parameters, and returns a QR code to the login webpage backend;
  3. Log in to the backend of the web page and transmit the QR code to the frontend for display;

2. WeChat client authorization login

  1. The user uses the WeChat client to scan the QR code and authorize login;
  2. The WeChat client binds the QR code-specific UID to the WeChat account and sends it to the WeChat development platform;
  3. The WeChat development platform verifies the binding data, calls the callback interface of the login webpage background, and sends the authorization temporary ticket code;

3. Web page background request data

  1. The code is received after logging into the backend of the webpage, indicating that the WeChat development platform agrees to the data request;
  2. Log in to the backend of the web page and request the WeChat development platform to exchange access_token based on the code parameter, plus AppID and AppSecret;
  3. The WeChat development platform verifies the parameters and returns access_token;
  4. After logging in to the webpage background and receiving the access_token, you can perform parameter analysis to obtain the user account data.

accomplish

After understanding the general principle, we will start to implement this logic. Because the WeChat development platform is not directly called, this is just a demonstration effect. You can also visit:

  1. https://www.maomin.club/qrcodelogin/

This is my online website experience. The following code is the main logic, which is easier to understand when combined with the online website experience.

  1. let http = require( "http" );
  2. let express = require( "express" );
  3. let qrcode = require( "qr-image" );
  4. let app = express();
  5. let path = require( "path" );
  6. let server = http.createServer(app);
  7. let url = require( "url" );
  8. let fs = require( "fs" );
  9. let UUID = require( "uuid-js" );
  10. let generateHTML = null ;
  11.  
  12. app.use(express. static ( "./public" ));
  13.  
  14. /*
  15. * Description: Read web page files to replace keywords, equivalent to a simple template
  16. * Params:
  17. * sessionID - generated uid
  18. * req - web page request
  19. * res - web page response
  20. * fileName - the path where the web page file is located
  21. */
  22. generateHTML = function (sessionID, req, res, fileName) {
  23. fs.readFile(fileName, "UTF-8" , function (err, data) {
  24. if (!err) {
  25. data = data.replace (/SESSION_UID/g, sessionID);
  26. res.writeHead(200, {
  27. "Content-Type" : "text/html; charset=UTF-8" ,
  28. });
  29. res.end (data);
  30. } else {
  31. console.log(err);
  32.  
  33. res.writeHead(404, {
  34. "Content-Type" : "text/html; charset=UTF-8" ,
  35. });
  36. res.end ();
  37. }
  38. });
  39. };
  40.  
  41. /*
  42. * Description: Write to JSON file
  43. * Params:
  44. * fileName - the path where the JSON file is located
  45. * uid - generated uid
  46. * writeData - JSON format data to be written
  47. *
  48. */
  49. let setJSONValue = function (fileName, uid, writeData) {
  50. let data = fs.readFileSync(fileName);
  51.  
  52. let users = JSON.parse(data.toString());
  53. let addFlag = true ;
  54. let delFlag = writeData === null ;
  55.  
  56. for (let i = 0; i < users.data.length; i++) {
  57. if (users.data[i].uid === uid) {
  58. addFlag = false ;
  59.  
  60. if (delFlag) {
  61. users.data.splice(i, 1);
  62. } else {
  63. users.data[i].status = writeData.status;
  64.  
  65. console.log(
  66. "writeJSON: " + JSON.stringify(users.data[i]) + " modified."  
  67. );
  68. }
  69. }
  70. }
  71.  
  72. if (addFlag) {
  73. users.data.push(writeData);
  74. console.log( "writeJSON: " + JSON.stringify(writeData) + "inserted." );
  75. }
  76.  
  77. // Synchronously write to the file
  78. let writeJSON = JSON.stringify(users);
  79. fs.writeFileSync(fileName, writeJSON);
  80. };
  81.  
  82. /*
  83. * Description: Read JSON file (to return data, select synchronous reading)
  84. * Params:
  85. * fileName - the path where the JSON file is located
  86. * uid - generated uid
  87. *
  88. */
  89. getJSONValue = function (fileName, uid) {
  90. let readData = null ;
  91.  
  92. // Read the file synchronously
  93. let data = fs.readFileSync(fileName);
  94.  
  95. let users = JSON.parse(data.toString());
  96.  
  97. for (let i = 0; i < users.data.length; i++) {
  98. if (users.data[i].uid === uid) {
  99. readData = JSON.stringify(users.data[i]);
  100. break;
  101. }
  102. }
  103.  
  104. return readData;
  105. };
  106.  
  107. // Display the homepage of the website
  108. app.get( "/" , function (req, res) {
  109. // Generate a unique ID
  110. let uid = UUID.create ();
  111. console.log( "uid: '" + uid + "' generated." );
  112. //Replace the UID keyword in the web page template
  113. generateHTML(uid, req, res, path. join (__dirname, "/views/main.html" ));
  114. });
  115.  
  116. // Generate a QR code image and display it
  117. app.get( "/qrcode" , function (req, res, next ) {
  118. let uid = url.parse(req.url, true ).query.uid;
  119.  
  120. try {
  121. if (typeof uid !== "undefined" ) {
  122. //Write the URL in the QR code, and it will automatically jump after WeChat scans it. The URL below is my website, https://www.maomin.club/qrcodelogin, you can change it to your own online website or local server. Add the following "/scanned?uid="  
  123. let jumpURL = "https://www.maomin.club/qrcodelogin/scanned?uid=" + uid;
  124. // Generate QR code ( size : image size, margin: border blank)
  125. let img = qrcode.image(jumpURL, { size : 6, margin: 2 });
  126. res.writeHead(200, { "Content-Type" : "image/png" });
  127. img.pipe(res);
  128. } else {
  129. res.writeHead(414, { "Content-Type" : "text/html" });
  130. res. end ( "<h1>414 Request-URI Too Large</h1>" );
  131. }
  132. } catch (e) {
  133. res.writeHead(414, { "Content-Type" : "text/html" });
  134. res. end ( "<h1>414 Request-URI Too Large</h1>" );
  135. }
  136. });
  137.  
  138. // Display the confirmation interface after the mobile phone scans
  139. app.get( "/scanned" , function (req, res) {
  140. let uid = url.parse(req.url, true ).query.uid;
  141.  
  142. if (typeof uid !== "undefined" ) {
  143. generateHTML(uid, req, res, path. join (__dirname, "/views/confirm.html" ));
  144.  
  145. console.log( "uid: '" + uid + "' scanned." );
  146.  
  147. // Get the data corresponding to uid in the JSON file and change its data status
  148. let jsonData = getJSONValue(path. join (__dirname, "/bin/data.json" ), uid);
  149.  
  150. if (jsonData === null ) {
  151. jsonData = {
  152. uid: uid,
  153. status: "scanned" ,
  154. name : "USER" ,
  155. };
  156. } else {
  157. jsonData = JSON.parse(jsonData);
  158. jsonData.status = "scanned" ;
  159. }
  160.  
  161. // Write to JSON file
  162. setJSONValue( path.join (__dirname, "/bin/data.json" ), uid, jsonData);
  163. } else {
  164. res.writeHead(414, { "Content-Type" : "text/html" });
  165. res. end ( "<h1>414 Request-URI Too Large</h1>" );
  166. }
  167. });
  168.  
  169. // Response to the confirmation interface operation
  170. app.get( "/confirmed" , function (req, res) {
  171. let uid = url.parse(req.url, true ).query.uid;
  172. let operate = url.parse(req.url, true ).query.operate;
  173.  
  174. if (typeof uid !== "undefined" ) {
  175. console.log( "uid: '" + uid + "' " + operate);
  176.  
  177. let jsonData = getJSONValue(path. join (__dirname, "/bin/data.json" ), uid);
  178. let status = operate === "confirm" ? "verified" : "canceled" ;
  179.  
  180. if (jsonData === null ) {
  181. jsonData = {
  182. uid: uid,
  183. status: status,
  184. name : "USER" ,
  185. };
  186. } else {
  187. jsonData = JSON.parse(jsonData);
  188. jsonData.status = status;
  189. }
  190.  
  191. setJSONValue( path.join (__dirname, "/bin/data.json" ), uid, jsonData);
  192.  
  193. if (status === "verified" ) {
  194. res.writeHead(200, { "Content-Type" : "text/html" });
  195. res.end ( "<h1 style='textAlign:center;'>Login successful!</h1>" );
  196. } else {
  197. res.writeHead(200, { "Content-Type" : "text/html" });
  198. res. end ( "<h1 style='textAlign:center;'>Canceled!</h1>" );
  199. }
  200. } else {
  201. res.writeHead(414, { "Content-Type" : "text/html" });
  202. res. end ( "<h1 style='textAlign:center;'>414 Request-URI Too Large</h1>" );
  203. }
  204. });
  205.  
  206. // Respond to continuous AJAX requests from the home page
  207. app.get( "/verified" , function (req, res) {
  208. let uid = url.parse(req.url, true ).query.uid;
  209.  
  210. // normal - nothing is triggered
  211. // scanned - scanned
  212. // canceled - canceled
  213. // verified - verified
  214. let dataStatus = {
  215. cmd: "normal" ,
  216. user : "" ,
  217. };
  218.  
  219. console.log( "uid: '" + uid + "' query ..." );
  220.  
  221. if (typeof uid !== "undefined" ) {
  222. let userData = getJSONValue(path. join (__dirname, "/bin/data.json" ), uid);
  223.  
  224. // Return JSON data for home page AJAX operation
  225. if (userData !== null ) {
  226. userData = JSON.parse(userData);
  227. dataStatus.cmd = userData.status;
  228. dataStatus.user = userData.name ;
  229. }
  230. }
  231.  
  232. res.end (JSON.stringify(dataStatus));
  233. });
  234.  
  235. server.listen(4000);
  236. console.log(
  237. "Express server listening on port %d in %s mode" ,
  238. server.address().port,
  239. app.settings.env
  240. );

Do you think the code is not complete enough? Why only a main logic code is given? Don't worry, the full code will be provided immediately. You can see the comments for code explanation! The following is the github URL. If you think it is useful to you, you are welcome to star~

  1. https://github.com/maomincoding/qrcodelogin.git

Conclusion

After reading this, you may directly pull the code and find that the project cannot run? The effect is not the same as the online website. It is like this. If you have an online server, you can deploy it to the cloud. If you don’t have an online server, you can build a local LAN server yourself. Make sure that the mobile phone and the computer web page are on the same IP segment.

The effect diagram is as follows:

Login web page


Login authorization page


<<:  From DoH to ODoH, operators can no longer hijack DNS

>>:  When to use 5G and when to use Wi-Fi 6

Recommend

HTTP 2.0 is a bit explosive!

[[420793]] Hey guys, hello everyone, this is prog...

5G, edge computing and IoT are expected to reshape networks

5G provides wireless cellular connectivity with h...

CNNIC releases the 49th "Statistical Report on Internet Development in China"

The China Internet Network Information Center (CN...

Communications man, what on earth have you done to Everest?

Do you still remember the 5G "cloud supervis...

Five ways edge computing drives digital business

Every industry has created a new normal: if your ...

Common ways to manage networks through AIOps

NetOps teams in enterprises are faced with the ch...

What changes will 6G bring by 2030?

We are in the midst of a great digital wave. Inno...