Trabalhando com JSON em requisições HTTP em node.js
Vamos supor que queremos enviar dados para o servidor, cadastrando um usuário. Enviamos da seguinte maneira:
{
"name: "Fulano",
"email": "fulano@exemple.com.br"
}
Nosso servidor de exemplo atualmente se encontra assim:
import http from 'node:http'
const users = []
const server = http.createServer((req, res) => {
const {method, url} = req
if (method === 'GET' && url === '/users'){
return res
.setHeader('Content-type', 'application/json') //envio de headers do back para o front
.end(JSON.stringify(users))
}
if (method === 'POST' && url === '/users'){
users.push({
id: 1,
name: 'Fulano',
email: 'fulano@exemple.com'
})
return res.writeHead(201).end()
}
return res.writeHead(404).end()
})
server.listen(3333)
Como estamos lidando com pedacinhos de código, ao lidar com JSON precisamos da informação completa, então aguardamos até que todos os pedacinhos sejam retornados e juntamos em um só. Para isso, adicionamos o seguinte código:
const buffers = []
for await (const chunk of req){
buffers.push(chunk)
}
const body = Buffer.concat(buffers).toString()
Explicando o código acima: imagine que você recebe um quebra-cabeças com várias peças, mas as peças são entregues em partes, uma de cada vez. O objetivo é montar o quebra-cabeças completo, mas você precisa primeiro coletar todas as partes antes.
Agora, vamos associar cada parte do quebra-cabeças a um pedaço de dados da solicitação HTTP:
- A matriz
buffers
é como uma mesa vazia onde você irá colocar as partes do quebra-cabeças (pedaços de dados). - O loop
for await
representa o processo de receber uma peça do quebra-cabeças de cada vez. Ele itera assincronamente pelos pedaços de dados da solicitação HTTP. - O comando
buffers.push
(chunk) é como colocar cada pedaço de dados recebido na mesa, ou seja, na matriz buffers. Cada pedaço é adicionado ao final da matriz. - Após o loop ser concluído e todas as partes (pedaços de dados) terem sido recebidas e colocadas na mesa, você tem todas as peças necessárias para montar o quebra-cabeças.
- O
Buffer.concat(buffers)
é como pegar todas as peças da mesa e juntar tudo em um único quebra-cabeças completo. - Por fim,
toString()
é como olhar para o quebra-cabeças completo e descrevê-lo em forma de texto, permitindo que você o compreenda e manipule mais facilmente.
Pronto! Já temos o nosso body
(corpo) da requisição! Mas isso não é só, pois como vimos, o body
está como texto. Se se tentarmos acessar uma propriedade, como, por exemplo,body.name
, vai retornar como undefined
Para corrigirmos esse problema, usamos um carinha muito legal chamado JSON:
const body = JSON.parse(Buffer.concat(buffers).toString())
O método JSON.parse()
analisa uma string JSON e a transforma em um objeto JavaScript. Agora quando acessamos body.name
é nos retornado Fulano
Agora podemos trocar a parte do nosso método POST:
if (method === 'POST' && url === '/users'){
const {name, email} = body
users.push({
id: 1,
name,
email,
})
return res.writeHead(201).end()
}
Agora, ao fazer uma requisição enviando os dados do usuário, agora teremos como pegar corretamente esses dados.
Porém, quando fizermos uma requisição com o método GET, não enviando nada para o body, teremos erro. Vamos contornar isso com um trycatch e adicionando o body
no nosso req:
try {
req.body = JSON.parse(Buffer.concat(buffers).toString())
} catch (error) {
req.body = null
}
E modificamos o código aqui também para pegar o corpo da requisição:
const {name, email} = req.body
Prontinho! Agora podemos lidar com JSON em nossas requisições. Até a próxima!
Show de bola!