зеркало из https://github.com/docker/compose-cli.git
389 строки
12 KiB
Go
389 строки
12 KiB
Go
//go:build kube
|
|
// +build kube
|
|
|
|
/*
|
|
Copyright 2020 Docker Compose CLI authors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package kube
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/compose-spec/compose-go/types"
|
|
"github.com/docker/compose/v2/pkg/api"
|
|
"github.com/docker/compose/v2/pkg/progress"
|
|
utils2 "github.com/docker/compose/v2/pkg/utils"
|
|
|
|
apicontext "github.com/docker/compose-cli/api/context"
|
|
"github.com/docker/compose-cli/api/context/store"
|
|
"github.com/docker/compose-cli/kube/client"
|
|
"github.com/docker/compose-cli/kube/helm"
|
|
"github.com/docker/compose-cli/kube/resources"
|
|
"github.com/docker/compose-cli/utils"
|
|
)
|
|
|
|
type composeService struct {
|
|
sdk *helm.Actions
|
|
client *client.KubeClient
|
|
}
|
|
|
|
// NewComposeService create a kubernetes implementation of the api.Service API
|
|
func NewComposeService() (api.Service, error) {
|
|
contextStore := store.Instance()
|
|
currentContext := apicontext.Current()
|
|
var kubeContext store.KubeContext
|
|
|
|
if err := contextStore.GetEndpoint(currentContext, &kubeContext); err != nil {
|
|
return nil, err
|
|
}
|
|
config, err := resources.LoadConfig(kubeContext)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
actions, err := helm.NewActions(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
apiClient, err := client.NewKubeClient(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &composeService{
|
|
sdk: actions,
|
|
client: apiClient,
|
|
}, nil
|
|
}
|
|
|
|
// Up executes the equivalent to a `compose up`
|
|
func (s *composeService) Up(ctx context.Context, project *types.Project, options api.UpOptions) error {
|
|
if err := checkUnsupportedUpOptions(ctx, options); err != nil {
|
|
return err
|
|
}
|
|
return progress.Run(ctx, func(ctx context.Context) error {
|
|
return s.up(ctx, project)
|
|
})
|
|
}
|
|
|
|
func checkUnsupportedUpOptions(ctx context.Context, o api.UpOptions) error {
|
|
var errs error
|
|
checks := []struct {
|
|
toCheck, expected interface{}
|
|
option string
|
|
}{
|
|
{o.Create.Inherit, true, "renew-anon-volumes"},
|
|
{o.Create.RemoveOrphans, false, "remove-orphans"},
|
|
{o.Create.QuietPull, false, "quiet-pull"},
|
|
{o.Create.Recreate, api.RecreateDiverged, "force-recreate"},
|
|
{o.Create.RecreateDependencies, api.RecreateDiverged, "always-recreate-deps"},
|
|
{len(o.Start.AttachTo), 0, "attach-dependencies"},
|
|
{len(o.Start.ExitCodeFrom), 0, "exit-code-from"},
|
|
{o.Create.Timeout, nil, "timeout"},
|
|
}
|
|
for _, c := range checks {
|
|
errs = utils.CheckUnsupported(ctx, errs, c.toCheck, c.expected, "up", c.option)
|
|
}
|
|
return errs
|
|
}
|
|
|
|
func (s *composeService) up(ctx context.Context, project *types.Project) error {
|
|
w := progress.ContextWriter(ctx)
|
|
|
|
eventName := "Convert Compose file to Helm charts"
|
|
w.Event(progress.CreatingEvent(eventName))
|
|
|
|
chart, err := helm.GetChartInMemory(project)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
w.Event(progress.NewEvent(eventName, progress.Done, ""))
|
|
|
|
stack, err := s.sdk.Get(project.Name)
|
|
if err != nil || stack == nil {
|
|
// install stack
|
|
eventName = "Install Compose stack"
|
|
w.Event(progress.CreatingEvent(eventName))
|
|
|
|
err = s.sdk.InstallChart(project.Name, chart, func(format string, v ...interface{}) {
|
|
message := fmt.Sprintf(format, v...)
|
|
w.Event(progress.NewEvent(eventName, progress.Done, message))
|
|
})
|
|
|
|
} else {
|
|
// update stack
|
|
eventName = "Updating Compose stack"
|
|
w.Event(progress.CreatingEvent(eventName))
|
|
|
|
err = s.sdk.UpdateChart(project.Name, chart, func(format string, v ...interface{}) {
|
|
message := fmt.Sprintf(format, v...)
|
|
w.Event(progress.NewEvent(eventName, progress.Done, message))
|
|
})
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
w.Event(progress.NewEvent(eventName, progress.Done, ""))
|
|
|
|
return s.client.WaitForPodState(ctx, client.WaitForStatusOptions{
|
|
ProjectName: project.Name,
|
|
Services: project.ServiceNames(),
|
|
Status: api.RUNNING,
|
|
Log: func(pod string, stateReached bool, message string) {
|
|
state := progress.Done
|
|
if !stateReached {
|
|
state = progress.Working
|
|
}
|
|
w.Event(progress.NewEvent(pod, state, message))
|
|
},
|
|
})
|
|
}
|
|
|
|
// Down executes the equivalent to a `compose down`
|
|
func (s *composeService) Down(ctx context.Context, projectName string, options api.DownOptions) error {
|
|
if err := checkUnsupportedDownOptions(ctx, options); err != nil {
|
|
return err
|
|
}
|
|
return progress.Run(ctx, func(ctx context.Context) error {
|
|
return s.down(ctx, projectName, options)
|
|
})
|
|
}
|
|
|
|
func checkUnsupportedDownOptions(ctx context.Context, o api.DownOptions) error {
|
|
var errs error
|
|
checks := []struct {
|
|
toCheck, expected interface{}
|
|
option string
|
|
}{
|
|
{o.Volumes, false, "volumes"},
|
|
{o.Images, "", "images"},
|
|
{o.RemoveOrphans, false, "remove-orphans"},
|
|
}
|
|
for _, c := range checks {
|
|
errs = utils.CheckUnsupported(ctx, errs, c.toCheck, c.expected, "down", c.option)
|
|
}
|
|
return errs
|
|
}
|
|
|
|
func (s *composeService) down(ctx context.Context, projectName string, options api.DownOptions) error {
|
|
w := progress.ContextWriter(ctx)
|
|
eventName := fmt.Sprintf("Remove %s", projectName)
|
|
w.Event(progress.CreatingEvent(eventName))
|
|
|
|
logger := func(format string, v ...interface{}) {
|
|
message := fmt.Sprintf(format, v...)
|
|
if strings.Contains(message, "Starting delete") {
|
|
action := strings.Replace(message, "Starting delete for", "Delete", 1)
|
|
|
|
w.Event(progress.CreatingEvent(action))
|
|
w.Event(progress.NewEvent(action, progress.Done, ""))
|
|
return
|
|
}
|
|
w.Event(progress.NewEvent(eventName, progress.Working, message))
|
|
}
|
|
err := s.sdk.Uninstall(projectName, logger)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
events := []string{}
|
|
err = s.client.WaitForPodState(ctx, client.WaitForStatusOptions{
|
|
ProjectName: projectName,
|
|
Services: nil,
|
|
Status: api.REMOVING,
|
|
Timeout: options.Timeout,
|
|
Log: func(pod string, stateReached bool, message string) {
|
|
state := progress.Done
|
|
if !stateReached {
|
|
state = progress.Working
|
|
}
|
|
w.Event(progress.NewEvent(pod, state, message))
|
|
if !utils2.StringContains(events, pod) {
|
|
events = append(events, pod)
|
|
}
|
|
},
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, e := range events {
|
|
w.Event(progress.NewEvent(e, progress.Done, ""))
|
|
}
|
|
w.Event(progress.NewEvent(eventName, progress.Done, ""))
|
|
return nil
|
|
}
|
|
|
|
// List executes the equivalent to a `docker stack ls`
|
|
func (s *composeService) List(ctx context.Context, opts api.ListOptions) ([]api.Stack, error) {
|
|
if err := checkUnsupportedListOptions(ctx, opts); err != nil {
|
|
return nil, err
|
|
}
|
|
return s.sdk.ListReleases()
|
|
}
|
|
|
|
func checkUnsupportedListOptions(ctx context.Context, o api.ListOptions) error {
|
|
return utils.CheckUnsupported(ctx, nil, o.All, false, "ls", "all")
|
|
}
|
|
|
|
// Build executes the equivalent to a `compose build`
|
|
func (s *composeService) Build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Push executes the equivalent ot a `compose push`
|
|
func (s *composeService) Push(ctx context.Context, project *types.Project, options api.PushOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Pull executes the equivalent of a `compose pull`
|
|
func (s *composeService) Pull(ctx context.Context, project *types.Project, options api.PullOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Create executes the equivalent to a `compose create`
|
|
func (s *composeService) Create(ctx context.Context, project *types.Project, opts api.CreateOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Start executes the equivalent to a `compose start`
|
|
func (s *composeService) Start(ctx context.Context, project *types.Project, options api.StartOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Restart executes the equivalent to a `compose restart`
|
|
func (s *composeService) Restart(ctx context.Context, project *types.Project, options api.RestartOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Stop executes the equivalent to a `compose stop`
|
|
func (s *composeService) Stop(ctx context.Context, project *types.Project, options api.StopOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Copy copies a file/folder between a service container and the local filesystem
|
|
func (s *composeService) Copy(ctx context.Context, project *types.Project, options api.CopyOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Logs executes the equivalent to a `compose logs`
|
|
func (s *composeService) Logs(ctx context.Context, projectName string, consumer api.LogConsumer, options api.LogOptions) error {
|
|
if err := checkUnsupportedLogOptions(ctx, options); err != nil {
|
|
return err
|
|
}
|
|
if len(options.Services) > 0 {
|
|
consumer = utils.FilteredLogConsumer(consumer, options.Services)
|
|
}
|
|
return s.client.GetLogs(ctx, projectName, consumer, options.Follow)
|
|
}
|
|
|
|
func checkUnsupportedLogOptions(ctx context.Context, o api.LogOptions) error {
|
|
var errs error
|
|
checks := []struct {
|
|
toCheck, expected interface{}
|
|
option string
|
|
}{
|
|
{o.Since, "", "since"},
|
|
{o.Timestamps, false, "timestamps"},
|
|
{o.Until, "", "until"},
|
|
}
|
|
for _, c := range checks {
|
|
errs = utils.CheckUnsupported(ctx, errs, c.toCheck, c.expected, "logs", c.option)
|
|
}
|
|
return errs
|
|
}
|
|
|
|
// Ps executes the equivalent to a `compose ps`
|
|
func (s *composeService) Ps(ctx context.Context, projectName string, options api.PsOptions) ([]api.ContainerSummary, error) {
|
|
return s.client.GetContainers(ctx, projectName, options.All)
|
|
}
|
|
|
|
// Convert translate compose model into backend's native format
|
|
func (s *composeService) Convert(ctx context.Context, project *types.Project, options api.ConvertOptions) ([]byte, error) {
|
|
chart, err := helm.GetChartInMemory(project)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if options.Output != "" {
|
|
_, err := helm.SaveChart(chart, options.Output)
|
|
return nil, err
|
|
}
|
|
|
|
buff := []byte{}
|
|
for _, f := range chart.Raw {
|
|
header := "\n" + f.Name + "\n" + strings.Repeat("-", len(f.Name)) + "\n"
|
|
buff = append(buff, []byte(header)...)
|
|
buff = append(buff, f.Data...)
|
|
buff = append(buff, []byte("\n")...)
|
|
}
|
|
return buff, nil
|
|
}
|
|
|
|
func (s *composeService) Kill(ctx context.Context, project *types.Project, options api.KillOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// RunOneOffContainer creates a service oneoff container and starts its dependencies
|
|
func (s *composeService) RunOneOffContainer(ctx context.Context, project *types.Project, opts api.RunOptions) (int, error) {
|
|
return 0, api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) Remove(ctx context.Context, project *types.Project, options api.RemoveOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
// Exec executes a command in a running service container
|
|
func (s *composeService) Exec(ctx context.Context, project string, opts api.RunOptions) (int, error) {
|
|
if err := checkUnsupportedExecOptions(ctx, opts); err != nil {
|
|
return 0, err
|
|
}
|
|
return 0, s.client.Exec(ctx, project, opts)
|
|
}
|
|
|
|
func checkUnsupportedExecOptions(ctx context.Context, o api.RunOptions) error {
|
|
var errs error
|
|
errs = utils.CheckUnsupported(ctx, errs, o.Index, 0, "exec", "index")
|
|
errs = utils.CheckUnsupported(ctx, errs, o.Privileged, false, "exec", "privileged")
|
|
errs = utils.CheckUnsupported(ctx, errs, o.WorkingDir, "", "exec", "workdir")
|
|
return errs
|
|
}
|
|
|
|
func (s *composeService) Pause(ctx context.Context, project string, options api.PauseOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) UnPause(ctx context.Context, project string, options api.PauseOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) Top(ctx context.Context, projectName string, services []string) ([]api.ContainerProcSummary, error) {
|
|
return nil, api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) Events(ctx context.Context, project string, options api.EventsOptions) error {
|
|
return api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) Port(ctx context.Context, project string, service string, port int, options api.PortOptions) (string, int, error) {
|
|
return "", 0, api.ErrNotImplemented
|
|
}
|
|
|
|
func (s *composeService) Images(ctx context.Context, projectName string, options api.ImagesOptions) ([]api.ImageSummary, error) {
|
|
return nil, api.ErrNotImplemented
|
|
}
|