2021-12-06 12:24:22 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2021-12-24 03:50:21 +00:00
|
|
|
"net/http"
|
2021-12-06 12:24:22 +00:00
|
|
|
|
|
|
|
"github.com/cuigh/auxo/data"
|
|
|
|
"github.com/cuigh/auxo/log"
|
|
|
|
"github.com/cuigh/auxo/net/web"
|
|
|
|
"github.com/cuigh/swirl/biz"
|
2022-01-06 08:54:14 +00:00
|
|
|
"github.com/cuigh/swirl/misc"
|
2021-12-06 12:24:22 +00:00
|
|
|
"github.com/gobwas/ws"
|
|
|
|
"github.com/gobwas/ws/wsutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ContainerHandler encapsulates container related handlers.
|
|
|
|
type ContainerHandler struct {
|
|
|
|
Search web.HandlerFunc `path:"/search" auth:"container.view" desc:"search containers"`
|
|
|
|
Find web.HandlerFunc `path:"/find" auth:"container.view" desc:"find container by name"`
|
|
|
|
Delete web.HandlerFunc `path:"/delete" method:"post" auth:"container.delete" desc:"delete container"`
|
|
|
|
FetchLogs web.HandlerFunc `path:"/fetch-logs" auth:"container.logs" desc:"fetch logs of container"`
|
2021-12-22 09:43:26 +00:00
|
|
|
Connect web.HandlerFunc `path:"/connect" auth:"container.execute" desc:"connect to a running container"`
|
2021-12-23 07:25:51 +00:00
|
|
|
Prune web.HandlerFunc `path:"/prune" method:"post" auth:"container.delete" desc:"delete unused containers"`
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewContainer creates an instance of ContainerHandler
|
|
|
|
func NewContainer(b biz.ContainerBiz) *ContainerHandler {
|
|
|
|
return &ContainerHandler{
|
|
|
|
Search: containerSearch(b),
|
|
|
|
Find: containerFind(b),
|
|
|
|
Delete: containerDelete(b),
|
|
|
|
FetchLogs: containerFetchLogs(b),
|
|
|
|
Connect: containerConnect(b),
|
2021-12-23 07:25:51 +00:00
|
|
|
Prune: containerPrune(b),
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func containerSearch(b biz.ContainerBiz) web.HandlerFunc {
|
|
|
|
type Args struct {
|
2021-12-17 12:13:58 +00:00
|
|
|
Node string `json:"node" bind:"node"`
|
2021-12-06 12:24:22 +00:00
|
|
|
Name string `json:"name" bind:"name"`
|
|
|
|
Status string `json:"status" bind:"status"`
|
|
|
|
PageIndex int `json:"pageIndex" bind:"pageIndex"`
|
|
|
|
PageSize int `json:"pageSize" bind:"pageSize"`
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) (err error) {
|
2021-12-06 12:24:22 +00:00
|
|
|
var (
|
|
|
|
args = &Args{}
|
|
|
|
containers []*biz.Container
|
|
|
|
total int
|
|
|
|
)
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
if err = c.Bind(args); err == nil {
|
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
containers, total, err = b.Search(ctx, args.Node, args.Name, args.Status, args.PageIndex, args.PageSize)
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
return success(c, data.Map{
|
2021-12-06 12:24:22 +00:00
|
|
|
"items": containers,
|
|
|
|
"total": total,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func containerFind(b biz.ContainerBiz) web.HandlerFunc {
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) error {
|
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
node := c.Query("node")
|
|
|
|
id := c.Query("id")
|
|
|
|
container, raw, err := b.Find(ctx, node, id)
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2021-12-24 11:52:29 +00:00
|
|
|
} else if container == nil {
|
|
|
|
return web.NewError(http.StatusNotFound)
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
2022-01-06 08:54:14 +00:00
|
|
|
return success(c, data.Map{"container": container, "raw": raw})
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func containerDelete(b biz.ContainerBiz) web.HandlerFunc {
|
|
|
|
type Args struct {
|
2021-12-17 12:13:58 +00:00
|
|
|
Node string `json:"node"`
|
|
|
|
ID string `json:"id"`
|
2021-12-23 07:25:51 +00:00
|
|
|
Name string `json:"name"`
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) (err error) {
|
2021-12-06 12:24:22 +00:00
|
|
|
args := &Args{}
|
2022-01-06 08:54:14 +00:00
|
|
|
if err = c.Bind(args); err == nil {
|
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
err = b.Delete(ctx, args.Node, args.ID, args.Name, c.User())
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
2022-01-06 08:54:14 +00:00
|
|
|
return ajax(c, err)
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func containerFetchLogs(b biz.ContainerBiz) web.HandlerFunc {
|
|
|
|
type Args struct {
|
2021-12-17 12:13:58 +00:00
|
|
|
Node string `json:"node" bind:"node"`
|
2021-12-06 12:24:22 +00:00
|
|
|
ID string `json:"id" bind:"id"`
|
|
|
|
Lines int `json:"lines" bind:"lines"`
|
|
|
|
Timestamps bool `json:"timestamps" bind:"timestamps"`
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) (err error) {
|
2021-12-06 12:24:22 +00:00
|
|
|
var (
|
|
|
|
args = &Args{}
|
|
|
|
stdout, stderr string
|
|
|
|
)
|
2022-01-06 08:54:14 +00:00
|
|
|
if err = c.Bind(args); err == nil {
|
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
stdout, stderr, err = b.FetchLogs(ctx, args.Node, args.ID, args.Lines, args.Timestamps)
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-01-06 08:54:14 +00:00
|
|
|
return success(c, data.Map{"stdout": stdout, "stderr": stderr})
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func containerConnect(b biz.ContainerBiz) web.HandlerFunc {
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) error {
|
2021-12-06 12:24:22 +00:00
|
|
|
var (
|
2022-01-06 08:54:14 +00:00
|
|
|
node = c.Query("node")
|
|
|
|
id = c.Query("id")
|
|
|
|
cmd = c.Query("cmd")
|
2021-12-06 12:24:22 +00:00
|
|
|
)
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
container, _, err := b.Find(ctx, node, id)
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2021-12-24 11:52:29 +00:00
|
|
|
} else if container == nil {
|
|
|
|
return web.NewError(http.StatusNotFound)
|
2021-12-06 12:24:22 +00:00
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
conn, _, _, err := ws.UpgradeHTTP(c.Request(), c.Response())
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
idResp, err := b.ExecCreate(ctx, node, id, cmd)
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
resp, err := b.ExecAttach(ctx, node, idResp.ID)
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
err = b.ExecStart(ctx, node, idResp.ID)
|
2021-12-06 12:24:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
closed = false
|
|
|
|
logger = log.Get("container")
|
|
|
|
disposer = func() {
|
|
|
|
if !closed {
|
|
|
|
closed = true
|
|
|
|
_ = conn.Close()
|
|
|
|
resp.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// input
|
|
|
|
go func() {
|
|
|
|
defer disposer()
|
|
|
|
|
|
|
|
var (
|
|
|
|
msg []byte
|
|
|
|
op ws.OpCode
|
|
|
|
)
|
|
|
|
|
|
|
|
for {
|
|
|
|
msg, op, err = wsutil.ReadClientData(conn)
|
|
|
|
if err != nil {
|
|
|
|
if !closed {
|
|
|
|
logger.Error("failed to read data from client: ", err)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if op == ws.OpClose {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = resp.Conn.Write(msg)
|
|
|
|
if err != nil {
|
|
|
|
logger.Error("failed to write data to container: ", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// output
|
|
|
|
go func() {
|
|
|
|
defer disposer()
|
|
|
|
|
|
|
|
var (
|
|
|
|
n int
|
|
|
|
buf = make([]byte, 1024)
|
|
|
|
)
|
|
|
|
|
|
|
|
for {
|
|
|
|
n, err = resp.Reader.Read(buf)
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
} else if err != nil {
|
|
|
|
logger.Error("failed to read data from container: ", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
err = wsutil.WriteServerMessage(conn, ws.OpText, buf[:n])
|
|
|
|
if err != nil {
|
|
|
|
logger.Error("failed to write data to client: ", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2021-12-23 07:25:51 +00:00
|
|
|
|
|
|
|
func containerPrune(b biz.ContainerBiz) web.HandlerFunc {
|
|
|
|
type Args struct {
|
|
|
|
Node string `json:"node"`
|
|
|
|
}
|
2022-01-06 08:54:14 +00:00
|
|
|
return func(c web.Context) (err error) {
|
2021-12-23 07:25:51 +00:00
|
|
|
args := &Args{}
|
2022-01-06 08:54:14 +00:00
|
|
|
if err = c.Bind(args); err != nil {
|
2021-12-23 07:25:51 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
ctx, cancel := misc.Context(defaultTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
count, size, err := b.Prune(ctx, args.Node, c.User())
|
2021-12-23 07:25:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-06 08:54:14 +00:00
|
|
|
return success(c, data.Map{
|
2021-12-23 07:25:51 +00:00
|
|
|
"count": count,
|
|
|
|
"size": size,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|