Endpoint for running processes (in supervisor)
This commit is contained in:
		
							parent
							
								
									f6f26e89de
								
							
						
					
					
						commit
						4a22822c10
					
				
					 4 changed files with 534 additions and 429 deletions
				
			
		|  | @ -4,6 +4,7 @@ import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"log" | 	"log" | ||||||
|  | @ -364,9 +365,9 @@ func (d *Driver) Create(name string, image string, volumePath string, HTTPPort i | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Exec runs command cmd with stdin if it's not empty.
 | // Exec runs command cmd with stdin if it's not empty.
 | ||||||
| func (d *Driver) Exec(name string, cmd []string, stdin string, env []string) error { | func (d *Driver) Exec(name string, cmd []string, stdin string, env []string, attachStdout bool) (*[]byte, error) { | ||||||
| 	if len(cmd) == 0 { | 	if len(cmd) == 0 { | ||||||
| 		return errors.New("cmd needs at least one string in the slice") | 		return &[]byte{}, errors.New("cmd needs at least one string in the slice") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
|  | @ -379,43 +380,55 @@ func (d *Driver) Exec(name string, cmd []string, stdin string, env []string) err | ||||||
| 	log.Println("Command running in " + name) | 	log.Println("Command running in " + name) | ||||||
| 	cli, err := d.getClient() | 	cli, err := d.getClient() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return &[]byte{}, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	containerID, err := d.nameToID(name) | 	containerID, err := d.nameToID(name) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return &[]byte{}, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	execOpts := types.ExecConfig{ | 	execOpts := types.ExecConfig{ | ||||||
| 		AttachStdin:  stdinEnabled, | 		AttachStdin:  stdinEnabled, | ||||||
| 		AttachStdout: false, | 		AttachStdout: attachStdout, | ||||||
| 		AttachStderr: false, | 		AttachStderr: false, | ||||||
|  | 		Tty:          attachStdout, | ||||||
| 		Cmd:          cmd, | 		Cmd:          cmd, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, err := cli.ContainerExecCreate(ctx, containerID, execOpts) | 	resp, err := cli.ContainerExecCreate(ctx, containerID, execOpts) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return &[]byte{}, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	respAttach, err := cli.ContainerExecAttach(ctx, resp.ID, types.ExecConfig{}) | 	respAttach, err := cli.ContainerExecAttach(ctx, resp.ID, types.ExecConfig{ | ||||||
|  | 		AttachStdin:  stdinEnabled, | ||||||
|  | 		AttachStdout: attachStdout, | ||||||
|  | 		AttachStderr: false, | ||||||
|  | 		Tty:          attachStdout, | ||||||
|  | 	}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return &[]byte{}, err | ||||||
| 	} | 	} | ||||||
| 	defer respAttach.Close() | 	defer respAttach.Close() | ||||||
| 
 | 
 | ||||||
| 	err = cli.ContainerExecStart(ctx, resp.ID, types.ExecStartCheck{}) | 	err = cli.ContainerExecStart(ctx, resp.ID, types.ExecStartCheck{}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return &[]byte{}, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if stdinEnabled { | 	if stdinEnabled { | ||||||
| 		_, err = respAttach.Conn.Write([]byte(stdin)) | 		_, err = respAttach.Conn.Write([]byte(stdin)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return &[]byte{}, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return nil | 	stdouterr := []byte{} | ||||||
|  | 	if attachStdout { | ||||||
|  | 		stdouterr, err = ioutil.ReadAll(respAttach.Reader) | ||||||
|  | 		fmt.Println(string(stdouterr)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return &stdouterr, err | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -3,6 +3,7 @@ package docker | ||||||
| import ( | import ( | ||||||
| 	"log" | 	"log" | ||||||
| 	"path" | 	"path" | ||||||
|  | 	"strings" | ||||||
| 
 | 
 | ||||||
| 	"github.com/rosti-cz/node-api/apps" | 	"github.com/rosti-cz/node-api/apps" | ||||||
| ) | ) | ||||||
|  | @ -11,6 +12,12 @@ import ( | ||||||
| const appUsername = "app" | const appUsername = "app" | ||||||
| const passwordFile = "/srv/.rosti" | const passwordFile = "/srv/.rosti" | ||||||
| 
 | 
 | ||||||
|  | // Process contains info about background application usually running in supervisor
 | ||||||
|  | type Process struct { | ||||||
|  | 	Name  string | ||||||
|  | 	State string | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Container extends App struct from App
 | // Container extends App struct from App
 | ||||||
| type Container struct { | type Container struct { | ||||||
| 	App *apps.App `json:"app"` | 	App *apps.App `json:"app"` | ||||||
|  | @ -171,12 +178,12 @@ func (c *Container) Delete() error { | ||||||
| func (c *Container) SetPassword(password string) error { | func (c *Container) SetPassword(password string) error { | ||||||
| 	driver := c.getDriver() | 	driver := c.getDriver() | ||||||
| 
 | 
 | ||||||
| 	err := driver.Exec(c.App.Name, []string{"chpasswd"}, appUsername+":"+password, []string{}) | 	_, err := driver.Exec(c.App.Name, []string{"chpasswd"}, appUsername+":"+password, []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = driver.Exec(c.App.Name, []string{"tee", passwordFile}, password, []string{}) | 	_, err = driver.Exec(c.App.Name, []string{"tee", passwordFile}, password, []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | @ -191,27 +198,27 @@ func (c *Container) SetFileContent(filename string, text string, mode string) er | ||||||
| 
 | 
 | ||||||
| 	directory := path.Dir(filename) | 	directory := path.Dir(filename) | ||||||
| 
 | 
 | ||||||
| 	err := driver.Exec(c.App.Name, []string{"mkdir", "-p", directory}, "", []string{}) | 	_, err := driver.Exec(c.App.Name, []string{"mkdir", "-p", directory}, "", []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = driver.Exec(c.App.Name, []string{"tee", filename}, text, []string{}) | 	_, err = driver.Exec(c.App.Name, []string{"tee", filename}, text, []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = driver.Exec(c.App.Name, []string{"chown", directory, "app:app"}, "", []string{}) | 	_, err = driver.Exec(c.App.Name, []string{"chown", directory, "app:app"}, "", []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = driver.Exec(c.App.Name, []string{"chown", filename, "app:app"}, "", []string{}) | 	_, err = driver.Exec(c.App.Name, []string{"chown", filename, "app:app"}, "", []string{}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = driver.Exec(c.App.Name, []string{"chmod", mode, filename}, "", []string{}) | 	_, err = driver.Exec(c.App.Name, []string{"chmod", mode, filename}, "", []string{}, false) | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -220,6 +227,31 @@ func (c *Container) SetFileContent(filename string, text string, mode string) er | ||||||
| func (c *Container) SetTechnology(tech string) error { | func (c *Container) SetTechnology(tech string) error { | ||||||
| 	driver := c.getDriver() | 	driver := c.getDriver() | ||||||
| 
 | 
 | ||||||
| 	err := driver.Exec(c.App.Name, []string{"su", "app", "-c", "rosti " + tech}, "", []string{}) | 	_, err := driver.Exec(c.App.Name, []string{"su", "app", "-c", "rosti " + tech}, "", []string{}, false) | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // GetProcessList returns list of processes managed by supervisor.
 | ||||||
|  | func (c *Container) GetProcessList() (*[]Process, error) { | ||||||
|  | 	driver := c.getDriver() | ||||||
|  | 
 | ||||||
|  | 	processes := []Process{} | ||||||
|  | 
 | ||||||
|  | 	stdouterr, err := driver.Exec(c.App.Name, []string{"supervisorctl", "status"}, "", []string{}, true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return &processes, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	trimmed := strings.TrimSpace(string(*stdouterr)) | ||||||
|  | 	for _, row := range strings.Split(trimmed, "\n") { | ||||||
|  | 		fields := strings.Fields(row) | ||||||
|  | 		if len(fields) > 2 { | ||||||
|  | 			processes = append(processes, Process{ | ||||||
|  | 				Name:  fields[0], | ||||||
|  | 				State: fields[1], | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return &processes, nil | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										449
									
								
								handlers.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										449
									
								
								handlers.go
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,449 @@ | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
|  | 	"github.com/labstack/echo" | ||||||
|  | 	"github.com/rosti-cz/node-api/apps" | ||||||
|  | 	"github.com/rosti-cz/node-api/docker" | ||||||
|  | 	"github.com/rosti-cz/node-api/node" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func homeHandler(c echo.Context) error { | ||||||
|  | 	return c.Render(http.StatusOK, "index.html", templateData{ | ||||||
|  | 		Token: configuredToken, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func listAppsHandler(c echo.Context) error { | ||||||
|  | 	err := gatherStates() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	applications, err := apps.List() | ||||||
|  | 
 | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, applications) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Returns one app
 | ||||||
|  | func getAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	err := updateState(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app, err = apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, app) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Create a new app
 | ||||||
|  | // If you add register_only=1 into query string, it won't start or create any container, just adds record into the database.
 | ||||||
|  | func createAppHandler(c echo.Context) error { | ||||||
|  | 	registerOnly := c.QueryParam("register_only") == "1" | ||||||
|  | 
 | ||||||
|  | 	app := apps.App{} | ||||||
|  | 	err := c.Bind(&app) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = apps.New(app.Name, app.SSHPort, app.HTTPPort, app.Image, app.CPU, app.Memory) | ||||||
|  | 	if err != nil { | ||||||
|  | 		if validationError, ok := err.(apps.ValidationError); ok { | ||||||
|  | 			return c.JSONPretty(http.StatusBadRequest, Message{Errors: validationError.Errors}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if !registerOnly { | ||||||
|  | 		container := docker.Container{ | ||||||
|  | 			App: &app, | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		err = container.Create() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		err = container.Start() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Update existing app
 | ||||||
|  | func updateAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app := apps.App{} | ||||||
|  | 	err := c.Bind(&app) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	appPointer, err := apps.Update(name, app.SSHPort, app.HTTPPort, app.Image, app.CPU, app.Memory) | ||||||
|  | 	if err != nil { | ||||||
|  | 		if validationError, ok := err.(apps.ValidationError); ok { | ||||||
|  | 			return c.JSONPretty(http.StatusBadRequest, Message{Errors: validationError.Errors}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app = *appPointer | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: &app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Destroy() | ||||||
|  | 	if err != nil && err.Error() == "no container found" { | ||||||
|  | 		// We don't care if the container didn't exist anyway
 | ||||||
|  | 		err = nil | ||||||
|  | 	} | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Create() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Start() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Stop existing app
 | ||||||
|  | func stopAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Stop() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Start existing app
 | ||||||
|  | func startAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Start() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Stop existing app
 | ||||||
|  | func restartAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Restart() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Set password for the app user in the container
 | ||||||
|  | func setPasswordHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	password := Password{} | ||||||
|  | 	err := c.Bind(&password) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.SetPassword(password.Password) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Copies body of the request into /srv/.ssh/authorized_keys
 | ||||||
|  | func setKeysHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	body, err := ioutil.ReadAll(c.Request().Body) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.SetFileContent(sshPubKeysLocation, string(body)+"\n", "0600") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func setServicesHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	quickServices := &QuickServices{} | ||||||
|  | 	err := c.Bind(quickServices) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if quickServices.Python { | ||||||
|  | 		err = container.SetTechnology("python") | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if quickServices.PHP { | ||||||
|  | 		err = container.SetTechnology("php") | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if quickServices.Node { | ||||||
|  | 		err = container.SetTechnology("node") | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if quickServices.Memcached { | ||||||
|  | 		err = container.SetTechnology("memcached") | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if quickServices.Redis { | ||||||
|  | 		err = container.SetTechnology("redis") | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Rebuilds existing app, it keeps the data but created the container again
 | ||||||
|  | func rebuildAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Destroy() | ||||||
|  | 	if err != nil && err.Error() == "no container found" { | ||||||
|  | 		// We don't care if the container didn't exist anyway
 | ||||||
|  | 		err = nil | ||||||
|  | 	} | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Create() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = container.Start() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Adds new label
 | ||||||
|  | func addLabelHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 	label := apps.Label{} | ||||||
|  | 
 | ||||||
|  | 	err := c.Bind(&label) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = apps.AddLabel(name, label.Value) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Removes existing label
 | ||||||
|  | func deleteLabelHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 	label := apps.Label{} | ||||||
|  | 
 | ||||||
|  | 	err := c.Bind(&label) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = apps.RemoveLabel(name, label.Value) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "ok"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Delete one app
 | ||||||
|  | func deleteAppHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	status, err := container.Status() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 	if status != "no-container" { | ||||||
|  | 		err = container.Delete() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = apps.Delete(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, Message{Message: "deleted"}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Orphans returns directories in /srv that doesn't match any hosted application
 | ||||||
|  | func getOrphansHander(c echo.Context) error { | ||||||
|  | 	return c.JSON(http.StatusOK, []string{}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Return info about the node including performance index
 | ||||||
|  | func getNodeInfoHandler(c echo.Context) error { | ||||||
|  | 	node, err := node.GetNodeInfo() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, node) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Returns list of running processes (inside supervisor)
 | ||||||
|  | func getAppProcessesHandler(c echo.Context) error { | ||||||
|  | 	name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 	app, err := apps.Get(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	container := docker.Container{ | ||||||
|  | 		App: app, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	processes, err := container.GetProcessList() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return c.JSON(http.StatusOK, processes) | ||||||
|  | } | ||||||
							
								
								
									
										431
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										431
									
								
								main.go
									
									
									
									
									
								
							|  | @ -1,15 +1,11 @@ | ||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"io/ioutil" |  | ||||||
| 	"log" | 	"log" | ||||||
| 	"net/http" |  | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/labstack/echo" | 	"github.com/labstack/echo" | ||||||
| 	"github.com/rosti-cz/node-api/apps" |  | ||||||
| 	"github.com/rosti-cz/node-api/common" | 	"github.com/rosti-cz/node-api/common" | ||||||
| 	"github.com/rosti-cz/node-api/docker" |  | ||||||
| 	"github.com/rosti-cz/node-api/node" | 	"github.com/rosti-cz/node-api/node" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | @ -52,444 +48,59 @@ func main() { | ||||||
| 
 | 
 | ||||||
| 	/* e.Use(TokenMiddleware) */ | 	/* e.Use(TokenMiddleware) */ | ||||||
| 
 | 
 | ||||||
|  | 	// UI
 | ||||||
|  | 	e.GET("/", homeHandler) | ||||||
|  | 
 | ||||||
| 	// Returns list of apps
 | 	// Returns list of apps
 | ||||||
| 	e.GET("/", func(c echo.Context) error { | 	e.GET("/v1/apps", listAppsHandler) | ||||||
| 		return c.Render(http.StatusOK, "index.html", templateData{ |  | ||||||
| 			Token: configuredToken, |  | ||||||
| 		}) |  | ||||||
| 	}) |  | ||||||
| 	e.GET("/v1/apps", func(c echo.Context) error { |  | ||||||
| 		err := gatherStates() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		applications, err := apps.List() |  | ||||||
| 
 |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, applications) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Returns one app
 | 	// Returns one app
 | ||||||
| 	e.GET("/v1/apps/:name", func(c echo.Context) error { | 	e.GET("/v1/apps/:name", getAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		err := updateState(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app, err = apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, app) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Create a new app
 | 	// Create a new app
 | ||||||
| 	// If you add register_only=1 into query string, it won't start or create any container, just adds record into the database.
 | 	// If you add register_only=1 into query string, it won't start or create any container, just adds record into the database.
 | ||||||
| 	e.POST("/v1/apps", func(c echo.Context) error { | 	e.POST("/v1/apps", createAppHandler) | ||||||
| 		registerOnly := c.QueryParam("register_only") == "1" |  | ||||||
| 
 |  | ||||||
| 		app := apps.App{} |  | ||||||
| 		err := c.Bind(&app) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = apps.New(app.Name, app.SSHPort, app.HTTPPort, app.Image, app.CPU, app.Memory) |  | ||||||
| 		if err != nil { |  | ||||||
| 			if validationError, ok := err.(apps.ValidationError); ok { |  | ||||||
| 				return c.JSONPretty(http.StatusBadRequest, Message{Errors: validationError.Errors}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if !registerOnly { |  | ||||||
| 			container := docker.Container{ |  | ||||||
| 				App: &app, |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			err = container.Create() |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			err = container.Start() |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Update existing app
 | 	// Update existing app
 | ||||||
| 	e.PUT("/v1/apps/:name", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name", updateAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app := apps.App{} |  | ||||||
| 		err := c.Bind(&app) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		appPointer, err := apps.Update(name, app.SSHPort, app.HTTPPort, app.Image, app.CPU, app.Memory) |  | ||||||
| 		if err != nil { |  | ||||||
| 			if validationError, ok := err.(apps.ValidationError); ok { |  | ||||||
| 				return c.JSONPretty(http.StatusBadRequest, Message{Errors: validationError.Errors}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app = *appPointer |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: &app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Destroy() |  | ||||||
| 		if err != nil && err.Error() == "no container found" { |  | ||||||
| 			// We don't care if the container didn't exist anyway
 |  | ||||||
| 			err = nil |  | ||||||
| 		} |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Create() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Start() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Stop existing app
 | 	// Stop existing app
 | ||||||
| 	e.PUT("/v1/apps/:name/stop", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/stop", stopAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Stop() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Start existing app
 | 	// Start existing app
 | ||||||
| 	e.PUT("/v1/apps/:name/start", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/start", startAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Start() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Stop existing app
 | 	// Stop existing app
 | ||||||
| 	e.PUT("/v1/apps/:name/restart", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/restart", restartAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 | 
 | ||||||
| 		app, err := apps.Get(name) | 	// Application processes
 | ||||||
| 		if err != nil { | 	e.GET("/v1/apps/:name/processes", getAppProcessesHandler) | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Restart() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	// Stats for existing app
 |  | ||||||
| 	e.GET("/v1/apps/:name/stats", func(c echo.Context) error { |  | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		cpu, memory, err := container.ResourceUsage() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 		log.Println("DEBUG", cpu, memory) |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Set password for the app user in the container
 | 	// Set password for the app user in the container
 | ||||||
| 	e.PUT("/v1/apps/:name/password", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/password", setPasswordHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		password := Password{} |  | ||||||
| 		err := c.Bind(&password) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.SetPassword(password.Password) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Copies body of the request into /srv/.ssh/authorized_keys
 | 	// Copies body of the request into /srv/.ssh/authorized_keys
 | ||||||
| 	e.PUT("/v1/apps/:name/keys", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/keys", setKeysHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 | 
 | ||||||
| 		body, err := ioutil.ReadAll(c.Request().Body) | 	e.PUT("/v1/apps/:name/set-services", setServicesHandler) | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.SetFileContent(sshPubKeysLocation, string(body)+"\n", "0600") |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	e.PUT("/v1/apps/:name/set-services", func(c echo.Context) error { |  | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		quickServices := &QuickServices{} |  | ||||||
| 		err := c.Bind(quickServices) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if quickServices.Python { |  | ||||||
| 			err = container.SetTechnology("python") |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if quickServices.PHP { |  | ||||||
| 			err = container.SetTechnology("php") |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if quickServices.Node { |  | ||||||
| 			err = container.SetTechnology("node") |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if quickServices.Memcached { |  | ||||||
| 			err = container.SetTechnology("memcached") |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if quickServices.Redis { |  | ||||||
| 			err = container.SetTechnology("redis") |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Rebuilds existing app, it keeps the data but created the container again
 | 	// Rebuilds existing app, it keeps the data but created the container again
 | ||||||
| 	e.PUT("/v1/apps/:name/rebuild", func(c echo.Context) error { | 	e.PUT("/v1/apps/:name/rebuild", rebuildAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Destroy() |  | ||||||
| 		if err != nil && err.Error() == "no container found" { |  | ||||||
| 			// We don't care if the container didn't exist anyway
 |  | ||||||
| 			err = nil |  | ||||||
| 		} |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Create() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = container.Start() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Adds new label
 | 	// Adds new label
 | ||||||
| 	e.POST("/v1/apps/:name/labels", func(c echo.Context) error { | 	e.POST("/v1/apps/:name/labels", addLabelHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 		label := apps.Label{} |  | ||||||
| 
 |  | ||||||
| 		err := c.Bind(&label) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = apps.AddLabel(name, label.Value) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Removes existing label
 | 	// Removes existing label
 | ||||||
| 	e.DELETE("/v1/apps/:name/labels", func(c echo.Context) error { | 	e.DELETE("/v1/apps/:name/labels", deleteLabelHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 		label := apps.Label{} |  | ||||||
| 
 |  | ||||||
| 		err := c.Bind(&label) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusBadRequest, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = apps.RemoveLabel(name, label.Value) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "ok"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Delete one app
 | 	// Delete one app
 | ||||||
| 	e.DELETE("/v1/apps/:name", func(c echo.Context) error { | 	e.DELETE("/v1/apps/:name", deleteAppHandler) | ||||||
| 		name := c.Param("name") |  | ||||||
| 
 |  | ||||||
| 		app, err := apps.Get(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		container := docker.Container{ |  | ||||||
| 			App: app, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		status, err := container.Status() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 		if status != "no-container" { |  | ||||||
| 			err = container.Delete() |  | ||||||
| 			if err != nil { |  | ||||||
| 				return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		err = apps.Delete(name) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, Message{Message: "deleted"}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Orphans returns directories in /srv that doesn't match any hosted application
 | 	// Orphans returns directories in /srv that doesn't match any hosted application
 | ||||||
| 	e.GET("/v1/orphans", func(c echo.Context) error { | 	e.GET("/v1/orphans", getOrphansHander) | ||||||
| 		return c.JSON(http.StatusOK, []string{}) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	// Return info about the node including performance index
 | 	// Return info about the node including performance index
 | ||||||
| 	e.GET("/v1/node", func(c echo.Context) error { | 	e.GET("/v1/node", getNodeInfoHandler) | ||||||
| 		node, err := node.GetNodeInfo() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return c.JSONPretty(http.StatusInternalServerError, Message{Message: err.Error()}, JSONIndent) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return c.JSON(http.StatusOK, node) |  | ||||||
| 	}) |  | ||||||
| 
 | 
 | ||||||
| 	e.Logger.Fatal(e.Start("127.0.0.1:1323")) | 	e.Logger.Fatal(e.Start("127.0.0.1:1323")) | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue