2021-01-19 11:49:50 +01:00
|
|
|
// +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"
|
2021-01-29 14:22:22 +01:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
2021-01-19 11:49:50 +01:00
|
|
|
|
|
|
|
"github.com/compose-spec/compose-go/types"
|
2021-06-07 11:21:57 +02:00
|
|
|
"github.com/pkg/errors"
|
2021-01-26 22:42:49 +01:00
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
"github.com/docker/compose-cli/api/compose"
|
2021-01-29 14:22:22 +01:00
|
|
|
apicontext "github.com/docker/compose-cli/api/context"
|
|
|
|
"github.com/docker/compose-cli/api/context/store"
|
2021-01-19 11:49:50 +01:00
|
|
|
"github.com/docker/compose-cli/api/errdefs"
|
2021-01-29 14:22:22 +01:00
|
|
|
"github.com/docker/compose-cli/api/progress"
|
2021-02-04 15:36:57 +01:00
|
|
|
"github.com/docker/compose-cli/kube/client"
|
2021-01-29 14:22:22 +01:00
|
|
|
"github.com/docker/compose-cli/kube/helm"
|
|
|
|
"github.com/docker/compose-cli/kube/resources"
|
2021-02-04 10:04:42 +01:00
|
|
|
"github.com/docker/compose-cli/utils"
|
2021-01-19 11:49:50 +01:00
|
|
|
)
|
|
|
|
|
2021-01-29 14:22:22 +01:00
|
|
|
type composeService struct {
|
2021-02-04 15:36:57 +01:00
|
|
|
sdk *helm.Actions
|
|
|
|
client *client.KubeClient
|
2021-01-29 14:22:22 +01:00
|
|
|
}
|
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
// NewComposeService create a kubernetes implementation of the compose.Service API
|
2021-03-11 11:43:40 +01:00
|
|
|
func NewComposeService() (compose.Service, error) {
|
|
|
|
contextStore := store.Instance()
|
|
|
|
currentContext := apicontext.Current()
|
2021-01-29 14:22:22 +01:00
|
|
|
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)
|
2021-02-04 16:57:06 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-04 15:36:57 +01:00
|
|
|
apiClient, err := client.NewKubeClient(config)
|
2021-01-19 11:49:50 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-04 15:36:57 +01:00
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
return &composeService{
|
2021-02-04 15:36:57 +01:00
|
|
|
sdk: actions,
|
|
|
|
client: apiClient,
|
2021-01-19 11:49:50 +01:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Up executes the equivalent to a `compose up`
|
|
|
|
func (s *composeService) Up(ctx context.Context, project *types.Project, options compose.UpOptions) error {
|
2021-01-29 14:22:22 +01:00
|
|
|
w := progress.ContextWriter(ctx)
|
|
|
|
|
2021-04-20 11:55:29 +02:00
|
|
|
eventName := "Convert Compose file to Helm charts"
|
2021-01-29 14:22:22 +01:00
|
|
|
w.Event(progress.CreatingEvent(eventName))
|
|
|
|
|
|
|
|
chart, err := helm.GetChartInMemory(project)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
w.Event(progress.NewEvent(eventName, progress.Done, ""))
|
|
|
|
|
2021-04-20 11:55:29 +02:00
|
|
|
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))
|
|
|
|
})
|
|
|
|
}
|
2021-03-02 16:19:19 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-04-20 11:55:29 +02:00
|
|
|
|
2021-01-29 14:22:22 +01:00
|
|
|
w.Event(progress.NewEvent(eventName, progress.Done, ""))
|
2021-02-19 20:10:28 +01:00
|
|
|
|
2021-04-22 11:15:48 +02:00
|
|
|
err = s.client.WaitForPodState(ctx, client.WaitForStatusOptions{
|
2021-02-23 12:46:55 +01:00
|
|
|
ProjectName: project.Name,
|
|
|
|
Services: project.ServiceNames(),
|
|
|
|
Status: compose.RUNNING,
|
2021-02-23 13:26:45 +01:00
|
|
|
Log: func(pod string, stateReached bool, message string) {
|
|
|
|
state := progress.Done
|
|
|
|
if !stateReached {
|
|
|
|
state = progress.Working
|
|
|
|
}
|
|
|
|
w.Event(progress.NewEvent(pod, state, message))
|
|
|
|
},
|
2021-02-23 12:46:55 +01:00
|
|
|
})
|
2021-04-23 12:54:51 +02:00
|
|
|
//return err
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-04-22 11:15:48 +02:00
|
|
|
|
2021-04-23 12:54:51 +02:00
|
|
|
// check if there is a port mapping
|
|
|
|
services := map[string]client.Ports{}
|
|
|
|
|
|
|
|
for _, s := range project.Services {
|
|
|
|
if len(s.Ports) > 0 {
|
|
|
|
services[s.Name] = client.Ports{}
|
|
|
|
for _, p := range s.Ports {
|
|
|
|
services[s.Name] = append(services[s.Name], compose.PortPublisher{
|
|
|
|
TargetPort: int(p.Target),
|
|
|
|
PublishedPort: int(p.Published),
|
|
|
|
Protocol: p.Protocol,
|
|
|
|
})
|
2021-04-22 11:15:48 +02:00
|
|
|
}
|
|
|
|
}
|
2021-04-23 12:54:51 +02:00
|
|
|
}
|
|
|
|
if len(services) > 0 {
|
|
|
|
return s.client.MapPorts(ctx, client.PortMappingOptions{
|
|
|
|
ProjectName: project.Name,
|
|
|
|
Services: services,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return nil
|
2021-04-22 11:15:48 +02:00
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Down executes the equivalent to a `compose down`
|
|
|
|
func (s *composeService) Down(ctx context.Context, projectName string, options compose.DownOptions) error {
|
2021-06-07 11:21:57 +02:00
|
|
|
if options.Volumes {
|
|
|
|
return errors.Wrap(errdefs.ErrNotImplemented, "--volumes option is not supported on Kubernetes")
|
|
|
|
}
|
|
|
|
if options.Images != "" {
|
|
|
|
return errors.Wrap(errdefs.ErrNotImplemented, "--rmi option is not supported on Kubernetes")
|
|
|
|
}
|
|
|
|
return progress.Run(ctx, func(ctx context.Context) error {
|
|
|
|
return s.down(ctx, projectName, options)
|
|
|
|
})
|
|
|
|
}
|
2021-01-29 14:22:22 +01:00
|
|
|
|
2021-06-07 11:21:57 +02:00
|
|
|
func (s *composeService) down(ctx context.Context, projectName string, options compose.DownOptions) error {
|
|
|
|
w := progress.ContextWriter(ctx)
|
2021-01-29 14:22:22 +01:00
|
|
|
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)
|
2021-02-23 12:46:55 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-29 14:22:22 +01:00
|
|
|
|
2021-02-23 12:46:55 +01:00
|
|
|
events := []string{}
|
|
|
|
err = s.client.WaitForPodState(ctx, client.WaitForStatusOptions{
|
|
|
|
ProjectName: projectName,
|
|
|
|
Services: nil,
|
|
|
|
Status: compose.REMOVING,
|
2021-02-23 13:26:45 +01:00
|
|
|
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 !utils.StringContains(events, pod) {
|
|
|
|
events = append(events, pod)
|
|
|
|
}
|
|
|
|
},
|
2021-02-23 12:46:55 +01:00
|
|
|
})
|
|
|
|
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
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// List executes the equivalent to a `docker stack ls`
|
2021-02-24 12:06:17 +01:00
|
|
|
func (s *composeService) List(ctx context.Context, opts compose.ListOptions) ([]compose.Stack, error) {
|
2021-01-29 14:22:22 +01:00
|
|
|
return s.sdk.ListReleases()
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Build executes the equivalent to a `compose build`
|
2021-03-02 09:03:04 +01:00
|
|
|
func (s *composeService) Build(ctx context.Context, project *types.Project, options compose.BuildOptions) error {
|
2021-01-19 11:49:50 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
|
|
|
// Push executes the equivalent ot a `compose push`
|
2021-03-05 13:20:13 +01:00
|
|
|
func (s *composeService) Push(ctx context.Context, project *types.Project, options compose.PushOptions) error {
|
2021-01-19 11:49:50 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pull executes the equivalent of a `compose pull`
|
2021-03-22 22:50:46 +01:00
|
|
|
func (s *composeService) Pull(ctx context.Context, project *types.Project, options compose.PullOptions) error {
|
2021-01-19 11:49:50 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create executes the equivalent to a `compose create`
|
|
|
|
func (s *composeService) Create(ctx context.Context, project *types.Project, opts compose.CreateOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start executes the equivalent to a `compose start`
|
2021-02-08 11:04:46 +01:00
|
|
|
func (s *composeService) Start(ctx context.Context, project *types.Project, options compose.StartOptions) error {
|
2021-01-19 11:49:50 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-03-09 22:09:45 +01:00
|
|
|
// Restart executes the equivalent to a `compose restart`
|
|
|
|
func (s *composeService) Restart(ctx context.Context, project *types.Project, options compose.RestartOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-01-26 18:20:00 +01:00
|
|
|
// Stop executes the equivalent to a `compose stop`
|
2021-02-15 13:54:55 +01:00
|
|
|
func (s *composeService) Stop(ctx context.Context, project *types.Project, options compose.StopOptions) error {
|
2021-01-26 18:20:00 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-05-02 03:20:52 +02:00
|
|
|
// Copy copies a file/folder between a service container and the local filesystem
|
|
|
|
func (s *composeService) Copy(ctx context.Context, project *types.Project, options compose.CopyOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
// Logs executes the equivalent to a `compose logs`
|
|
|
|
func (s *composeService) Logs(ctx context.Context, projectName string, consumer compose.LogConsumer, options compose.LogOptions) error {
|
2021-02-04 10:04:42 +01:00
|
|
|
if len(options.Services) > 0 {
|
|
|
|
consumer = utils.FilteredLogConsumer(consumer, options.Services)
|
|
|
|
}
|
|
|
|
return s.client.GetLogs(ctx, projectName, consumer, options.Follow)
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ps executes the equivalent to a `compose ps`
|
2021-01-29 14:41:44 +01:00
|
|
|
func (s *composeService) Ps(ctx context.Context, projectName string, options compose.PsOptions) ([]compose.ContainerSummary, error) {
|
2021-02-04 16:57:06 +01:00
|
|
|
return s.client.GetContainers(ctx, projectName, options.All)
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Convert translate compose model into backend's native format
|
|
|
|
func (s *composeService) Convert(ctx context.Context, project *types.Project, options compose.ConvertOptions) ([]byte, error) {
|
2021-02-05 17:16:20 +01:00
|
|
|
|
2021-02-05 11:12:56 +01:00
|
|
|
chart, err := helm.GetChartInMemory(project)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-05 17:16:20 +01:00
|
|
|
|
|
|
|
if options.Output != "" {
|
2021-02-08 15:16:32 +01:00
|
|
|
_, err := helm.SaveChart(chart, options.Output)
|
|
|
|
return nil, err
|
2021-02-05 17:16:20 +01:00
|
|
|
}
|
|
|
|
|
2021-02-05 11:12:56 +01:00
|
|
|
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
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
|
|
|
|
2021-01-31 18:42:13 +01:00
|
|
|
func (s *composeService) Kill(ctx context.Context, project *types.Project, options compose.KillOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-01-19 11:49:50 +01:00
|
|
|
// RunOneOffContainer creates a service oneoff container and starts its dependencies
|
2021-02-12 17:33:59 +01:00
|
|
|
func (s *composeService) RunOneOffContainer(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) {
|
|
|
|
return 0, errdefs.ErrNotImplemented
|
2021-01-19 11:49:50 +01:00
|
|
|
}
|
2021-02-15 09:13:41 +01:00
|
|
|
|
2021-06-03 13:57:56 +02:00
|
|
|
func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
2021-02-15 09:13:41 +01:00
|
|
|
}
|
2021-02-15 09:56:34 +01:00
|
|
|
|
|
|
|
// Exec executes a command in a running service container
|
2021-04-28 17:59:37 +02:00
|
|
|
func (s *composeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) {
|
2021-04-23 12:54:51 +02:00
|
|
|
return 0, s.client.Exec(ctx, project.Name, opts)
|
2021-02-15 09:56:34 +01:00
|
|
|
}
|
2021-02-22 10:25:40 +01:00
|
|
|
|
2021-04-07 15:00:42 +02:00
|
|
|
func (s *composeService) Pause(ctx context.Context, project string, options compose.PauseOptions) error {
|
2021-02-22 10:25:40 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2021-04-07 15:00:42 +02:00
|
|
|
func (s *composeService) UnPause(ctx context.Context, project string, options compose.PauseOptions) error {
|
2021-02-22 10:25:40 +01:00
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
2021-03-05 12:40:56 +01:00
|
|
|
|
|
|
|
func (s *composeService) Top(ctx context.Context, projectName string, services []string) ([]compose.ContainerProcSummary, error) {
|
|
|
|
return nil, errdefs.ErrNotImplemented
|
|
|
|
}
|
2021-03-08 10:22:24 +01:00
|
|
|
|
2021-03-05 10:09:27 +01:00
|
|
|
func (s *composeService) Events(ctx context.Context, project string, options compose.EventsOptions) error {
|
|
|
|
return errdefs.ErrNotImplemented
|
|
|
|
}
|
2021-03-19 09:49:14 +01:00
|
|
|
|
|
|
|
func (s *composeService) Port(ctx context.Context, project string, service string, port int, options compose.PortOptions) (string, int, error) {
|
|
|
|
return "", 0, errdefs.ErrNotImplemented
|
|
|
|
}
|
2021-04-07 13:16:22 +02:00
|
|
|
|
|
|
|
func (s *composeService) Images(ctx context.Context, projectName string, options compose.ImagesOptions) ([]compose.ImageSummary, error) {
|
|
|
|
return nil, errdefs.ErrNotImplemented
|
|
|
|
}
|