| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293 |
- /*
- 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 compose
- import (
- "bytes"
- "fmt"
- "strings"
- "github.com/compose-spec/compose-go/v2/types"
- "github.com/docker/cli/cli/config/configfile"
- )
- // --use-api-socket is not actually supported by the Docker Engine
- // but is a client-side hack (see https://github.com/docker/cli/blob/master/cli/command/container/create.go#L246)
- // we replicate here by transforming the project model
- func (s *composeService) useAPISocket(project *types.Project) (*types.Project, error) {
- useAPISocket := false
- for _, service := range project.Services {
- if service.UseAPISocket {
- useAPISocket = true
- break
- }
- }
- if !useAPISocket {
- return project, nil
- }
- socket := s.dockerCli.DockerEndpoint().Host
- if !strings.HasPrefix(socket, "unix://") {
- return nil, fmt.Errorf("use_api_socket can only be used with unix sockets: docker endpoint %s is incompatible", socket)
- }
- socket = strings.TrimPrefix(socket, "unix://") // should we confirm absolute path?
- creds, err := s.dockerCli.ConfigFile().GetAllCredentials()
- if err != nil {
- return nil, fmt.Errorf("resolving credentials failed: %w", err)
- }
- newConfig := &configfile.ConfigFile{
- AuthConfigs: creds,
- }
- var configBuf bytes.Buffer
- if err := newConfig.SaveToWriter(&configBuf); err != nil {
- return nil, fmt.Errorf("saving creds for API socket: %w", err)
- }
- project.Configs["#apisocket"] = types.ConfigObjConfig{
- Content: configBuf.String(),
- }
- for name, service := range project.Services {
- if !service.UseAPISocket {
- continue
- }
- service.Volumes = append(service.Volumes, types.ServiceVolumeConfig{
- Type: types.VolumeTypeBind,
- Source: socket,
- Target: "/var/run/docker.sock",
- })
- _, envvarPresent := service.Environment["DOCKER_CONFIG"]
- // If the DOCKER_CONFIG env var is already present, we assume the client knows
- // what they're doing and don't inject the creds.
- if !envvarPresent {
- // Set our special little location for the config file.
- path := "/run/secrets/docker"
- service.Environment["DOCKER_CONFIG"] = &path
- }
- service.Configs = append(service.Configs, types.ServiceConfigObjConfig{
- Source: "#apisocket",
- Target: "/run/secrets/docker/config.json",
- })
- project.Services[name] = service
- }
- return project, nil
- }
|