copy.go 2.8 KB

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