Usar Service Workers
Este artículo brinda información sobre cómo comenzar con el service worker, incluida la arquitectura básica, el registro de un service worker, el proceso de instalación y activación de un nuevo service worker, la actualización de tu service worker, el control de caché y las respuestas personalizadas, todo en el contexto de una aplicación simple, con funcionalidad fuera de línea.
La premisa del service worker
Un problema primordial del que los usuarios de la web han adolecido durante años es la pérdida de conectividad. La mejor aplicación web del mundo proporcionará una experiencia de usuario terrible si no la puedes descargar. Ha habido varios intentos de crear tecnologías para resolver este problema, y algunos de los problemas se han resuelto. Pero el problema primordial es que todavía no existe un buen mecanismo de control general para el almacenamiento en caché de activos y las solicitudes de red personalizadas.
El intento anterior, AppCache, parecía ser una buena idea porque te permitía especificar activos para almacenar en caché con mucha facilidad. Sin embargo, hizo muchas suposiciones sobre lo que estabas tratando de hacer y luego se rompió horriblemente cuando tu aplicación no siguió exactamente esas suposiciones. Lee el documento de Jake Archibald (desafortunadamente mal titulado pero bien escrito) Application Cache is a Douchebag para obtener más detalles.
Nota: A partir de Firefox 84, se eliminó AppCache (Error 1619673 en Firefox). También se ha eliminado de Chromium 95 y está obsoleto en Safari.
El service worker finalmente debería solucionar estos problemas. La sintaxis del service worker es más compleja que la de AppCache, pero la compensación es que puedes usar JavaScript para controlar su comportamiento implícito en AppCache con un buen grado de fina granularidad, lo que te permite manejar este problema y muchos más. Al usar un service worker, puedes configurar fácilmente una aplicación para usar activos almacenados en caché primero, proporcionando así una experiencia predeterminada incluso cuando estás desconectado, antes de obtener más datos de la red (comúnmente conocido como Primero sin conexión). Esto ya está disponible con las aplicaciones nativas, que es una de las principales razones por las que las aplicaciones nativas a menudo se eligen en lugar de las aplicaciones web.
Configuración para jugar con el service worker
En estos días, el service worker está habilitado de forma predeterminada en todos los navegadores modernos. Para ejecutar código con el service worker, deberás entregar tu código a través de HTTPS: el service worker, por razones de seguridad, está restringido a ejecutarse a través de HTTPS. Por lo tanto, GitHub es un buen lugar para alojar experimentos, ya que admite HTTPS. Para facilitar el desarrollo local, los navegadores también consideran localhost
como un origen seguro.
Arquitectura básica
Con el service worker, generalmente se observan los siguientes pasos para la configuración básica:
- La URL del service worker se obtiene y registra a través de
serviceWorkerContainer.register()
. - Si tiene éxito, el service worker se ejecuta en
ServiceWorkerGlobalScope
; esto es básicamente un tipo especial de contexto de trabajo, que se ejecuta fuera del hilo principal de ejecución del script, sin acceso al DOM. - El service worker ahora está listo para procesar eventos.
- Se intenta la instalación del worker cuando se accede posteriormente a las páginas controladas por el service worker. Un evento de instalación siempre es el primero que se envía a un service worker (esto se puede usar para iniciar el proceso de completar una IndexedDB «base de datos indexada» y almacenar en caché los activos del sitio). Este es realmente el mismo tipo de procedimiento que instalar una aplicación nativa o Firefox OS: hace que todo esté disponible para usar sin conexión.
- Cuando se completa el controlador
oninstall
, se considera que el service worker está instalado. - Lo siguiente es la activación. Cuando se instala el service worker, recibe un evento de activación. El uso principal de
onactivate
es para la limpieza de los recursos utilizados en versiones anteriores de un script del service worker. - El service worker ahora controlará las páginas, pero solo aquellas que se abran después de que
register()
tenga éxito. En otras palabras, los documentos se deberán volver a cargar para controlarlos realmente, porque un documento comienza con o sin un service worker y lo mantiene durante toda su vida.
El siguiente gráfico muestra un resumen de los eventos de service worker disponibles:
Demostración del service worker
Para demostrar los conceptos básicos de registro e instalación de un service worker, hemos creado una demostración simple llamada service worker simple, que es una simple galería de imágenes de Star Wars Lego. Utiliza una función impulsada por promesas para leer datos de imagen de un objeto JSON y cargar las imágenes usando Ajax, antes de mostrar las imágenes en una línea hacia abajo en la página. Hemos mantenido las cosas estáticas y simples por ahora. También registra, instala y activa un service worker, y cuando los navegadores admiten más especificaciones, almacenará en caché todos los archivos necesarios para que funcione sin conexión.
Puedes ver el código fuente en GitHub y el Sencillo service worker ejecutándose en vivo.
Registra a tu worker
El primer bloque de código en el archivo JavaScript de nuestra aplicación, app.js
, es el siguiente. Este es nuestro punto de entrada en el uso del service worker.
const registerServiceWorker = async () => {
if ("serviceWorker" in navigator) {
try {
const registration = await navigator.serviceWorker.register("/sw.js", {
scope: "/",
});
if (registration.installing) {
console.log("Instalando el Service worker");
} else if (registration.waiting) {
console.log("Service worker instalado");
} else if (registration.active) {
console.log("Service worker activo");
}
} catch (error) {
console.error(`Falló el registro con el ${error}`);
}
}
};
// …
registerServiceWorker();
- El bloque if realiza una prueba de detección de características para asegurarse de que el service worker sea compatible antes de intentar registrar uno.
- A continuación, usamos la función
ServiceWorkerContainer.register()
para registrar el service worker para este sitio, que solo es un archivo JavaScript que reside dentro de nuestra aplicación (ten en cuenta que esta es la URL del archivo relativa al origen , no el archivo JS que hace referencia a él). - El parámetro
scope
es opcional y se puede usar para especificar el subconjunto de tu contenido que deseas controle el service worker. En este caso, hemos especificado'/'
, lo cual significa todo el contenido bajo el origen de la aplicación. Si lo omites, tendrá este valor predeterminado de todos modos, pero lo especificamos aquí con fines ilustrativos.
Esto registra un service worker, que se ejecuta en un contexto de trabajador y, por lo tanto, no tiene acceso al DOM. Luego ejecuta el código en el service worker fuera de tus páginas normales para controlar su carga.
Un solo service worker puede controlar muchas páginas. Cada vez que se carga una página dentro de su alcance, el service worker se instala en esa página y opera en ella. Por lo tanto, ten en cuenta que debes tener cuidado con las variables globales en el script del service worker: cada página no tiene su propio trabajador único.
Nota: Tu service worker funciona como un servidor proxy, lo que te permite modificar solicitudes y respuestas, reemplazarlas con elementos de su propio caché y más.
Nota: Una gran cosa acerca del service worker es que si usas la detección de funciones como se muestra arriba, los navegadores que no son compatibles con los service workers pueden usar tu aplicación en línea de la manera normal esperada. Además, si usas AppCache y SW en una página, los navegadores que no admiten SW pero sí AppCache lo usarán, y los navegadores que admiten ambos ignorarán AppCache y dejarán que SW tome el control.
¿Por qué mi service worker no se registra?
Esto se podría deber a las siguientes razones:
- No estás ejecutando tu aplicación a través de HTTPS.
- La ruta a tu archivo del service worker no está escrita correctamente — se debe escribir en relación con el origen, no con el directorio raíz de tu aplicación. En nuestro ejemplo, el trabajador está en
https://bncb2v.csb.app/sw.js
y la raíz de la aplicación eshttps://bncb2v.csb.app/
. Pero la ruta se debe escribir como/sw.js
. - Tampoco está permitido apuntar a un service worker de un origen diferente al de tu aplicación.
También ten en cuenta:
- El service worker solo capturará las solicitudes de los clientes bajo el alcance del service worker.
- El alcance máximo para un service worker es la ubicación del trabajador.
- Si tu service worker está activo en un cliente al que se atiende con el encabezado
Service-Worker-Allowed
, puedes especificar una lista de alcances máximos para ese trabajador. - En Firefox, las APIs de Service Worker están ocultas y no se pueden usar cuando el usuario está en modo de navegación privada.
Instalar y activar: llena tu caché
Después de que tu service worker esté registrado, el navegador intentará instalar y luego activar el service worker para tu página/sitio.
El evento install
se activa cuando una instalación se completa con éxito. El evento install
generalmente se usa para llenar las capacidades de almacenamiento en caché sin conexión de tu navegador con los activos que necesita para ejecutar tu aplicación sin conexión. Para hacer esto, usamos la API de almacenamiento de Service Worker: cache
— un objeto global en Service Worker que nos permite almacenar los activos entregados por las respuestas y con clave de sus solicitudes. Esta API funciona de manera similar a la memoria caché estándar del navegador, pero es específica para tu dominio. Persiste hasta que le dices que no lo haga — nuevamente, tienes el control total.
Así es como nuestro service worker maneja el evento install
:
const addResourcesToCache = async (resources) => {
const cache = await caches.open("v1");
await cache.addAll(resources);
};
self.addEventListener("install", (event) => {
event.waitUntil(
addResourcesToCache([
"/",
"/index.html",
"/style.css",
"/app.js",
"/image-list.js",
"/star-wars-logo.jpg",
"/gallery/bountyHunters.jpg",
"/gallery/myLittleVader.jpg",
"/gallery/snowTroopers.jpg",
]),
);
});
- Aquí agregamos un detector de eventos
install
al service worker (por lo tanto,self
), y luego encadenamos un métodoExtendableEvent.waitUntil()
al evento; esto garantiza que el service worker no se instale hasta que el código dentro dewaitUntil()
haya ocurrido con éxito. - Dentro de
addResourcesToCache
usamos el métodocaches.open()
para crear un nuevo caché llamadov1
, que será la versión 1 de nuestro caché de recursos del sitio. Luego llamamos a una función que llama aaddAll()
en el caché creado, que para su parámetro toma un arreglo de URLs relativas al origen de todos los recursos que deseas almacenar en caché. - Si se rechaza la promesa, la instalación falla y el trabajador no hará nada. Esto está bien, ya que puedes corregir tu código y luego intentarlo de nuevo la próxima vez que se registre.
- Después de una instalación exitosa, el service worker se activa. Esto no tiene mucho de un uso distinto la primera vez que se instala/activa tu service worker, pero significa más cuando se actualiza el service worker (consulta la sección Actualizar tu service worker más adelante).
Nota: localStorage funciona de manera similar a la memoria caché del service worker, pero es síncrono, por lo que no está permitido en el service worker.
Nota: IndexedDB se puede usar dentro de un service worker para el almacenamiento de datos si lo requieres.
Respuestas personalizadas a solicitudes
Ahora que tienes los activos de tu sitio almacenados en caché, debes decir al service worker que haga algo con el contenido almacenado en caché. Esto se hace fácilmente con el evento fetch
.
Un evento fetch
se activa cada vez que se recupera cualquier recurso controlado por un service worker, lo que incluye los documentos dentro del alcance especificado y cualquier recurso al que se haga referencia en esos documentos (por ejemplo, si index.html
hace una solicitud de origen cruzado para incrustar una imagen, que todavía pasa por su service worker).
Puedes adjuntar un detector de eventos fetch
al service worker, luego llamar al método respondWith()
en el evento para capturar nuestras respuestas HTTP y actualizarlas con tu propia magia.
self.addEventListener("fetch", (event) => {
event
.respondWith
// la magia va aquí
();
});
Podríamos empezar respondiendo con el recurso cuya URL coincida con la de la solicitud de red, en cada caso:
self.addEventListener("fetch", (event) => {
event.respondWith(caches.match(event.request));
});
caches.match(event.request)
nos permite hacer coincidir cada recurso solicitado de la red con el recurso equivalente disponible en caché, si hay uno coincidente disponible. La coincidencia se realiza a través de URL y varios encabezados, al igual que con las solicitudes HTTP normales.
Veamos algunas otras opciones que tenemos al definir nuestra magia (consulta nuestra documentación de la API Fetch para obtener más información sobre los objetos Request
y Response
.)
- El constructor
Response()
te permite crear una respuesta personalizada. En este caso, solo estamos devolviendo una cadena de texto simple:jsnew Response("¡Hola desde tu amigable vecindario del service worker!");
- Esta
Response
más compleja a continuación muestra que, opcionalmente, puedes pasar un conjunto de encabezados con tu respuesta, emulando los encabezados de respuesta HTTP estándar. Aquí solo le estamos diciendo al navegador cuál es el tipo de contenido de nuestra respuesta sintética:jsnew Response( "<p>¡Hola desde tu amigable vecindario del service worker!</p>", { headers: { "Content-Type": "text/html" }, }, );
- Si no se encontró una coincidencia en caché, le puedes decir al navegador que
fetch()
la solicitud de red predeterminada para ese recurso, para obtener el nuevo recurso de la red si está disponible:jsfetch(event.request);
- Si no se encontró una coincidencia en caché y la red no está disponible, puedes hacer coincidir la solicitud con algún tipo de página de respaldo predeterminada como respuesta usando
match()
, como esta:jscaches.match("./fallback.html");
- Puedes recuperar mucha información sobre cada solicitud llamando a los parámetros del objeto
Request
devuelto porFetchEvent
:jsevent.request.url; event.request.method; event.request.headers; event.request.body;
Recuperar solicitudes fallidas
Entonces caches.match(event.request)
es excelente cuando hay una coincidencia en caché del service worker, pero ¿qué pasa con los casos en los que no hay una coincidencia? Si no proporcionamos ningún tipo de manejo de fallas, nuestra promesa se resolvería con undefined
y no tendríamos nada devuelto.
Afortunadamente, la estructura basada en promesas del service worker hace que sea trivial brindar más opciones hacia el éxito. Podríamos hacer esto:
const cacheFirst = async (request) => {
const responseFromCache = await caches.match(request);
if (responseFromCache) {
return responseFromCache;
}
return fetch(request);
};
self.addEventListener("fetch", (event) => {
event.respondWith(cacheFirst(event.request));
});
Si los recursos no están en la memoria caché, se solicitan desde la red.
Si fuéramos realmente inteligentes, no solo solicitaríamos el recurso de la red; ¡también lo guardaríamos en caché para que las solicitudes posteriores de ese recurso también se puedan recuperar sin conexión! Esto significaría que si se agregaran imágenes adicionales a la galería de Star Wars, nuestra aplicación podría capturarlas automáticamente y almacenarlas en caché. Lo siguiente haría el truco:
const putInCache = async (request, response) => {
const cache = await caches.open("v1");
await cache.put(request, response);
};
const cacheFirst = async (request) => {
const responseFromCache = await caches.match(request);
if (responseFromCache) {
return responseFromCache;
}
const responseFromNetwork = await fetch(request);
putInCache(request, responseFromNetwork.clone());
return responseFromNetwork;
};
self.addEventListener("fetch", (event) => {
event.respondWith(cacheFirst(event.request));
});
Si la URL de la solicitud no está disponible en la memoria caché, solicitamos el recurso de la solicitud de red con await fetch(request)
. Después de eso, colocamos en caché un clon de la respuesta. La función putInCache
usa caches.open('v1')
y cache.put()
para agregar el recurso a la caché. La respuesta original se devuelve al navegador para que se proporcione a la página que la llamó.
La clonación de la respuesta es necesaria porque los flujos de solicitud y respuesta solo se pueden leer una vez. Para devolver la respuesta al navegador y ponerla en caché la tenemos que clonar. Entonces, el original se devuelve al navegador y el clon se envía a caché. Cada uno se lee una vez.
Lo que puede parecer un poco extraño es que no se espera la promesa devuelta por putInCache
. Pero la razón es que no queremos esperar hasta que el clon de respuesta se haya agregado a la caché antes de devolver una respuesta.
El único problema que tenemos ahora es que si la solicitud no coincide con nada en caché y la red no está disponible, nuestra solicitud seguirá fallando. Proporcionemos un respaldo predeterminado para que, pase lo que pase, el usuario al menos obtenga algo:
const putInCache = async (request, response) => {
const cache = await caches.open("v1");
await cache.put(request, response);
};
const cacheFirst = async ({ request, preloadResponsePromise, fallbackUrl }) => {
// Primero intenta obtener el recurso desde caché
const responseFromCache = await caches.match(request);
if (responseFromCache) {
return responseFromCache;
}
// A continuación, intenta obtener el recurso desde la red
try {
const responseFromNetwork = await fetch(request);
// la respuesta solo se puede usar una vez
// necesitamos guardar el clon para poner una copia en caché
// y servir el segundo
putInCache(request, responseFromNetwork.clone());
return responseFromNetwork;
} catch (error) {
const fallbackResponse = await caches.match(fallbackUrl);
if (fallbackResponse) {
return fallbackResponse;
}
// cuando incluso la respuesta alternativa no está disponible,
// no hay nada que podamos hacer, pero siempre debemos
// devolver un objeto Response
return new Response("Ocurrió un error de red", {
status: 408,
headers: { "Content-Type": "text/plain" },
});
}
};
self.addEventListener("fetch", (event) => {
event.respondWith(
cacheFirst({
request: event.request,
fallbackUrl: "/gallery/myLittleVader.jpg",
}),
);
});
Hemos optado por esta imagen alternativa porque las únicas actualizaciones que probablemente fallarán son las imágenes nuevas, ya que todo lo demás depende de la instalación en el detector de eventos install
que vimos anteriormente.
Precarga de navegación del service worker
Si está habilitada, la función de precarga de navegación comienza a descargar recursos tan pronto como se realiza la solicitud de recuperación y en paralelo con el inicio del service worker. Esto garantiza que la descarga comience de inmediato al navegar a una página, en lugar de tener que esperar hasta que se inicie el service worker. Ese retraso ocurre en muy raras ocasiones, pero es inevitable cuando ocurre y puede ser significativo.
Primero, la función debe estar habilitada durante la activación del service worker, usando registration.navigationPreload.enable()
:
const enableNavigationPreload = async () => {
if (self.registration.navigationPreload) {
// ¡Habilitar precargas de navegación!
await self.registration.navigationPreload.enable();
}
};
self.addEventListener("activate", (event) => {
event.waitUntil(enableNavigationPreload());
});
Luego usa event.preloadResponse
para esperar a que el recurso precargado se termine de descargar en el controlador de eventos fetch
.
Continuando con el ejemplo de las secciones anteriores, insertamos el código para esperar el recurso precargado después de la verificación de la caché y antes de recuperarlo de la red si eso no tiene éxito.
El nuevo proceso es:
- Comprobar la caché
-
Esperar en
event.preloadResponse
, que se pasa comopreloadResponsePromise
a la funcióncacheFirst
. Guardar en caché el resultado si regresa. - Si ninguno de estos está definido, vamos a la red.
const addResourcesToCache = async (resources) => {
const cache = await caches.open("v1");
await cache.addAll(resources);
};
const putInCache = async (request, response) => {
const cache = await caches.open("v1");
await cache.put(request, response);
};
const cacheFirst = async ({ request, preloadResponsePromise, fallbackUrl }) => {
// Primero intenta obtener el recurso desde caché
const responseFromCache = await caches.match(request);
if (responseFromCache) {
return responseFromCache;
}
// A continuación, intenta usar (y almacenar en caché) la respuesta precargada, si está allí
const preloadResponse = await preloadResponsePromise;
if (preloadResponse) {
console.info("using preload response", preloadResponse);
putInCache(request, preloadResponse.clone());
return preloadResponse;
}
// A continuación, intenta obtener el recurso desde la red
try {
const responseFromNetwork = await fetch(request);
// la respuesta solo se puede usar una vez
// necesitamos guardar el clon para poner una copia en caché
// y servir el segundo
putInCache(request, responseFromNetwork.clone());
return responseFromNetwork;
} catch (error) {
const fallbackResponse = await caches.match(fallbackUrl);
if (fallbackResponse) {
return fallbackResponse;
}
// cuando incluso la respuesta alternativa no está disponible,
// no hay nada que podamos hacer, pero siempre debemos
// devolver un objeto Response
return new Response("Ocurrió un error de red", {
status: 408,
headers: { "Content-Type": "text/plain" },
});
}
};
// Habilita la precarga de navegación
const enableNavigationPreload = async () => {
if (self.registration.navigationPreload) {
// ¡Habilitar precargas de navegación!
await self.registration.navigationPreload.enable();
}
};
self.addEventListener("activate", (event) => {
event.waitUntil(enableNavigationPreload());
});
self.addEventListener("install", (event) => {
event.waitUntil(
addResourcesToCache([
"/",
"/index.html",
"/style.css",
"/app.js",
"/image-list.js",
"/star-wars-logo.jpg",
"/gallery/bountyHunters.jpg",
"/gallery/myLittleVader.jpg",
"/gallery/snowTroopers.jpg",
]),
);
});
self.addEventListener("fetch", (event) => {
event.respondWith(
cacheFirst({
request: event.request,
preloadResponsePromise: event.preloadResponse,
fallbackUrl: "/gallery/myLittleVader.jpg",
}),
);
});
Ten en cuenta que en este ejemplo descargamos y almacenamos en caché los mismos datos para el recurso, ya sea que se descargue "normalmente" o se precargue. En su lugar, puedes optar por descargar y almacenar en caché un recurso diferente en la precarga. Para obtener más información, consulta NavigationPreloadManager > Respuestas personalizadas.
Actualizar tu service worker
Si tu service worker se instaló anteriormente, pero luego está disponible una nueva versión del trabajador al actualizar o cargar la página, la nueva versión se instala en segundo plano, pero aún no está activada. Solo se activa cuando ya no hay páginas cargadas que todavía estén usando el antiguo service worker. Tan pronto como no queden más páginas cargadas, se activa el nuevo service worker.
Querrás actualizar tu escucha de eventos install
en el nuevo service worker a algo como esto (observa el nuevo número de versión):
const addResourcesToCache = async (resources) => {
const cache = await caches.open("v2");
await cache.addAll(resources);
};
self.addEventListener("install", (event) => {
event.waitUntil(
addResourcesToCache([
"/",
"/index.html",
"/style.css",
"/app.js",
"/image-list.js",
// ...
// incluir otros nuevos recursos para la nueva versión…
]),
);
});
Mientras esto sucede, la versión anterior sigue siendo responsable de las recuperaciones. La nueva versión se está instalando en segundo plano. Estamos llamando al nuevo caché v2
, por lo que el caché anterior v1
no se ve afectado.
Cuando ninguna página está usando la versión actual, el nuevo trabajador se activa y se vuelve responsable de las recuperaciones.
Eliminar cachés antiguos
También obtienes un evento activate
. Esto generalmente se usa para hacer cosas que habrían roto la versión anterior mientras aún se estaba ejecutando, por ejemplo, deshacerse de los cachés antiguos. Esto también es útil para eliminar datos que ya no se necesitan para evitar llenar demasiado espacio en disco: cada navegador tiene un límite estricto en la cantidad de almacenamiento en caché que puede usar un determinado service worker. El navegador hace todo lo posible para administrar el espacio en disco, pero puede eliminar el almacenamiento en caché de un origen. El navegador, generalmente, eliminará todos los datos de un origen o ninguno de los datos de un origen.
Las promesas pasadas a waitUntil()
bloquearán otros eventos hasta que se completen, por lo que puedes estar seguro de que tu operación de limpieza se habrá completado cuando obtengas tu primer evento fetch
en el nuevo service worker.
const deleteCache = async (key) => {
await caches.delete(key);
};
const deleteOldCaches = async () => {
const cacheKeepList = ["v2"];
const keyList = await caches.keys();
const cachesToDelete = keyList.filter((key) => !cacheKeepList.includes(key));
await Promise.all(cachesToDelete.map(deleteCache));
};
self.addEventListener("activate", (event) => {
event.waitUntil(deleteOldCaches());
});
Herramientas de desarrollo
Chrome tiene chrome://inspect/#service-workers
, que muestra la actividad actual de los service workers y el almacenamiento en un dispositivo, y chrome://serviceworker-internals
, que muestra más detalles y te permite iniciar/detener/depurar el proceso del trabajador. En el futuro, tendrán modos de limitación/desconexión para simular conexiones defectuosas o inexistentes, lo que será algo realmente bueno.
Firefox también ha comenzado a implementar algunas herramientas útiles relacionadas con los service workers:
- Puedes navegar a
about:debugging
para ver qué SWs están registrados y actualizarlos/eliminarlos. - Al realizar pruebas, puedes sortear la restricción de HTTPS marcando la opción "Habilitar service worker a través de HTTP (cuando la caja de herramientas está abierta)" en la Configuración de herramientas de desarrollo de Firefox.
- El botón "Olvidar", disponible en las opciones de personalización de Firefox, se puede usar para borrar los service workers y sus cachés (Error 1252998 en Firefox).
Nota: Puedes servir tu aplicación desde http://localhost
(por ejemplo, usando me@localhost:/my/app$ python -m SimpleHTTPServer
) para el desarrollo local. Ve Consideraciones de seguridad