El día de hoy te enseñaré a detener un servidor web creado con Go desde el propio programa. Dicho de otra manera, el servidor se va a detener a sí mismo.
Para estoy hay varios enfoques pero te enseñaré 2 que yo considero son muy fáciles y no involucran canales o rutinas (go routines).
La forma correcta de terminar el programa de Go cuando se ejecuta un servidor http es invocando a Shutdown
del propio servidor. Normalmente el servidor es creado con el siguiente código:
var servidor *http.Server
O con el siguiente:
servidor = &http.Server{
Handler: enrutador,
Addr: direccion,
// Timeouts para evitar que el servidor se quede "colgado" por siempre
WriteTimeout: 15 * time.Second,
ReadTimeout: 15 * time.Second,
}
Cuando nosotros necesitemos detener el servidor, invocamos a servidor.Shutdown
y le pasamos el contexto context.Background()
(context viene del paquete context
).
Por ejemplo aquí algo simple donde el servidor se apaga desde una ruta creada con Gorilla mux:
var servidor *http.Server
enrutador := mux.NewRouter()
configurarRutas(enrutador)
enrutador.HandleFunc("/apagar", func(w http.ResponseWriter, r *http.Request) {
servidor.Shutdown(context.Background())
}).Methods("GET").Name(RutaApagarServidor)
enrutador.Use(middlewareCors)
// Preparar y encender servidor
direccion := ":8484"
servidor = &http.Server{
Handler: enrutador,
Addr: direccion,
// Timeouts para evitar que el servidor se quede "colgado" por siempre
WriteTimeout: 15 * time.Second,
ReadTimeout: 15 * time.Second,
}
Cuando se invoque a la ruta “apagar” el servidor se va a apagar. Esto, me parece, es la manera correcta y amable de apagar el servidor desde una ruta http y funciona bien, pero en mi caso cuando agrego algunos middlewares me da algunos errores así que probé con otra solución.
Yo recomiendo que uses la solución provista anteriormente, pero si no funciona sigue leyendo.
Otra manera un poco rara pero que funciona es… obtener el ID de proceso (pid) y pedirle al sistema operativo que termine el proceso por nosotros. Yo lo he hecho con Windows y funciona perfectamente; solo sería cuestión de adaptarlo a otros sistemas operativos.
Sé que este método es poco ortodoxo pero funciona y es muy simple (incluso me siento sucio por usarlo). El código queda así:
enrutador.HandleFunc("/apagar", func(w http.ResponseWriter, r *http.Request) {
pid := os.Getpid()
cmd := exec.Command("taskkill", "/F", "/PID", fmt.Sprintf("%d", pid))
err := cmd.Run()
if err != nil{
log.Printf("Error matando proceso: %v", err)
}
}).Methods(http.MethodGet)
Obtenemos el PID con os.Getpid
, luego invocamos a taskkill
en Windows y le pasamos el PID. El comando se va a ejecutar al invocar con cmd.Run
. Y así es como se va a terminar todo nuestro programa.
Entiendo que tal vez todo tu programa no dependa solo del servidor, y para esos casos puede que quieras ver la primera solución. Otras soluciones más complejas las he visto en: https://stackoverflow.com/questions/47448666/shutting-down-http-server-after-returning-response
Hoy te voy a presentar un creador de credenciales que acabo de programar y que…
Ya te enseñé cómo convertir una aplicación web de Vue 3 en una PWA. Al…
En este artículo voy a documentar la arquitectura que yo utilizo al trabajar con WebAssembly…
En un artículo anterior te enseñé a crear un PWA. Al final, cualquier aplicación que…
Al usar Comlink para trabajar con los workers usando JavaScript me han aparecido algunos errores…
En este artículo te voy a enseñar cómo usar un "top level await" esperando a…
Esta web usa cookies.