mirror of
https://github.com/cuigh/swirl
synced 2025-01-05 02:23:22 +00:00
403 lines
11 KiB
Go
403 lines
11 KiB
Go
package docker
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/cuigh/auxo/errors"
|
|
"github.com/cuigh/swirl/biz/docker/compose"
|
|
"github.com/cuigh/swirl/model"
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/api/types/container"
|
|
"github.com/docker/docker/api/types/filters"
|
|
"github.com/docker/docker/api/types/swarm"
|
|
"github.com/docker/docker/client"
|
|
)
|
|
|
|
const stackLabel = "com.docker.stack.namespace"
|
|
|
|
// StackList return all stacks.
|
|
func StackList() (stacks []*model.StackListInfo, err error) {
|
|
err = mgr.Do(func(ctx context.Context, cli *client.Client) (err error) {
|
|
var services []swarm.Service
|
|
opts := types.ServiceListOptions{
|
|
Filters: filters.NewArgs(),
|
|
}
|
|
opts.Filters.Add("label", stackLabel)
|
|
services, err = cli.ServiceList(ctx, opts)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
m := make(map[string]*model.StackListInfo)
|
|
for _, service := range services {
|
|
labels := service.Spec.Labels
|
|
name, ok := labels[stackLabel]
|
|
if !ok {
|
|
err = fmt.Errorf("cannot get label %s for service %s(%s)", stackLabel, service.Spec.Name, service.ID)
|
|
return
|
|
}
|
|
|
|
if stack, ok := m[name]; ok {
|
|
stack.Services = append(stack.Services, service.Spec.Name)
|
|
} else {
|
|
m[name] = &model.StackListInfo{
|
|
Name: name,
|
|
Services: []string{service.Spec.Name},
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, stack := range m {
|
|
stacks = append(stacks, stack)
|
|
}
|
|
sort.Slice(stacks, func(i, j int) bool {
|
|
return stacks[i].Name < stacks[j].Name
|
|
})
|
|
return
|
|
})
|
|
return
|
|
}
|
|
|
|
// StackCount return number of stacks.
|
|
func StackCount() (count int, err error) {
|
|
err = mgr.Do(func(ctx context.Context, cli *client.Client) (err error) {
|
|
var services []swarm.Service
|
|
opts := types.ServiceListOptions{
|
|
Filters: filters.NewArgs(),
|
|
}
|
|
opts.Filters.Add("label", stackLabel)
|
|
services, err = cli.ServiceList(ctx, opts)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
m := make(map[string]struct{})
|
|
for _, service := range services {
|
|
labels := service.Spec.Labels
|
|
if name, ok := labels[stackLabel]; ok {
|
|
m[name] = struct{}{}
|
|
} else {
|
|
mgr.Logger().Warnf("cannot get label %s for service %s(%s)", stackLabel, service.Spec.Name, service.ID)
|
|
}
|
|
}
|
|
count = len(m)
|
|
return
|
|
})
|
|
return
|
|
}
|
|
|
|
// StackRemove remove a stack.
|
|
func StackRemove(name string) error {
|
|
return mgr.Do(func(ctx context.Context, cli *client.Client) (err error) {
|
|
var (
|
|
services []swarm.Service
|
|
networks []types.NetworkResource
|
|
secrets []swarm.Secret
|
|
configs []swarm.Config
|
|
errs []error
|
|
)
|
|
|
|
args := filters.NewArgs()
|
|
args.Add("label", stackLabel+"="+name)
|
|
|
|
services, err = cli.ServiceList(ctx, types.ServiceListOptions{Filters: args})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
networks, err = cli.NetworkList(ctx, types.NetworkListOptions{Filters: args})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// API version >= 1.25
|
|
secrets, err = cli.SecretList(ctx, types.SecretListOptions{Filters: args})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// API version >= 1.30
|
|
configs, err = cli.ConfigList(ctx, types.ConfigListOptions{Filters: args})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
if len(services)+len(networks)+len(secrets)+len(configs) == 0 {
|
|
return fmt.Errorf("nothing found in stack: %s", name)
|
|
}
|
|
|
|
// Remove services
|
|
for _, service := range services {
|
|
if err = cli.ServiceRemove(ctx, service.ID); err != nil {
|
|
e := errors.Format("Failed to remove service %s: %s", service.Spec.Name, err)
|
|
errs = append(errs, e)
|
|
mgr.Logger().Warn(e)
|
|
}
|
|
}
|
|
|
|
// Remove secrets
|
|
for _, secret := range secrets {
|
|
if err = cli.SecretRemove(ctx, secret.ID); err != nil {
|
|
e := errors.Format("Failed to remove secret %s: %s", secret.Spec.Name, err)
|
|
errs = append(errs, e)
|
|
mgr.Logger().Warn(e)
|
|
}
|
|
}
|
|
|
|
// Remove configs
|
|
for _, config := range configs {
|
|
if err = cli.ConfigRemove(ctx, config.ID); err != nil {
|
|
e := errors.Format("Failed to remove config %s: %s", config.Spec.Name, err)
|
|
errs = append(errs, e)
|
|
mgr.Logger().Warn(e)
|
|
}
|
|
}
|
|
|
|
// Remove networks
|
|
for _, network := range networks {
|
|
if err = cli.NetworkRemove(ctx, network.ID); err != nil {
|
|
e := errors.Format("Failed to remove network %s: %s", network.Name, err)
|
|
errs = append(errs, e)
|
|
mgr.Logger().Warn(e)
|
|
}
|
|
}
|
|
|
|
if len(errs) > 0 {
|
|
return errors.List(errs...)
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
// StackDeploy deploy a stack.
|
|
func StackDeploy(name, content string, authes map[string]string) error {
|
|
ctx, cli, err := mgr.Client()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cfg, err := compose.Parse(name, content)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
namespace := compose.NewNamespace(name)
|
|
|
|
serviceNetworks := compose.GetServicesDeclaredNetworks(cfg.Services)
|
|
networks, externalNetworks := compose.Networks(namespace, cfg.Networks, serviceNetworks)
|
|
if err = validateExternalNetworks(ctx, cli, externalNetworks); err != nil {
|
|
return err
|
|
}
|
|
if err = createNetworks(ctx, cli, namespace, networks); err != nil {
|
|
return err
|
|
}
|
|
|
|
secrets, err := compose.Secrets(namespace, cfg.Secrets)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = createSecrets(ctx, cli, secrets); err != nil {
|
|
return err
|
|
}
|
|
|
|
configs, err := compose.Configs(namespace, cfg.Configs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = createConfigs(ctx, cli, configs); err != nil {
|
|
return err
|
|
}
|
|
|
|
services, err := compose.Services(namespace, cfg, cli)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return deployServices(ctx, cli, services, namespace, authes)
|
|
}
|
|
|
|
func validateExternalNetworks(ctx context.Context, cli *client.Client, externalNetworks []string) error {
|
|
for _, networkName := range externalNetworks {
|
|
if !container.NetworkMode(networkName).IsUserDefined() {
|
|
// Networks that are not user defined always exist on all nodes as
|
|
// local-scoped networks, so there's no need to inspect them.
|
|
continue
|
|
}
|
|
network, err := cli.NetworkInspect(ctx, networkName, types.NetworkInspectOptions{})
|
|
switch {
|
|
case client.IsErrNotFound(err):
|
|
return errors.Format("network %q is declared as external, but could not be found. You need to create a swarm-scoped network before the stack is deployed", networkName)
|
|
case err != nil:
|
|
return err
|
|
case network.Scope != "swarm":
|
|
return errors.Format("network %q is declared as external, but it is not in the right scope: %q instead of \"swarm\"", networkName, network.Scope)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func createNetworks(ctx context.Context, cli *client.Client, namespace compose.Namespace, networks map[string]types.NetworkCreate) error {
|
|
opts := types.NetworkListOptions{
|
|
Filters: filters.NewArgs(),
|
|
}
|
|
opts.Filters.Add("label", stackLabel+"="+namespace.Name())
|
|
existingNetworks, err := cli.NetworkList(ctx, opts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
existingNetworkMap := make(map[string]types.NetworkResource)
|
|
for _, network := range existingNetworks {
|
|
existingNetworkMap[network.Name] = network
|
|
}
|
|
|
|
for internalName, createOpts := range networks {
|
|
name := namespace.Scope(internalName)
|
|
if _, exists := existingNetworkMap[name]; exists {
|
|
continue
|
|
}
|
|
|
|
if createOpts.Driver == "" {
|
|
createOpts.Driver = "overlay"
|
|
}
|
|
|
|
mgr.Logger().Infof("Creating network %s", name)
|
|
if _, err = cli.NetworkCreate(ctx, name, createOpts); err != nil {
|
|
return errors.Wrap(err, "failed to create network "+internalName)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func createSecrets(ctx context.Context, cli *client.Client, secrets []swarm.SecretSpec) error {
|
|
for _, secretSpec := range secrets {
|
|
secret, _, err := cli.SecretInspectWithRaw(ctx, secretSpec.Name)
|
|
switch {
|
|
case err == nil:
|
|
// secret already exists, then we update that
|
|
if err = cli.SecretUpdate(ctx, secret.ID, secret.Meta.Version, secretSpec); err != nil {
|
|
return errors.Wrap(err, "failed to update secret "+secretSpec.Name)
|
|
}
|
|
case client.IsErrNotFound(err):
|
|
// secret does not exist, then we create a new one.
|
|
if _, err = cli.SecretCreate(ctx, secretSpec); err != nil {
|
|
return errors.Wrap(err, "failed to create secret "+secretSpec.Name)
|
|
}
|
|
default:
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func createConfigs(ctx context.Context, cli *client.Client, configs []swarm.ConfigSpec) error {
|
|
for _, configSpec := range configs {
|
|
config, _, err := cli.ConfigInspectWithRaw(ctx, configSpec.Name)
|
|
switch {
|
|
case err == nil:
|
|
// config already exists, then we update that
|
|
if err = cli.ConfigUpdate(ctx, config.ID, config.Meta.Version, configSpec); err != nil {
|
|
errors.Wrap(err, "failed to update config "+configSpec.Name)
|
|
}
|
|
case client.IsErrNotFound(err):
|
|
// config does not exist, then we create a new one.
|
|
if _, err = cli.ConfigCreate(ctx, configSpec); err != nil {
|
|
errors.Wrap(err, "failed to create config "+configSpec.Name)
|
|
}
|
|
default:
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func getServices(
|
|
ctx context.Context,
|
|
cli *client.Client,
|
|
namespace string,
|
|
) ([]swarm.Service, error) {
|
|
opts := types.ServiceListOptions{
|
|
Filters: filters.NewArgs(),
|
|
}
|
|
opts.Filters.Add("label", stackLabel+"="+namespace)
|
|
return cli.ServiceList(ctx, opts)
|
|
}
|
|
|
|
func deployServices(
|
|
ctx context.Context,
|
|
cli *client.Client,
|
|
services map[string]swarm.ServiceSpec,
|
|
namespace compose.Namespace,
|
|
authes map[string]string,
|
|
//sendAuth bool,
|
|
//resolveImage string,
|
|
) error {
|
|
existingServices, err := getServices(ctx, cli, namespace.Name())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
existingServiceMap := make(map[string]swarm.Service)
|
|
for _, service := range existingServices {
|
|
existingServiceMap[service.Spec.Name] = service
|
|
}
|
|
|
|
for internalName, serviceSpec := range services {
|
|
name := namespace.Scope(internalName)
|
|
|
|
// TODO: Add auth
|
|
encodedAuth := authes[serviceSpec.TaskTemplate.ContainerSpec.Image]
|
|
//image := serviceSpec.TaskTemplate.ContainerSpec.Image
|
|
//if sendAuth {
|
|
// // Retrieve encoded auth token from the image reference
|
|
// encodedAuth, err = command.RetrieveAuthTokenFromImage(ctx, dockerCli, image)
|
|
// if err != nil {
|
|
// return err
|
|
// }
|
|
//}
|
|
|
|
if service, exists := existingServiceMap[name]; exists {
|
|
mgr.Logger().Infof("Updating service %s (id: %s)", name, service.ID)
|
|
|
|
updateOpts := types.ServiceUpdateOptions{
|
|
RegistryAuthFrom: types.RegistryAuthFromSpec,
|
|
EncodedRegistryAuth: encodedAuth,
|
|
}
|
|
|
|
//if resolveImage == resolveImageAlways || (resolveImage == resolveImageChanged && image != service.Spec.Labels[compose.LabelImage]) {
|
|
// updateOpts.QueryRegistry = true
|
|
//}
|
|
|
|
response, err := cli.ServiceUpdate(
|
|
ctx,
|
|
service.ID,
|
|
service.Version,
|
|
serviceSpec,
|
|
updateOpts,
|
|
)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to update service "+name)
|
|
}
|
|
|
|
for _, warning := range response.Warnings {
|
|
mgr.Logger().Warn(warning)
|
|
}
|
|
} else {
|
|
mgr.Logger().Infof("Creating service %s", name)
|
|
|
|
createOpts := types.ServiceCreateOptions{EncodedRegistryAuth: encodedAuth}
|
|
|
|
// query registry if flag disabling it was not set
|
|
//if resolveImage == resolveImageAlways || resolveImage == resolveImageChanged {
|
|
// createOpts.QueryRegistry = true
|
|
//}
|
|
|
|
if _, err = cli.ServiceCreate(ctx, serviceSpec, createOpts); err != nil {
|
|
return errors.Wrap(err, "failed to create service "+name)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|