| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277 |
- // +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() (compose.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 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, options compose.PushOptions) error {
- return errdefs.ErrNotImplemented
- }
- // Pull executes the equivalent of a `compose pull`
- func (s *composeService) Pull(ctx context.Context, project *types.Project, options compose.PullOptions) 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
- }
- // 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
- }
- // 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
- }
- func (s *composeService) Top(ctx context.Context, projectName string, services []string) ([]compose.ContainerProcSummary, error) {
- return nil, errdefs.ErrNotImplemented
- }
- func (s *composeService) Events(ctx context.Context, project string, options compose.EventsOptions) error {
- return errdefs.ErrNotImplemented
- }
- func (s *composeService) Port(ctx context.Context, project string, service string, port int, options compose.PortOptions) (string, int, error) {
- return "", 0, errdefs.ErrNotImplemented
- }
- func (s *composeService) Images(ctx context.Context, projectName string, options compose.ImagesOptions) ([]compose.ImageSummary, error) {
- return nil, errdefs.ErrNotImplemented
- }
|