secrets.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /*
  2. Copyright 2020 Docker Compose CLI authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package compose
  14. import (
  15. "archive/tar"
  16. "bytes"
  17. "context"
  18. "fmt"
  19. "strconv"
  20. "time"
  21. "github.com/compose-spec/compose-go/v2/types"
  22. "github.com/docker/docker/api/types/container"
  23. )
  24. func (s *composeService) injectSecrets(ctx context.Context, project *types.Project, service types.ServiceConfig, id string) error {
  25. for _, config := range service.Secrets {
  26. file := project.Secrets[config.Source]
  27. if file.Environment == "" {
  28. continue
  29. }
  30. if service.ReadOnly {
  31. return fmt.Errorf("cannot create secret %q in read-only service %s: `file` is the sole supported option", file.Name, service.Name)
  32. }
  33. if config.Target == "" {
  34. config.Target = "/run/secrets/" + config.Source
  35. } else if !isAbsTarget(config.Target) {
  36. config.Target = "/run/secrets/" + config.Target
  37. }
  38. env, ok := project.Environment[file.Environment]
  39. if !ok {
  40. return fmt.Errorf("environment variable %q required by secret %q is not set", file.Environment, file.Name)
  41. }
  42. b, err := createTar(env, types.FileReferenceConfig(config))
  43. if err != nil {
  44. return err
  45. }
  46. err = s.apiClient().CopyToContainer(ctx, id, "/", &b, container.CopyToContainerOptions{
  47. CopyUIDGID: config.UID != "" || config.GID != "",
  48. })
  49. if err != nil {
  50. return err
  51. }
  52. }
  53. return nil
  54. }
  55. func (s *composeService) injectConfigs(ctx context.Context, project *types.Project, service types.ServiceConfig, id string) error {
  56. for _, config := range service.Configs {
  57. file := project.Configs[config.Source]
  58. content := file.Content
  59. if file.Environment != "" {
  60. env, ok := project.Environment[file.Environment]
  61. if !ok {
  62. return fmt.Errorf("environment variable %q required by config %q is not set", file.Environment, file.Name)
  63. }
  64. content = env
  65. }
  66. if content == "" {
  67. continue
  68. }
  69. if service.ReadOnly {
  70. return fmt.Errorf("cannot create config %q in read-only service %s: `file` is the sole supported option", file.Name, service.Name)
  71. }
  72. if config.Target == "" {
  73. config.Target = "/" + config.Source
  74. }
  75. b, err := createTar(content, types.FileReferenceConfig(config))
  76. if err != nil {
  77. return err
  78. }
  79. err = s.apiClient().CopyToContainer(ctx, id, "/", &b, container.CopyToContainerOptions{
  80. CopyUIDGID: config.UID != "" || config.GID != "",
  81. })
  82. if err != nil {
  83. return err
  84. }
  85. }
  86. return nil
  87. }
  88. func createTar(env string, config types.FileReferenceConfig) (bytes.Buffer, error) {
  89. value := []byte(env)
  90. b := bytes.Buffer{}
  91. tarWriter := tar.NewWriter(&b)
  92. mode := uint32(0o444)
  93. if config.Mode != nil {
  94. mode = *config.Mode
  95. }
  96. var uid, gid int
  97. if config.UID != "" {
  98. v, err := strconv.Atoi(config.UID)
  99. if err != nil {
  100. return b, err
  101. }
  102. uid = v
  103. }
  104. if config.GID != "" {
  105. v, err := strconv.Atoi(config.GID)
  106. if err != nil {
  107. return b, err
  108. }
  109. gid = v
  110. }
  111. header := &tar.Header{
  112. Name: config.Target,
  113. Size: int64(len(value)),
  114. Mode: int64(mode),
  115. ModTime: time.Now(),
  116. Uid: uid,
  117. Gid: gid,
  118. }
  119. err := tarWriter.WriteHeader(header)
  120. if err != nil {
  121. return bytes.Buffer{}, err
  122. }
  123. _, err = tarWriter.Write(value)
  124. if err != nil {
  125. return bytes.Buffer{}, err
  126. }
  127. err = tarWriter.Close()
  128. return b, err
  129. }