mirror of
https://github.com/cuigh/swirl
synced 2025-01-15 07:21:22 +00:00
216 lines
6.7 KiB
Go
216 lines
6.7 KiB
Go
package biz
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/cuigh/auxo/data"
|
|
"github.com/cuigh/auxo/net/web"
|
|
"github.com/cuigh/swirl/docker"
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/api/types/mount"
|
|
)
|
|
|
|
type ContainerBiz interface {
|
|
Search(ctx context.Context, node, name, status string, pageIndex, pageSize int) ([]*Container, int, error)
|
|
Find(ctx context.Context, node, id string) (container *Container, raw string, err error)
|
|
Delete(ctx context.Context, node, id, name string, user web.User) (err error)
|
|
FetchLogs(ctx context.Context, node, id string, lines int, timestamps bool) (stdout, stderr string, err error)
|
|
ExecCreate(ctx context.Context, node, id string, cmd string) (resp types.IDResponse, err error)
|
|
ExecAttach(ctx context.Context, node, id string) (resp types.HijackedResponse, err error)
|
|
ExecStart(ctx context.Context, node, id string) error
|
|
Prune(ctx context.Context, node string, user web.User) (count int, size uint64, err error)
|
|
}
|
|
|
|
func NewContainer(d *docker.Docker, eb EventBiz) ContainerBiz {
|
|
return &containerBiz{d: d, eb: eb}
|
|
}
|
|
|
|
type containerBiz struct {
|
|
d *docker.Docker
|
|
eb EventBiz
|
|
}
|
|
|
|
func (b *containerBiz) Find(ctx context.Context, node, id string) (c *Container, raw string, err error) {
|
|
var (
|
|
cj types.ContainerJSON
|
|
r []byte
|
|
)
|
|
|
|
cj, r, err = b.d.ContainerInspect(ctx, node, id)
|
|
if err != nil {
|
|
if docker.IsErrNotFound(err) {
|
|
err = nil
|
|
}
|
|
return
|
|
}
|
|
|
|
if raw, err = indentJSON(r); err == nil {
|
|
c = newContainerDetail(&cj)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *containerBiz) Search(ctx context.Context, node, name, status string, pageIndex, pageSize int) (containers []*Container, total int, err error) {
|
|
list, total, err := b.d.ContainerList(ctx, node, name, status, pageIndex, pageSize)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
containers = make([]*Container, len(list))
|
|
for i, nr := range list {
|
|
containers[i] = newContainerSummary(&nr)
|
|
}
|
|
return containers, total, nil
|
|
}
|
|
|
|
func (b *containerBiz) Delete(ctx context.Context, node, id, name string, user web.User) (err error) {
|
|
err = b.d.ContainerRemove(ctx, node, id)
|
|
if err == nil {
|
|
b.eb.CreateContainer(EventActionDelete, node, id, name, user)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *containerBiz) ExecCreate(ctx context.Context, node, id, cmd string) (resp types.IDResponse, err error) {
|
|
return b.d.ContainerExecCreate(ctx, node, id, cmd)
|
|
}
|
|
|
|
func (b *containerBiz) ExecAttach(ctx context.Context, node, id string) (resp types.HijackedResponse, err error) {
|
|
return b.d.ContainerExecAttach(ctx, node, id)
|
|
}
|
|
|
|
func (b *containerBiz) ExecStart(ctx context.Context, node, id string) error {
|
|
return b.d.ContainerExecStart(ctx, node, id)
|
|
}
|
|
|
|
func (b *containerBiz) FetchLogs(ctx context.Context, node, id string, lines int, timestamps bool) (string, string, error) {
|
|
stdout, stderr, err := b.d.ContainerLogs(ctx, node, id, lines, timestamps)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
return stdout.String(), stderr.String(), nil
|
|
}
|
|
|
|
func (b *containerBiz) Prune(ctx context.Context, node string, user web.User) (count int, size uint64, err error) {
|
|
var report types.ContainersPruneReport
|
|
if report, err = b.d.ContainerPrune(ctx, node); err == nil {
|
|
count, size = len(report.ContainersDeleted), report.SpaceReclaimed
|
|
b.eb.CreateContainer(EventActionPrune, node, "", "", user)
|
|
}
|
|
return
|
|
}
|
|
|
|
type Container struct {
|
|
ID string `json:"id"`
|
|
Name string `json:"name"`
|
|
Image string `json:"image,omitempty"`
|
|
Command string `json:"command,omitempty"`
|
|
CreatedAt string `json:"createdAt"`
|
|
Ports []*ContainerPort `json:"ports,omitempty"`
|
|
SizeRw int64 `json:"sizeRw"`
|
|
SizeRootFs int64 `json:"sizeRootFs"`
|
|
Labels data.Options `json:"labels"`
|
|
State string `json:"state"`
|
|
Status string `json:"status"`
|
|
NetworkMode string `json:"networkMode"`
|
|
Mounts []*ContainerMount `json:"mounts"`
|
|
PID int `json:"pid,omitempty"`
|
|
StartedAt string `json:"startedAt,omitempty"`
|
|
}
|
|
|
|
type ContainerPort struct {
|
|
IP string `json:"ip,omitempty"`
|
|
PrivatePort uint16 `json:"privatePort,omitempty"`
|
|
PublicPort uint16 `json:"publicPort,omitempty"`
|
|
Type string `json:"type,omitempty"`
|
|
}
|
|
|
|
type ContainerMount struct {
|
|
Type mount.Type `json:"type,omitempty"`
|
|
Name string `json:"name,omitempty"`
|
|
Source string `json:"source,omitempty"`
|
|
Destination string `json:"destination,omitempty"`
|
|
Driver string `json:"driver,omitempty"`
|
|
Mode string `json:"mode,omitempty"`
|
|
RW bool `json:"rw,omitempty"`
|
|
Propagation mount.Propagation `json:"propagation,omitempty"`
|
|
}
|
|
|
|
func newContainerMount(m types.MountPoint) *ContainerMount {
|
|
return &ContainerMount{
|
|
Type: m.Type,
|
|
Name: m.Name,
|
|
Source: m.Source,
|
|
Destination: m.Destination,
|
|
Driver: m.Driver,
|
|
Mode: m.Mode,
|
|
RW: m.RW,
|
|
Propagation: m.Propagation,
|
|
}
|
|
}
|
|
|
|
func newContainerSummary(c *types.Container) *Container {
|
|
container := &Container{
|
|
ID: c.ID,
|
|
Name: c.Names[0],
|
|
Image: normalizeImage(c.Image),
|
|
Command: c.Command,
|
|
CreatedAt: formatTime(time.Unix(c.Created, 0)),
|
|
SizeRw: c.SizeRw,
|
|
SizeRootFs: c.SizeRootFs,
|
|
Labels: mapToOptions(c.Labels),
|
|
State: c.State,
|
|
Status: c.Status,
|
|
NetworkMode: c.HostConfig.NetworkMode,
|
|
}
|
|
for _, p := range c.Ports {
|
|
container.Ports = append(container.Ports, &ContainerPort{
|
|
IP: p.IP,
|
|
PrivatePort: p.PrivatePort,
|
|
PublicPort: p.PublicPort,
|
|
Type: p.Type,
|
|
})
|
|
}
|
|
for _, m := range c.Mounts {
|
|
container.Mounts = append(container.Mounts, newContainerMount(m))
|
|
}
|
|
return container
|
|
}
|
|
|
|
func newContainerDetail(c *types.ContainerJSON) *Container {
|
|
created, _ := time.Parse(time.RFC3339Nano, c.Created)
|
|
startedAt, _ := time.Parse(time.RFC3339Nano, c.State.StartedAt)
|
|
container := &Container{
|
|
ID: c.ID,
|
|
Name: c.Name,
|
|
Image: c.Image,
|
|
//Command: c.Command,
|
|
CreatedAt: formatTime(created),
|
|
Labels: mapToOptions(c.Config.Labels),
|
|
State: c.State.Status,
|
|
//Status: c.Status,
|
|
NetworkMode: string(c.HostConfig.NetworkMode),
|
|
PID: c.State.Pid,
|
|
StartedAt: formatTime(startedAt),
|
|
}
|
|
if c.SizeRw != nil {
|
|
container.SizeRw = *c.SizeRw
|
|
}
|
|
if c.SizeRootFs != nil {
|
|
container.SizeRootFs = *c.SizeRootFs
|
|
}
|
|
//for _, p := range c.Ports {
|
|
// container.Ports = append(container.Ports, &ContainerPort{
|
|
// IP: p.IP,
|
|
// PrivatePort: p.PrivatePort,
|
|
// PublicPort: p.PublicPort,
|
|
// Type: p.Type,
|
|
// })
|
|
//}
|
|
for _, m := range c.Mounts {
|
|
container.Mounts = append(container.Mounts, newContainerMount(m))
|
|
}
|
|
return container
|
|
}
|