Back to sh0
sh0

Ciclo de vida de aplicaciones desde la terminal

Cómo construimos sh0 restart, stop, start, delete y domains -- cinco comandos CLI para gestionar el ciclo de vida de aplicaciones con patrones de confirmación orientados a la seguridad.

Juste A. Gnimavo (Thales) & Claude | March 27, 2026 3 min sh0
EN/ FR/ ES
clirustapp-managementdomainsconfirmation-promptssafety

Desplegar una app es el comienzo, no el final. Después del despliegue viene el trabajo diario de gestionarla: reiniciar después de un cambio de configuración, detener durante mantenimiento, eliminar cuando un proyecto termina, agregar dominios personalizados cuando va a producción.

La Fase 3 agregó cinco comandos para manejar el ciclo de vida completo de aplicaciones. La implementación técnica es directa -- son wrappers delgados alrededor de endpoints API existentes. Lo que los hace interesantes es el diseño de seguridad: ¿cómo permites que un desarrollador elimine una app desde la terminal sin hacer demasiado fácil eliminar la equivocada?

El patrón de wrapper delgado

Los cinco comandos de la Fase 3 siguen la misma estructura: parsear el argumento de app, resolverlo vía client.resolve_app(), hacer la llamada API, imprimir el resultado.

restart en su totalidad son siete líneas. Sin lógica de negocio, sin gestión de estado, sin manejo complejo de errores. El servidor hace el trabajo. El CLI provee la interfaz.

Prompts de confirmación: dos niveles de precaución

Detener una app es reversible. Eliminar una app no lo es. El CLI refleja esto con dos patrones de confirmación diferentes.

Stop: Sí/No simple

El valor predeterminado es N (no). Un Enter accidental no hace nada. La bandera --yes omite el prompt para scripting.

Delete: Escribe el nombre

Detener es un mal día. Eliminar es una catástrofe. La confirmación para delete requiere que el desarrollador escriba el nombre completo de la app. El nombre escrito debe coincidir exactamente. Sin coincidencia fuzzy, sin atajo "yes". Este es el mismo patrón que GitHub usa para eliminar repositorios, y por la misma razón: la fricción es la funcionalidad.

El bug del parámetro de consulta

La auditoría encontró el bug más sutil de todo el proyecto de mejora del CLI en este comando. La implementación original usaba ?cleanup=true como parámetro de consulta. El servidor esperaba ?delete_volumes=true.

¿Por qué era crítico? Porque serde silenciosamente ignora parámetros de consulta desconocidos. La llamada de eliminación tenía éxito -- la app se eliminaba -- pero sus volúmenes Docker se preservaban. Volúmenes huérfanos acumulándose en disco, invisibles para el usuario, consumiendo almacenamiento indefinidamente.

La corrección fue una palabra: cleanup a delete_volumes. Pero el bug habría sido invisible en pruebas porque la operación de eliminación misma tenía éxito.

Gestión de dominios: un subcomando

Los dominios son diferentes de los otros comandos porque tienen múltiples acciones. En lugar de cuatro comandos separados, usamos un patrón de subcomandos: sh0 domains my-app list, add, remove.

La auditoría mejoró la acción remove. La implementación original solo aceptaba IDs de dominio. La corrección fue resolver dominios por nombre primero, recurriendo al ID.

El CLI completo

Después de la Fase 3, el CLI de sh0 tiene 25 comandos en seis categorías. Cada acción disponible en el dashboard ahora está disponible desde la terminal. El CLI no es un subconjunto del dashboard -- es una interfaz alternativa completa.


Siguiente en la serie: Modo watch y streaming WebSocket -- Auto-despliegue al cambiar archivos, y actualización de polling HTTP a streaming de logs de build en tiempo real por WebSocket.

Share this article:

Responses

Write a response
0/2000
Loading responses...

Related Articles

Thales & Claude deblo

El Step Zero no bastó: cómo validar un constructor pero no el runtime tumbó cada sesión de voz de Déblo la hora en que enviamos streaming de cámara en tiempo real

La Fase 14 envió Déblo Eyes — streaming de cámara en tiempo real por LiveKit hacia Gemini Live native audio. El primer despliegue tumbó cada sesión de voz en producción en noventa segundos porque nuestro Step 0 había validado el constructor sin ejercitar el runtime. El build log de cómo Déblo obtuvo ojos, lo que costó un pre-vuelo incompleto, y qué pulidos enviamos versus aplazamos.

33 min May 20, 2026
debloclaude-opus-4.7claude-codegemini-live +25
Thales & Claude deblo

La raya que mató producción: cómo un eslogan de marketing en un encabezado HTTP tumbó el chat de Déblo durante 24 horas

Dos días antes del envío a la App Store, todo el producto de chat de Déblo se rompió en silencio. Sin spinner, sin toast, sin error en la UI — solo aire muerto. La interrupción de 24 horas se reducía a una sola « é » en el valor de un encabezado HTTP que lanzaba UnicodeEncodeError antes de que cualquier petición a OpenRouter saliera del backend. El post-mortem de una falsa hipótesis, una traza de Sentry, y un fix de seis líneas que desbloqueó el lanzamiento.

29 min May 19, 2026
debloclaude-opus-4.7claude-codeincident +19
Thales & Claude deblo

Seis horas, de página en blanco a Apple Review — Cómo enviamos Déblo a la App Store, en vivo

Recorrido en vivo del envío de Déblo a la App Store iOS en seis horas: lo que rechazaron los validadores de Apple (un superíndice Unicode), lo que corregimos (un Promotional Text desperdiciado en marcas de terceros), y los mecanismos del ASO de iOS que casi todos se pierden.

30 min May 13, 2026
debloclaude-opus-4.7claude-codeapp-store +16