secrets.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  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. content := file.Content
  39. if content == "" {
  40. env, ok := project.Environment[file.Environment]
  41. if !ok {
  42. return fmt.Errorf("environment variable %q required by secret %q is not set", file.Environment, file.Name)
  43. }
  44. content = env
  45. }
  46. b, err := createTar(content, types.FileReferenceConfig(config))
  47. if err != nil {
  48. return err
  49. }
  50. err = s.apiClient().CopyToContainer(ctx, id, "/", &b, container.CopyToContainerOptions{
  51. CopyUIDGID: config.UID != "" || config.GID != "",
  52. })
  53. if err != nil {
  54. return err
  55. }
  56. }
  57. return nil
  58. }
  59. func (s *composeService) injectConfigs(ctx context.Context, project *types.Project, service types.ServiceConfig, id string) error {
  60. for _, config := range service.Configs {
  61. file := project.Configs[config.Source]
  62. content := file.Content
  63. if file.Environment != "" {
  64. env, ok := project.Environment[file.Environment]
  65. if !ok {
  66. return fmt.Errorf("environment variable %q required by config %q is not set", file.Environment, file.Name)
  67. }
  68. content = env
  69. }
  70. if content == "" {
  71. continue
  72. }
  73. if service.ReadOnly {
  74. return fmt.Errorf("cannot create config %q in read-only service %s: `file` is the sole supported option", file.Name, service.Name)
  75. }
  76. if config.Target == "" {
  77. config.Target = "/" + config.Source
  78. }
  79. b, err := createTar(content, types.FileReferenceConfig(config))
  80. if err != nil {
  81. return err
  82. }
  83. err = s.apiClient().CopyToContainer(ctx, id, "/", &b, container.CopyToContainerOptions{
  84. CopyUIDGID: config.UID != "" || config.GID != "",
  85. })
  86. if err != nil {
  87. return err
  88. }
  89. }
  90. return nil
  91. }
  92. func createTar(env string, config types.FileReferenceConfig) (bytes.Buffer, error) {
  93. value := []byte(env)
  94. b := bytes.Buffer{}
  95. tarWriter := tar.NewWriter(&b)
  96. mode := types.FileMode(0o444)
  97. if config.Mode != nil {
  98. mode = *config.Mode
  99. }
  100. var uid, gid int
  101. if config.UID != "" {
  102. v, err := strconv.Atoi(config.UID)
  103. if err != nil {
  104. return b, err
  105. }
  106. uid = v
  107. }
  108. if config.GID != "" {
  109. v, err := strconv.Atoi(config.GID)
  110. if err != nil {
  111. return b, err
  112. }
  113. gid = v
  114. }
  115. header := &tar.Header{
  116. Name: config.Target,
  117. Size: int64(len(value)),
  118. Mode: int64(mode),
  119. ModTime: time.Now(),
  120. Uid: uid,
  121. Gid: gid,
  122. }
  123. err := tarWriter.WriteHeader(header)
  124. if err != nil {
  125. return bytes.Buffer{}, err
  126. }
  127. _, err = tarWriter.Write(value)
  128. if err != nil {
  129. return bytes.Buffer{}, err
  130. }
  131. err = tarWriter.Close()
  132. return b, err
  133. }