Implementa el manejo de errores cuando se usa la API de recuperación

En este artículo, se muestran algunos enfoques de control de errores cuando se trabaja con la API de Fetch. La API de Fetch te permite realizar una solicitud a un recurso de red remoto. Cuando haces una llamada de red remota, tu página web queda sujeta a una variedad de posibles errores de red.

En las siguientes secciones, se describen posibles errores y cómo escribir código que proporcione un nivel razonable de funcionalidad que sea resistente a errores y condiciones de red inesperadas. El código resiliente mantiene a los usuarios contentos y mantiene un nivel de servicio estándar para tu sitio web.

Anticípate a posibles errores de red

En esta sección, se describe una situación en la que el usuario crea un video nuevo llamado "My Travels.mp4" y, luego, intenta subirlo a un sitio web para compartir videos.

Cuando se trabaja con Fetch, es fácil considerar la ruta ideal en la que el usuario sube el video correctamente. Sin embargo, hay otras rutas que no son tan sencillas, pero para las que los desarrolladores web deben planificar. Estas rutas (desagradables) pueden ocurrir debido a un error del usuario, a condiciones ambientales inesperadas o a un error en el sitio web de intercambio de videos.

Ejemplos de errores del usuario

  • El usuario sube un archivo de imagen (como JPEG) en lugar de un archivo de video.
  • El usuario comienza a subir el archivo de video incorrecto. Luego, a mitad de la carga, el usuario especifica el archivo de video correcto para subir.
  • El usuario hace clic accidentalmente en "Cancelar carga" mientras se sube el video.

Ejemplos de cambios ambientales

  • La conexión a Internet se desconecta mientras se sube el video.
  • El navegador se reinicia mientras se sube el video.
  • Los servidores del sitio web de uso compartido de videos se reinician mientras se sube el video.

Ejemplos de errores con el sitio web de uso compartido de videos

  • El sitio web para compartir videos no puede admitir un nombre de archivo con un espacio. En lugar de "My Travels.mp4", espera un nombre como "My_Travels.mp4" o "MyTravels.mp4".
  • El sitio web para compartir videos no puede subir un video que supere el tamaño máximo de archivo aceptable.
  • El sitio web para compartir videos no es compatible con el códec de video que se subió.

Estos ejemplos pueden ocurrir y ocurren en el mundo real. Es posible que hayas encontrado ejemplos de este tipo en el pasado. Elige un ejemplo de cada una de las categorías anteriores y analiza los siguientes puntos:

  • ¿Cuál es el comportamiento predeterminado si el servicio de uso compartido de videos no puede controlar el ejemplo dado?
  • ¿Qué espera que suceda el usuario en el ejemplo?
  • ¿Cómo podemos mejorar el proceso?
Acción El usuario comienza a subir el archivo de video incorrecto. Luego, a mitad de la carga, el usuario especifica el archivo de video correcto para subir.
Qué sucede de forma predeterminada El archivo original se seguirá subiendo en segundo plano mientras se sube el nuevo archivo al mismo tiempo.
Lo que espera el usuario El usuario espera que se detenga la carga original para que no se desperdicie el ancho de banda de Internet adicional.
Qué se puede mejorar JavaScript cancela la solicitud de recuperación del archivo original antes de que comience a subirse el nuevo.
Acción El usuario pierde la conexión a Internet mientras sube el video.
Qué sucede de forma predeterminada La barra de progreso de la carga parece estar detenida en el 50%. Con el tiempo, la API de Fetch experimenta un tiempo de espera y se descartan los datos subidos. Cuando se restablece la conexión a Internet, el usuario debe volver a subir el archivo.
Lo que espera el usuario El usuario espera recibir una notificación cuando no se puede subir su archivo y espera que la carga se reanude automáticamente al 50% cuando vuelva a estar en línea.
Qué se puede mejorar La página de carga le informa al usuario sobre los problemas de conectividad a Internet y le asegura que la carga se reanudará cuando se restablezca la conexión.
Acción El sitio web para compartir videos no puede admitir un nombre de archivo con un espacio. En lugar de "Mis viajes.mp4", espera nombres como "Mis_viajes.mp4" o "Misviajes.mp4".
Qué sucede de forma predeterminada El usuario debe esperar a que la carga termine por completo. Una vez que se sube el archivo y la barra de progreso muestra "100%", se muestra el mensaje "Vuelve a intentarlo".
Lo que espera el usuario El usuario espera que se le informe sobre las limitaciones de los nombres de archivo antes de que comience la carga o, al menos, durante el primer segundo de la carga.
Qué se puede mejorar Lo ideal es que el servicio de uso compartido de videos admita nombres de archivo con espacios. Las opciones alternativas son notificar al usuario sobre las limitaciones de los nombres de archivo antes de que comience la carga. O bien, el servicio de uso compartido de videos debe rechazar la carga con un mensaje de error detallado.

Controla los errores con la API de Fetch

Ten en cuenta que los siguientes ejemplos de código usan await de nivel superior (compatibilidad con navegadores) porque esta función puede simplificar tu código.

Cuando la API de Fetch genera errores

En este ejemplo, se usa una sentencia bloque try/catch para detectar cualquier error que se genere dentro del bloque try. Por ejemplo, si la API de Fetch no puede recuperar el recurso especificado, se genera un error. Dentro de un bloque catch como este, asegúrate de proporcionar una experiencia del usuario significativa. Si se le muestra al usuario un ícono giratorio, una interfaz de usuario común que representa algún tipo de progreso, puedes realizar las siguientes acciones dentro de un bloque catch:

  1. Quita el ícono giratorio de la página.
  2. Proporciona mensajes útiles que expliquen qué salió mal y qué opciones tiene el usuario.
  3. Según las opciones disponibles, presenta al usuario un botón "Reintentar".
  4. En segundo plano, envía los detalles del error a tu servicio de seguimiento de errores o al backend. Esta acción registra el error para que se pueda diagnosticar en una etapa posterior.
try {
 
const response = await fetch('https://website');
} catch (error) {
 
// TypeError: Failed to fetch
  console
.log('There was an error', error);
}

En una etapa posterior, mientras diagnosticas el error que registraste, puedes escribir un caso de prueba para detectarlo antes de que los usuarios sepan que algo está mal. Según el error, la prueba puede ser de unidad, integración o aceptación.

Cuando el código de estado de la red representa un error

En este ejemplo de código, se realiza una solicitud a un servicio de prueba HTTP que siempre responde con el código de estado HTTP 429 Too Many Requests. Curiosamente, la respuesta no llega al bloque catch. Un estado 404, entre otros códigos de estado, no muestra un error de red, sino que se resuelve de forma normal.

Para verificar que el código de estado HTTP se haya ejecutado correctamente, puedes usar cualquiera de las siguientes opciones:

  • Usa la propiedad Response.ok para determinar si el código de estado estaba en el rango de 200 a 299.
  • Usa la propiedad Response.status para determinar si la respuesta se realizó correctamente.
  • Usa cualquier otro metadato, como Response.headers, para evaluar si la respuesta se realizó correctamente.
let response;

try {
  response
= await fetch('https://httpbin.org/status/429');
} catch (error) {
  console
.log('There was an error', error);
}

// Uses the 'optional chaining' operator
if (response?.ok) {
  console
.log('Use the response here!');
} else {
  console
.log(`HTTP Response Code: ${response?.status}`)
}

La práctica recomendada es trabajar con las personas de tu organización y equipo para comprender los posibles códigos de estado de respuesta HTTP. Los desarrolladores de backend, las operaciones de desarrolladores y los ingenieros de servicios a veces pueden proporcionar estadísticas únicas sobre posibles casos extremos que quizás no preveas.

Cuando se produce un error al analizar la respuesta de la red

En este ejemplo de código, se muestra otro tipo de error que puede surgir cuando se analiza un cuerpo de respuesta. La interfaz Response ofrece métodos convenientes para analizar diferentes tipos de datos, como texto o JSON. En el siguiente código, se realiza una solicitud de red a un servicio de prueba HTTP que muestra una cadena HTML como cuerpo de la respuesta. Sin embargo, se intenta analizar el cuerpo de la respuesta como JSON, lo que genera un error.

let json;

try {
 
const response = await fetch('https://httpbin.org/html');
  json
= await response.json();
} catch (error) {
 
if (error instanceof SyntaxError) {
   
// Unexpected token < in JSON
    console
.log('There was a SyntaxError', error);
 
} else {
    console
.log('There was an error', error);
 
}
}

if (json) {
  console
.log('Use the JSON here!', json);
}

Debes preparar tu código para que admita una variedad de formatos de respuesta y verificar que una respuesta inesperada no dañe la página web del usuario.

Considera la siguiente situación: tienes un recurso remoto que muestra una respuesta JSON válida y se analiza correctamente con el método Response.json(). Es posible que el servicio deje de funcionar. Una vez que se baja, se muestra un 500 Internal Server Error. Si no se usan las técnicas de manejo de errores adecuadas durante el análisis de JSON, es posible que se dañe la página para el usuario porque se genera un error no controlado.

Cuando se debe cancelar la solicitud de red antes de que se complete

En este ejemplo de código, se usa un AbortController para cancelar una solicitud en curso. Una solicitud en curso es una solicitud de red que se inició, pero que no se completó.

Las situaciones en las que es posible que debas cancelar una solicitud en curso pueden variar, pero, en última instancia, depende de tu caso de uso y entorno. En el siguiente código, se muestra cómo pasar un AbortSignal a la API de Fetch. El AbortSignal está conectado a un AbortController, y el AbortController incluye un método abort(), que le indica al navegador que se debe cancelar la solicitud de red.

const controller = new AbortController();
const signal = controller.signal;

// Cancel the fetch request in 500ms
setTimeout
(() => controller.abort(), 500);

try {
 
const url = 'https://httpbin.org/delay/1';
 
const response = await fetch(url, { signal });
  console
.log(response);
} catch (error) {
 
// DOMException: The user aborted a request.
  console
.log('Error: ', error)
}

Conclusión

Un aspecto importante de la administración de errores es definir las diferentes partes que pueden fallar. Para cada situación, asegúrate de tener un resguardo adecuado para el usuario. En relación con una solicitud de recuperación, hazte preguntas como las siguientes:

  • ¿Qué sucede si se cae el servidor de destino?
  • ¿Qué sucede si Fetch recibe una respuesta inesperada?
  • ¿Qué sucede si falla la conexión a Internet del usuario?

Según la complejidad de tu página web, también puedes esbozar un diagrama de flujo que describa la funcionalidad y la interfaz de usuario para diferentes situaciones.