fetch()
te permite realizar solicitudes de red similares a XMLHttpRequest (XHR). El
La principal diferencia es que la API de Fetch usa promesas, que tiene un
API más sencilla para evitar las complicadas devoluciones de llamadas en el
API de XMLHttpRequest.
Si nunca usaste Promises, consulta Introducción a las promesas de JavaScript.
Solicitud de recuperación básica
Este es un ejemplo implementado con un elemento XMLHttpRequest
y, luego, con fetch
. Queremos solicitar una URL, obtener una respuesta y analizarla como JSON.
XMLHttpRequest
Un XMLHttpRequest
necesita dos objetos de escucha para controlar los casos de éxito y error, y una llamada a open()
y send()
.
Ejemplo de los documentos de MDN.
function reqListener () {
const data = JSON.parse(this.responseText);
console.log(data);
}
function reqError (err) {
console.log('Fetch Error :-S', err);
}
const oReq = new XMLHttpRequest();
oReq.onload = reqListener;
oReq.onerror = reqError;
oReq.open('get', './api/some.json', true);
oReq.send();
Recuperar
Nuestra solicitud de recuperación se ve de la siguiente manera:
fetch('./api/some.json')
.then(response => {
if (response.status !== 200) {
console.log(`Looks like there was a problem. Status Code: ${response.status}`);
return;
}
// Examine the text in the response
response.json().then(function(data) {
console.log(data);
});
})
.catch(err => {
console.log('Fetch Error :-S', err);
});
La solicitud fetch()
solo necesita una llamada para realizar el mismo trabajo que el XHR.
ejemplo. Para procesar la respuesta, primero verificamos que el estado de la respuesta sea
200 y, luego, analiza la respuesta como JSON. La respuesta a una solicitud fetch()
es un objeto Stream, lo que significa que, después de llamar al método json()
, se muestra una promesa.
La transmisión se realiza de forma asíncrona.
Metadatos de la respuesta
En el ejemplo anterior, se mostró el estado del objeto Response y cómo analizar la respuesta como JSON. A continuación, te mostramos cómo controlar otros metadatos a los que podrías querer acceder, como los encabezados:
fetch('users.json').then(response => {
console.log(response.headers.get('Content-Type'));
console.log(response.headers.get('Date'));
console.log(response.status);
console.log(response.statusText);
console.log(response.type);
console.log(response.url);
});
Tipos de respuesta
Cuando hacemos una solicitud de recuperación, la respuesta recibirá un response.type
de "basic
", "cors
" o "opaque
". Estos types
muestran de dónde proviene el recurso y puedes usarlos para determinar cómo tratar el objeto de respuesta.
Cuando el navegador solicita un recurso en el mismo origen, la respuesta tiene un tipo basic
con restricciones sobre lo que puedes ver desde la respuesta.
Si se realiza una solicitud para un recurso en otro origen y ese origen muestra encabezados de CORS, el tipo es cors
. cors
son similares a las respuestas de basic
, pero restringen los encabezados que
puede ver como Cache-Control
, Content-Language
, Content-Type
, Expires
,
Last-Modified
y Pragma
.
Las respuestas de opaque
provienen de un origen diferente que no muestra encabezados de CORS. Con una respuesta opaca, no podremos
leer los datos devueltos o ver el estado de la solicitud, lo que significa que no puedes verificar
si la solicitud fue exitosa.
Puedes definir un modo para una solicitud de recuperación para que solo se resuelvan ciertos tipos de solicitudes. Los modos que puedes configurar son los siguientes:
same-origin
solo se realiza correctamente para las solicitudes de recursos en el mismo origen y rechaza todas las demás solicitudes.cors
permite solicitudes de activos del mismo origen y de otros orígenes que devuelve los encabezados COR correspondientes.cors-with-forced-preflight
realiza una comprobación previa verificación antes de realizar cualquier solicitud.no-cors
está diseñado para realizar solicitudes a otros orígenes que no tengan CORS. y dan como resultado una respuestaopaque
. posible en el permiso global de la ventana en este momento.
Para definir el modo, agrega un objeto de opciones como segundo parámetro en la solicitud fetch
y define el modo en ese objeto:
fetch('http://some-site.com/cors-enabled/some.json', {mode: 'cors'})
.then(response => response.text())
.then(text => {
console.log('Request successful', text);
})
.catch(error => {
log('Request failed', error)
});
Cadena de promesas
Una de las mejores características de las promesas es la capacidad de encadenarlas. En el caso de fetch()
, esto te permite compartir la lógica entre las solicitudes de recuperación.
Si trabajas con una API de JSON, debes verificar el estado y analizar el JSON de cada respuesta. Para simplificar tu código, define el estado y el análisis de JSON en funciones separadas que devuelvan promesas y usa la solicitud de recuperación para controlar solo los datos finales y el caso de error.
function status (response) {
if (response.status >= 200 && response.status < 300) {
return Promise.resolve(response)
} else {
return Promise.reject(new Error(response.statusText))
}
}
function json (response) {
return response.json()
}
fetch('users.json')
.then(status)
.then(json)
.then(data => {
console.log('Request succeeded with JSON response', data);
}).catch(error => {
console.log('Request failed', error);
});
En este ejemplo, se define una función status
que verifica response.status
y
devuelve una promesa resuelta como
Promise.resolve()
,
o una promesa rechazada, ya que
Promise.reject()
.
Este es el primer método al que se llama en la cadena fetch()
.
Si se resuelve la promesa, la secuencia de comandos llama al método json()
, que
devuelve una segunda promesa a partir de la llamada a response.json()
y crea una
objeto que contenga el JSON analizado. Si el análisis falla, se rechaza la promesa y se ejecuta la sentencia catch.
Esta estructura te permite compartir la lógica en todas tus solicitudes de recuperación, lo que facilita el mantenimiento, la lectura y la prueba del código.
Solicitud POST
A veces, una app web necesita llamar a una API con un método POST y, luego, incluir algunos parámetros en el cuerpo de la solicitud. Para ello, configura method
y body
en las opciones fetch()
:
fetch(url, {
method: 'post',
headers: {
"Content-type": "application/x-www-form-urlencoded; charset=UTF-8"
},
body: 'foo=bar&lorem=ipsum'
})
.then(json)
.then(data => {
console.log('Request succeeded with JSON response', data);
})
.catch(error => {
console.log('Request failed', error);
});
Envía credenciales con una solicitud de recuperación
Para realizar una solicitud de recuperación con credenciales, como cookies, establece el valor credentials
de la solicitud en "include"
:
fetch(url, {
credentials: 'include'
})