package main import ( "context" "io" "log" "net/http" "time" "github.com/docker/docker/pkg/stdcopy" "github.com/gin-gonic/gin" "nhooyr.io/websocket" "github.com/nemunaire/minifaas/engine/docker" "github.com/nemunaire/minifaas/ui" ) type App struct { router *gin.Engine srv *http.Server } func NewApp() App { //gin.SetMode(gin.ReleaseMode) gin.ForceConsoleColor() router := gin.Default() ui.DeclareRoutes(router) router.GET("/api/version", func(c *gin.Context) { c.JSON(http.StatusOK, gin.H{"version": 0.1}) }) router.GET("/api/ps", func(c *gin.Context) { containers, err := docker.Ps() if err != nil { c.JSON(http.StatusInternalServerError, gin.H{"error": err}) return } c.JSON(http.StatusOK, containers) }) router.GET("/api/run", func(c *gin.Context) { ctrid, err := docker.Run("alpine", []string{"sh", "-c", "for i in `seq 20`; do echo $i; sleep 0.5; done"}) if err != nil { c.JSON(http.StatusInternalServerError, gin.H{"error": err}) } c.JSON(http.StatusOK, gin.H{"jobid": ctrid}) }) router.GET("/api/jobs/:jobid/logs", func(c *gin.Context) { ws, err := websocket.Accept(c.Writer, c.Request, nil) if err != nil { log.Fatal("error get connection", err) } defer ws.Close(websocket.StatusInternalError, "the sky is falling") stream, err := docker.Logs(c.Param("jobid")) if err != nil { log.Fatal("Unable to get log stream:", err) } r, w := io.Pipe() ctx, cancel := context.WithTimeout(c.Request.Context(), time.Second*10) defer cancel() go func(reader io.Reader) { buf := make([]byte, 128) for { n, err := reader.Read(buf) if err != nil { break } ws.Write(ctx, websocket.MessageText, buf[:n]) } }(r) _, err = stdcopy.StdCopy(w, w, stream) if err != nil && err != io.EOF { log.Fatal("Something goes wrong during stream:", err) } ws.Close(websocket.StatusNormalClosure, "") }) app := App{ router: router, } return app } func (app *App) Start() { app.srv = &http.Server{ Addr: ":8082", Handler: app.router, } if err := app.srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { log.Fatalf("listen: %s\n", err) } } func (app *App) Stop() { ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() if err := app.srv.Shutdown(ctx); err != nil { log.Fatal("Server Shutdown:", err) } }