swirl/biz/stack.go
2022-01-06 16:54:14 +08:00

193 lines
4.4 KiB
Go

package biz
import (
"context"
"strings"
"github.com/cuigh/auxo/errors"
"github.com/cuigh/auxo/net/web"
"github.com/cuigh/swirl/dao"
"github.com/cuigh/swirl/docker"
"github.com/cuigh/swirl/docker/compose"
"github.com/cuigh/swirl/misc"
)
type StackBiz interface {
Search(ctx context.Context, name, filter string) (stacks []*dao.Stack, err error)
Find(ctx context.Context, name string) (stack *dao.Stack, err error)
Delete(ctx context.Context, name string, user web.User) (err error)
Shutdown(ctx context.Context, name string, user web.User) (err error)
Deploy(ctx context.Context, name string, user web.User) (err error)
Create(ctx context.Context, s *dao.Stack, user web.User) (err error)
Update(ctx context.Context, s *dao.Stack, user web.User) (err error)
}
func NewStack(d *docker.Docker, s dao.Interface, eb EventBiz) StackBiz {
return &stackBiz{d: d, s: s, eb: eb}
}
type stackBiz struct {
d *docker.Docker
s dao.Interface
eb EventBiz
}
func (b *stackBiz) Search(ctx context.Context, name, filter string) (stacks []*dao.Stack, err error) {
var (
activeStacks map[string][]string
internalStacks []*dao.Stack
)
// load real stacks
activeStacks, err = b.d.StackList(ctx)
if err != nil {
return
}
// load stack definitions
internalStacks, err = b.s.StackGetAll(ctx)
if err != nil {
return
}
// merge stacks and definitions
for i := range internalStacks {
stack := internalStacks[i]
if services, ok := activeStacks[stack.Name]; ok {
stack.Services = services
delete(activeStacks, stack.Name)
}
if !b.filter(stack, name, filter) {
stacks = append(stacks, stack)
}
}
for n, services := range activeStacks {
stack := &dao.Stack{Name: n, Services: services}
if !b.filter(stack, name, filter) {
stacks = append(stacks, stack)
}
}
return
}
func (b *stackBiz) Find(ctx context.Context, name string) (s *dao.Stack, err error) {
s, err = b.s.StackGet(ctx, name)
if err != nil {
return nil, err
} else if s == nil {
s = &dao.Stack{Name: name}
}
return
}
func (b *stackBiz) filter(stack *dao.Stack, name, filter string) bool {
if name != "" {
if !strings.Contains(strings.ToLower(stack.Name), strings.ToLower(name)) {
return true
}
}
switch filter {
case "up":
if len(stack.Services) == 0 {
return true
}
case "internal":
if !stack.Internal {
return true
}
case "external":
if stack.Internal {
return true
}
}
return false
}
func (b *stackBiz) Create(ctx context.Context, s *dao.Stack, user web.User) (err error) {
stack := &dao.Stack{
Name: s.Name,
Content: s.Content,
CreatedAt: now(),
CreatedBy: newOperator(user),
}
stack.UpdatedAt = stack.CreatedAt
stack.UpdatedBy = stack.CreatedBy
err = b.s.StackCreate(ctx, stack)
if err == nil {
b.eb.CreateStack(EventActionCreate, stack.Name, user)
}
return
}
func (b *stackBiz) Update(ctx context.Context, s *dao.Stack, user web.User) (err error) {
stack := &dao.Stack{
Name: s.Name,
Content: s.Content,
UpdatedAt: now(),
UpdatedBy: newOperator(user),
}
err = b.s.StackUpdate(ctx, stack)
if err == nil {
b.eb.CreateStack(EventActionUpdate, stack.Name, user)
}
return
}
func (b *stackBiz) Delete(ctx context.Context, name string, user web.User) (err error) {
err = b.d.StackRemove(ctx, name)
if err == nil {
err = b.s.StackDelete(ctx, name)
}
if err == nil {
b.eb.CreateStack(EventActionDelete, name, user)
}
return
}
func (b *stackBiz) Shutdown(ctx context.Context, name string, user web.User) (err error) {
err = b.d.StackRemove(ctx, name)
if err == nil {
b.eb.CreateStack(EventActionShutdown, name, user)
}
return
}
func (b *stackBiz) Deploy(ctx context.Context, name string, user web.User) (err error) {
stack, err := b.s.StackGet(ctx, name)
if err != nil {
return err
} else if stack == nil {
return errors.Coded(misc.ErrExternalStack, "can not deploy external stack")
}
cfg, err := compose.Parse(stack.Name, stack.Content)
if err != nil {
return err
}
registries, err := b.s.RegistryGetAll(ctx)
if err != nil {
return err
}
// Find auth info from registry
authes := map[string]string{}
for _, sc := range cfg.Services {
if _, ok := authes[sc.Image]; !ok {
for _, r := range registries {
if r.Match(sc.Image) {
authes[sc.Image] = r.GetEncodedAuth()
}
}
}
}
err = b.d.StackDeploy(ctx, cfg, authes)
if err == nil {
b.eb.CreateStack(EventActionDeploy, name, user)
}
return
}