diff_unix.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. //+build !windows
  2. package chrootarchive
  3. import (
  4. "bytes"
  5. "encoding/json"
  6. "flag"
  7. "fmt"
  8. "io/ioutil"
  9. "os"
  10. "path/filepath"
  11. "runtime"
  12. "github.com/docker/docker/pkg/archive"
  13. "github.com/docker/docker/pkg/reexec"
  14. "github.com/docker/docker/pkg/system"
  15. )
  16. type applyLayerResponse struct {
  17. LayerSize int64 `json:"layerSize"`
  18. }
  19. // applyLayer is the entry-point for docker-applylayer on re-exec. This is not
  20. // used on Windows as it does not support chroot, hence no point sandboxing
  21. // through chroot and rexec.
  22. func applyLayer() {
  23. var (
  24. tmpDir = ""
  25. err error
  26. )
  27. runtime.LockOSThread()
  28. flag.Parse()
  29. if err := chroot(flag.Arg(0)); err != nil {
  30. fatal(err)
  31. }
  32. // We need to be able to set any perms
  33. oldmask, err := system.Umask(0)
  34. defer system.Umask(oldmask)
  35. if err != nil {
  36. fatal(err)
  37. }
  38. if tmpDir, err = ioutil.TempDir("/", "temp-docker-extract"); err != nil {
  39. fatal(err)
  40. }
  41. os.Setenv("TMPDIR", tmpDir)
  42. size, err := archive.UnpackLayer("/", os.Stdin)
  43. os.RemoveAll(tmpDir)
  44. if err != nil {
  45. fatal(err)
  46. }
  47. encoder := json.NewEncoder(os.Stdout)
  48. if err := encoder.Encode(applyLayerResponse{size}); err != nil {
  49. fatal(fmt.Errorf("unable to encode layerSize JSON: %s", err))
  50. }
  51. flush(os.Stdout)
  52. flush(os.Stdin)
  53. os.Exit(0)
  54. }
  55. // ApplyLayer parses a diff in the standard layer format from `layer`,
  56. // and applies it to the directory `dest`. The stream `layer` can only be
  57. // uncompressed.
  58. // Returns the size in bytes of the contents of the layer.
  59. func ApplyLayer(dest string, layer archive.ArchiveReader) (size int64, err error) {
  60. return applyLayerHandler(dest, layer, true)
  61. }
  62. // ApplyUncompressedLayer parses a diff in the standard layer format from
  63. // `layer`, and applies it to the directory `dest`. The stream `layer`
  64. // can only be uncompressed.
  65. // Returns the size in bytes of the contents of the layer.
  66. func ApplyUncompressedLayer(dest string, layer archive.ArchiveReader) (int64, error) {
  67. return applyLayerHandler(dest, layer, false)
  68. }
  69. func applyLayerHandler(dest string, layer archive.ArchiveReader, decompress bool) (size int64, err error) {
  70. dest = filepath.Clean(dest)
  71. if decompress {
  72. decompressed, err := archive.DecompressStream(layer)
  73. if err != nil {
  74. return 0, err
  75. }
  76. defer decompressed.Close()
  77. layer = decompressed
  78. }
  79. cmd := reexec.Command("docker-applyLayer", dest)
  80. cmd.Stdin = layer
  81. outBuf, errBuf := new(bytes.Buffer), new(bytes.Buffer)
  82. cmd.Stdout, cmd.Stderr = outBuf, errBuf
  83. if err = cmd.Run(); err != nil {
  84. return 0, fmt.Errorf("ApplyLayer %s stdout: %s stderr: %s", err, outBuf, errBuf)
  85. }
  86. // Stdout should be a valid JSON struct representing an applyLayerResponse.
  87. response := applyLayerResponse{}
  88. decoder := json.NewDecoder(outBuf)
  89. if err = decoder.Decode(&response); err != nil {
  90. return 0, fmt.Errorf("unable to decode ApplyLayer JSON response: %s", err)
  91. }
  92. return response.LayerSize, nil
  93. }