Introducing a request library - How to use several Undici APIs

Introducing a request library - How to use several Undici APIs

[[430223]]

Preface

In the browser, if we want to make a request, we used to use xhr, but this low-level API is often called in a simple way. In order to improve development efficiency, jQuery's $.ajax may be the best choice. Fortunately, a more modern fetch api appeared later.

However, considering the compatibility of fetch, and the fact that it does not support some global configurations and request interruption, in actual use, we may use the axios request library to make some requests. In Node.js, almost all requests are made through the request library. Unfortunately, request stopped maintenance two years ago, and it is not easy to find a replacement library in Node.js.

In the request issues[1], there is a table that recommends some commonly used request libraries in Node.js:

Axios, which is widely used in browsers, is not easy to use in Node.js, especially when uploading files, there will be many unexpected problems.

Recently, when I was surfing the Internet, I found that Node.js officially has a request library: undici. The name is quite complicated. So, today's article will introduce undici. By the way, undici means 11 in Italian. It seems that Double 11 is coming soon, which is good for Moutai.

  • Undici means eleven in Italian. 1.1 -> 11 -> Eleven -> Undici. It is also a Stranger Things reference.

Get Started

We can install undici directly through npm:

  1. npm install undici -S

undici exposes an object to the outside world, which provides several APIs:

  • undici.fetch: initiates a request, which is consistent with the fetch method in the browser;
  • undici.request: initiates a request, which is somewhat similar to the request library. This method supports Promise;
  • undici.stream: processes file streams and can be used to download files;

undici.fetch

Note: This method requires node version >= v16.5.0

Before requesting the service through undici.fetch, you need to start a simple login service through koa.

  1. const Koa = require( 'koa' )
  2. const bodyParser = require( 'koa-bodyparser' )
  3.  
  4. const app = new Koa()
  5.  
  6. app.use(bodyParser())
  7. app.use(ctx => {
  8. const { url, method, body } = ctx.request
  9. if (url === '/login' ) {
  10. if (method === 'POST' ) {
  11. if (body.account === 'shenfq' && body. password === '123456' ) {
  12. ctx.body = JSON.stringify({
  13. name : 'shenfq' ,
  14. mobile: '130xxxxxx'  
  15. })
  16. return  
  17. }
  18. }
  19. }
  20. ctx.status = 404
  21. ctx.body = JSON.stringify({})
  22. })
  23.  
  24. app.listen(3100)

The above code is very simple and only supports accepting a POST method to the /login route. Next, we use undici.fetch to initiate a POST request.

  1. const { fetch } = require( 'undici' )
  2.  
  3. const bootstrap = async () => {
  4. const api = 'http://localhost:3100/login'  
  5. const rsp = await fetch (api, {
  6. method: 'POST' ,
  7. headers: {
  8. 'content-type' : 'application/json'  
  9. },
  10. body: JSON.stringify({
  11. account: 'shenfq' ,
  12. password : '123456'  
  13. })
  14. })
  15. if (rsp.status !== 200) {
  16. console.log(rsp.status, 'Request failed' )
  17. return  
  18. }
  19. const json = await rsp.json()
  20. console.log(rsp.status, json)
  21. }
  22.  
  23. bootstrap()

If the request method is changed to GET, 404 will be returned.

  1. const rsp = await fetch (api, {
  2. method: 'GET'  
  3. })

undici.request

The calling method of undici.request is similar to undici.fetch, and the parameter passing format is also similar .

  1. const { request } = require( 'undici' )
  2.  
  3. const bootstrap = async () => {
  4. const api = 'http://localhost:3100/login'  
  5. const { body, statusCode } = await request(api, {
  6. method: 'POST' ,
  7. headers: {
  8. 'content-type' : 'application/json'  
  9. },
  10. body: JSON.stringify({
  11. account: 'shenfq' ,
  12. password : '123456'  
  13. })
  14. })
  15. const json = await body.json()
  16. console.log(statusCode, json)
  17. }
  18.  
  19. bootstrap()

The only difference is that the returned result is a little different. The http response result returned by the request method is in the body attribute, and this attribute also supports methods such as .json()/.text() similar to fetch.

Interrupt Request

Install the abort-controller library, then instantiate the abort-controller and pass the interrupt signal into the request configuration.

  1. npm i abort-controller

  1. const undici = require( 'undici' )
  2. const AbortController = require( 'abort-controller' )
  3.  
  4. // Instantiate abort-controller
  5. const abortController = new AbortController()
  6. undici.request( 'http://127.0.0.1:3100' , {
  7. method: 'GET' ,
  8. //Incoming interrupt signal
  9. signal: abortController.signal,
  10. }). then (({ statusCode, body }) => {
  11. body. on ( 'data' , (data) => {
  12. console.log(statusCode, data.toString())
  13. })
  14. })

We run the code and find that the request is successful because we did not actively call the interrupt method.

  1. undici.request( 'http://127.0.0.1:3100' , {
  2. method: 'GET' ,
  3. signal: abortController.signal,
  4. }). then (({ statusCode, body }) => {
  5. console.log( 'Request successful' )
  6. body. on ( 'data' , (data) => {
  7. console.log(statusCode, data.toString())
  8. })
  9. }).catch(error => {
  10. // Capture errors triggered by interrupts
  11. console.log( 'error' , error. name )
  12. })
  13.  
  14. //Call interrupt
  15. abortController.abort()

Now when you run the code, you will find that there is no log of successful request output. The catch logic is entered and the request is successfully interrupted.

undici.steam

The undici.steam method can be used to download files or interface proxy.

Download the file

  1. const fs = require( 'fs' )
  2. const { stream } = require( 'undici' )
  3.  
  4. const out = fs.createWriteStream( './Song Dynasty-Ge Kiln-Gold Wire and Iron Wire.jpg' )
  5. const url = 'https://img.dpm.org.cn/Uploads/Picture/dc/cegift/cegift6389.jpg'  
  6.  
  7. stream(url, { opaque: out }, ({ opaque }) => opaque)

Interface Proxy

  1. const http = require( 'http' )
  2. const undici = require( 'undici' )
  3.  
  4. // Proxy requests from port 3100 to port 80
  5. const client = new undici.Client( 'http://localhost' )
  6. http.createServer((req, res) => {
  7. const { url, method } = req
  8. client.stream(
  9. { method, path: url,opaque: res },
  10. ({ opaque }) => opaque
  11. )
  12. }).listen(3100)

Summarize

This article only introduces the usage of several undici APIs. It seems that the ease of using undici is still relatively low. However, the compatibility is not very good. For example, fetch only supports [email protected] and above.

For this kind of relatively new library, I personally recommend waiting for a while. Although request has been abandoned, we still use some time-tested libraries, such as urllib[7] used in the egg framework, and node-fetch[8], which is relatively easy to use and is used in the same way as the fetch API in the browser.

References

[1]issues: https://github.com/request/request/issues/3143

[2]node-fetch: https://www.npmjs.com/package/node-fetch

[3]got: https://www.npmjs.com/package/got

[4]axios: https://www.npmjs.com/package/axios

[5]superagent: https://www.npmjs.com/package/superagent

[6]urllib: https://www.npmjs.com/package/urllib

[7]urllib: https://www.npmjs.com/package/urllib

[8]node-fetch: https://www.npmjs.com/package/node-fetch

<<:  Our company’s “Double 11” flow control plan, come and copy our homework!

>>:  China Mobile: 5G package customers reach 331 million

Recommend

Potential application scenarios of 6G in the future

Although 6G is not yet a viable technology, it wi...

How to build supply chain finance modernization from Shengye

The report of the 20th National Congress of the C...

Accelerating the development of edge computing

5G is being rolled out faster than any previous g...

Transforming the Enterprise with 5G Technology

For years, people have been talking about the tra...

What is 5G network slicing?

In today's fast-paced, hyper-connected world,...

What impact will 5G have on us?

1. First of all, it is clear that fiber optic tra...

...

Ten major trends in the future of industrial Internet

In recent years, major countries around the world...

10g.biz Los Angeles CN2 VPS simple test

Yesterday we tested the 10g.biz Hong Kong VPS, to...