| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 | 
							- // +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-cli/api/compose"
 
- 	apicontext "github.com/docker/compose-cli/api/context"
 
- 	"github.com/docker/compose-cli/api/context/store"
 
- 	"github.com/docker/compose-cli/api/errdefs"
 
- 	"github.com/docker/compose-cli/api/progress"
 
- 	"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 compose.Service API
 
- func NewComposeService(ctx context.Context) (compose.Service, error) {
 
- 	contextStore := store.ContextStore(ctx)
 
- 	currentContext := apicontext.CurrentContext(ctx)
 
- 	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 compose.UpOptions) error {
 
- 	w := progress.ContextWriter(ctx)
 
- 	eventName := "Convert 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, ""))
 
- 	eventName = "Install Helm charts"
 
- 	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))
 
- 	})
 
- 	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:      compose.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 compose.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:      compose.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 !utils.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 compose.ListOptions) ([]compose.Stack, error) {
 
- 	return s.sdk.ListReleases()
 
- }
 
- // Build executes the equivalent to a `compose build`
 
- func (s *composeService) Build(ctx context.Context, project *types.Project, options compose.BuildOptions) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- // Push executes the equivalent ot a `compose push`
 
- func (s *composeService) Push(ctx context.Context, project *types.Project) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- // Pull executes the equivalent of a `compose pull`
 
- func (s *composeService) Pull(ctx context.Context, project *types.Project) error {
 
- 	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`
 
- func (s *composeService) Start(ctx context.Context, project *types.Project, options compose.StartOptions) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- // Stop executes the equivalent to a `compose stop`
 
- func (s *composeService) Stop(ctx context.Context, project *types.Project, options compose.StopOptions) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- // Logs executes the equivalent to a `compose logs`
 
- func (s *composeService) Logs(ctx context.Context, projectName string, consumer compose.LogConsumer, options compose.LogOptions) error {
 
- 	if len(options.Services) > 0 {
 
- 		consumer = utils.FilteredLogConsumer(consumer, options.Services)
 
- 	}
 
- 	return s.client.GetLogs(ctx, projectName, consumer, options.Follow)
 
- }
 
- // Ps executes the equivalent to a `compose ps`
 
- func (s *composeService) Ps(ctx context.Context, projectName string, options compose.PsOptions) ([]compose.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 compose.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 compose.KillOptions) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- // RunOneOffContainer creates a service oneoff container and starts its dependencies
 
- func (s *composeService) RunOneOffContainer(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) {
 
- 	return 0, errdefs.ErrNotImplemented
 
- }
 
- func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) {
 
- 	return nil, errdefs.ErrNotImplemented
 
- }
 
- // Exec executes a command in a running service container
 
- func (s *composeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- func (s *composeService) Pause(ctx context.Context, project *types.Project) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
- func (s *composeService) UnPause(ctx context.Context, project *types.Project) error {
 
- 	return errdefs.ErrNotImplemented
 
- }
 
 
  |