copy.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. package container
  2. import (
  3. "encoding/base64"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "os"
  9. "strings"
  10. "github.com/docker/docker/api/server/httputils"
  11. "github.com/docker/docker/api/types"
  12. "github.com/docker/docker/api/types/versions"
  13. "golang.org/x/net/context"
  14. )
  15. // postContainersCopy is deprecated in favor of getContainersArchive.
  16. func (s *containerRouter) postContainersCopy(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  17. // Deprecated since 1.8, Errors out since 1.12
  18. version := httputils.VersionFromContext(ctx)
  19. if versions.GreaterThanOrEqualTo(version, "1.24") {
  20. w.WriteHeader(http.StatusNotFound)
  21. return nil
  22. }
  23. if err := httputils.CheckForJSON(r); err != nil {
  24. return err
  25. }
  26. cfg := types.CopyConfig{}
  27. if err := json.NewDecoder(r.Body).Decode(&cfg); err != nil {
  28. return err
  29. }
  30. if cfg.Resource == "" {
  31. return fmt.Errorf("Path cannot be empty")
  32. }
  33. data, err := s.backend.ContainerCopy(vars["name"], cfg.Resource)
  34. if err != nil {
  35. if strings.Contains(strings.ToLower(err.Error()), "no such container") {
  36. w.WriteHeader(http.StatusNotFound)
  37. return nil
  38. }
  39. if os.IsNotExist(err) {
  40. return fmt.Errorf("Could not find the file %s in container %s", cfg.Resource, vars["name"])
  41. }
  42. return err
  43. }
  44. defer data.Close()
  45. w.Header().Set("Content-Type", "application/x-tar")
  46. if _, err := io.Copy(w, data); err != nil {
  47. return err
  48. }
  49. return nil
  50. }
  51. // // Encode the stat to JSON, base64 encode, and place in a header.
  52. func setContainerPathStatHeader(stat *types.ContainerPathStat, header http.Header) error {
  53. statJSON, err := json.Marshal(stat)
  54. if err != nil {
  55. return err
  56. }
  57. header.Set(
  58. "X-Docker-Container-Path-Stat",
  59. base64.StdEncoding.EncodeToString(statJSON),
  60. )
  61. return nil
  62. }
  63. func (s *containerRouter) headContainersArchive(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  64. v, err := httputils.ArchiveFormValues(r, vars)
  65. if err != nil {
  66. return err
  67. }
  68. stat, err := s.backend.ContainerStatPath(v.Name, v.Path)
  69. if err != nil {
  70. return err
  71. }
  72. return setContainerPathStatHeader(stat, w.Header())
  73. }
  74. func (s *containerRouter) getContainersArchive(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  75. v, err := httputils.ArchiveFormValues(r, vars)
  76. if err != nil {
  77. return err
  78. }
  79. tarArchive, stat, err := s.backend.ContainerArchivePath(v.Name, v.Path)
  80. if err != nil {
  81. return err
  82. }
  83. defer tarArchive.Close()
  84. if err := setContainerPathStatHeader(stat, w.Header()); err != nil {
  85. return err
  86. }
  87. w.Header().Set("Content-Type", "application/x-tar")
  88. _, err = io.Copy(w, tarArchive)
  89. return err
  90. }
  91. func (s *containerRouter) putContainersArchive(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  92. v, err := httputils.ArchiveFormValues(r, vars)
  93. if err != nil {
  94. return err
  95. }
  96. noOverwriteDirNonDir := httputils.BoolValue(r, "noOverwriteDirNonDir")
  97. return s.backend.ContainerExtractToDir(v.Name, v.Path, noOverwriteDirNonDir, r.Body)
  98. }