1
0
Fork 0
mirror of https://github.com/moby/moby.git synced 2022-11-09 12:21:53 -05:00

Merge pull request #15141 from Microsoft/10662-cp

Windows [TP3] - make docker cp functional
This commit is contained in:
Arnaud Porterie 2015-08-06 17:15:09 -07:00
commit 04cd81401a
5 changed files with 57 additions and 20 deletions

View file

@ -3,6 +3,7 @@ package server
import ( import (
"fmt" "fmt"
"net/http" "net/http"
"path/filepath"
"strconv" "strconv"
"strings" "strings"
) )
@ -43,7 +44,7 @@ func archiveFormValues(r *http.Request, vars map[string]string) (archiveOptions,
} }
name := vars["name"] name := vars["name"]
path := r.Form.Get("path") path := filepath.FromSlash(r.Form.Get("path"))
switch { switch {
case name == "": case name == "":

View file

@ -28,8 +28,12 @@ var (
// path already ends in a `.` path segment, then another is not added. If the // path already ends in a `.` path segment, then another is not added. If the
// clean path already ends in a path separator, then another is not added. // clean path already ends in a path separator, then another is not added.
func PreserveTrailingDotOrSeparator(cleanedPath, originalPath string) string { func PreserveTrailingDotOrSeparator(cleanedPath, originalPath string) string {
if !SpecifiesCurrentDir(cleanedPath) && SpecifiesCurrentDir(originalPath) { // Ensure paths are in platform semantics
if !HasTrailingPathSeparator(cleanedPath) { cleanedPath = normalizePath(cleanedPath)
originalPath = normalizePath(originalPath)
if !specifiesCurrentDir(cleanedPath) && specifiesCurrentDir(originalPath) {
if !hasTrailingPathSeparator(cleanedPath) {
// Add a separator if it doesn't already end with one (a cleaned // Add a separator if it doesn't already end with one (a cleaned
// path would only end in a separator if it is the root). // path would only end in a separator if it is the root).
cleanedPath += string(filepath.Separator) cleanedPath += string(filepath.Separator)
@ -37,29 +41,29 @@ func PreserveTrailingDotOrSeparator(cleanedPath, originalPath string) string {
cleanedPath += "." cleanedPath += "."
} }
if !HasTrailingPathSeparator(cleanedPath) && HasTrailingPathSeparator(originalPath) { if !hasTrailingPathSeparator(cleanedPath) && hasTrailingPathSeparator(originalPath) {
cleanedPath += string(filepath.Separator) cleanedPath += string(filepath.Separator)
} }
return cleanedPath return cleanedPath
} }
// AssertsDirectory returns whether the given path is // assertsDirectory returns whether the given path is
// asserted to be a directory, i.e., the path ends with // asserted to be a directory, i.e., the path ends with
// a trailing '/' or `/.`, assuming a path separator of `/`. // a trailing '/' or `/.`, assuming a path separator of `/`.
func AssertsDirectory(path string) bool { func assertsDirectory(path string) bool {
return HasTrailingPathSeparator(path) || SpecifiesCurrentDir(path) return hasTrailingPathSeparator(path) || specifiesCurrentDir(path)
} }
// HasTrailingPathSeparator returns whether the given // hasTrailingPathSeparator returns whether the given
// path ends with the system's path separator character. // path ends with the system's path separator character.
func HasTrailingPathSeparator(path string) bool { func hasTrailingPathSeparator(path string) bool {
return len(path) > 0 && os.IsPathSeparator(path[len(path)-1]) return len(path) > 0 && os.IsPathSeparator(path[len(path)-1])
} }
// SpecifiesCurrentDir returns whether the given path specifies // specifiesCurrentDir returns whether the given path specifies
// a "current directory", i.e., the last path segment is `.`. // a "current directory", i.e., the last path segment is `.`.
func SpecifiesCurrentDir(path string) bool { func specifiesCurrentDir(path string) bool {
return filepath.Base(path) == "." return filepath.Base(path) == "."
} }
@ -67,9 +71,9 @@ func SpecifiesCurrentDir(path string) bool {
// basename by first cleaning the path but preserves a trailing "." if the // basename by first cleaning the path but preserves a trailing "." if the
// original path specified the current directory. // original path specified the current directory.
func SplitPathDirEntry(path string) (dir, base string) { func SplitPathDirEntry(path string) (dir, base string) {
cleanedPath := filepath.Clean(path) cleanedPath := filepath.Clean(normalizePath(path))
if SpecifiesCurrentDir(path) { if specifiesCurrentDir(path) {
cleanedPath += string(filepath.Separator) + "." cleanedPath += string(filepath.Separator) + "."
} }
@ -90,6 +94,7 @@ func TarResource(sourceInfo CopyInfo) (content Archive, err error) {
// TarResourceRebase is like TarResource but renames the first path element of // TarResourceRebase is like TarResource but renames the first path element of
// items in the resulting tar archive to match the given rebaseName if not "". // items in the resulting tar archive to match the given rebaseName if not "".
func TarResourceRebase(sourcePath, rebaseName string) (content Archive, err error) { func TarResourceRebase(sourcePath, rebaseName string) (content Archive, err error) {
sourcePath = normalizePath(sourcePath)
if _, err = os.Lstat(sourcePath); err != nil { if _, err = os.Lstat(sourcePath); err != nil {
// Catches the case where the source does not exist or is not a // Catches the case where the source does not exist or is not a
// directory if asserted to be a directory, as this also causes an // directory if asserted to be a directory, as this also causes an
@ -132,6 +137,7 @@ type CopyInfo struct {
func CopyInfoSourcePath(path string) (CopyInfo, error) { func CopyInfoSourcePath(path string) (CopyInfo, error) {
// Split the given path into its Directory and Base components. We will // Split the given path into its Directory and Base components. We will
// evaluate symlinks in the directory component then append the base. // evaluate symlinks in the directory component then append the base.
path = normalizePath(path)
dirPath, basePath := filepath.Split(path) dirPath, basePath := filepath.Split(path)
resolvedDirPath, err := filepath.EvalSymlinks(dirPath) resolvedDirPath, err := filepath.EvalSymlinks(dirPath)
@ -144,7 +150,7 @@ func CopyInfoSourcePath(path string) (CopyInfo, error) {
resolvedPath := resolvedDirPath + string(filepath.Separator) + basePath resolvedPath := resolvedDirPath + string(filepath.Separator) + basePath
var rebaseName string var rebaseName string
if HasTrailingPathSeparator(path) && filepath.Base(path) != filepath.Base(resolvedPath) { if hasTrailingPathSeparator(path) && filepath.Base(path) != filepath.Base(resolvedPath) {
// In the case where the path had a trailing separator and a symlink // In the case where the path had a trailing separator and a symlink
// evaluation has changed the last path component, we will need to // evaluation has changed the last path component, we will need to
// rebase the name in the archive that is being copied to match the // rebase the name in the archive that is being copied to match the
@ -170,6 +176,7 @@ func CopyInfoSourcePath(path string) (CopyInfo, error) {
// operation. The given path should be an absolute local path. // operation. The given path should be an absolute local path.
func CopyInfoDestinationPath(path string) (info CopyInfo, err error) { func CopyInfoDestinationPath(path string) (info CopyInfo, err error) {
maxSymlinkIter := 10 // filepath.EvalSymlinks uses 255, but 10 already seems like a lot. maxSymlinkIter := 10 // filepath.EvalSymlinks uses 255, but 10 already seems like a lot.
path = normalizePath(path)
originalPath := path originalPath := path
stat, err := os.Lstat(path) stat, err := os.Lstat(path)
@ -247,6 +254,10 @@ func CopyInfoDestinationPath(path string) (info CopyInfo, err error) {
// described by dstInfo. Returns the possibly modified content archive along // described by dstInfo. Returns the possibly modified content archive along
// with the path to the destination directory which it should be extracted to. // with the path to the destination directory which it should be extracted to.
func PrepareArchiveCopy(srcContent Reader, srcInfo, dstInfo CopyInfo) (dstDir string, content Archive, err error) { func PrepareArchiveCopy(srcContent Reader, srcInfo, dstInfo CopyInfo) (dstDir string, content Archive, err error) {
// Ensure in platform semantics
srcInfo.Path = normalizePath(srcInfo.Path)
dstInfo.Path = normalizePath(dstInfo.Path)
// Separate the destination path between its directory and base // Separate the destination path between its directory and base
// components in case the source archive contents need to be rebased. // components in case the source archive contents need to be rebased.
dstDir, dstBase := SplitPathDirEntry(dstInfo.Path) dstDir, dstBase := SplitPathDirEntry(dstInfo.Path)
@ -276,7 +287,7 @@ func PrepareArchiveCopy(srcContent Reader, srcInfo, dstInfo CopyInfo) (dstDir st
// The source content entries will have to be renamed to have a // The source content entries will have to be renamed to have a
// basename which matches the destination path's basename. // basename which matches the destination path's basename.
return dstDir, rebaseArchiveEntries(srcContent, srcBase, dstBase), nil return dstDir, rebaseArchiveEntries(srcContent, srcBase, dstBase), nil
case AssertsDirectory(dstInfo.Path): case assertsDirectory(dstInfo.Path):
// The destination does not exist and is asserted to be created as a // The destination does not exist and is asserted to be created as a
// directory, but the source content is not a directory. This is an // directory, but the source content is not a directory. This is an
// error condition since you cannot create a directory from a file // error condition since you cannot create a directory from a file
@ -297,7 +308,7 @@ func PrepareArchiveCopy(srcContent Reader, srcInfo, dstInfo CopyInfo) (dstDir st
// rebaseArchiveEntries rewrites the given srcContent archive replacing // rebaseArchiveEntries rewrites the given srcContent archive replacing
// an occurance of oldBase with newBase at the beginning of entry names. // an occurance of oldBase with newBase at the beginning of entry names.
func rebaseArchiveEntries(srcContent Reader, oldBase, newBase string) Archive { func rebaseArchiveEntries(srcContent Reader, oldBase, newBase string) Archive {
if oldBase == "/" { if oldBase == string(os.PathSeparator) {
// If oldBase specifies the root directory, use an empty string as // If oldBase specifies the root directory, use an empty string as
// oldBase instead so that newBase doesn't replace the path separator // oldBase instead so that newBase doesn't replace the path separator
// that all paths will start with. // that all paths will start with.
@ -349,6 +360,10 @@ func CopyResource(srcPath, dstPath string) error {
err error err error
) )
// Ensure in platform semantics
srcPath = normalizePath(srcPath)
dstPath = normalizePath(dstPath)
// Clean the source and destination paths. // Clean the source and destination paths.
srcPath = PreserveTrailingDotOrSeparator(filepath.Clean(srcPath), srcPath) srcPath = PreserveTrailingDotOrSeparator(filepath.Clean(srcPath), srcPath)
dstPath = PreserveTrailingDotOrSeparator(filepath.Clean(dstPath), dstPath) dstPath = PreserveTrailingDotOrSeparator(filepath.Clean(dstPath), dstPath)
@ -371,7 +386,7 @@ func CopyResource(srcPath, dstPath string) error {
func CopyTo(content Reader, srcInfo CopyInfo, dstPath string) error { func CopyTo(content Reader, srcInfo CopyInfo, dstPath string) error {
// The destination path need not exist, but CopyInfoDestinationPath will // The destination path need not exist, but CopyInfoDestinationPath will
// ensure that at least the parent directory exists. // ensure that at least the parent directory exists.
dstInfo, err := CopyInfoDestinationPath(dstPath) dstInfo, err := CopyInfoDestinationPath(normalizePath(dstPath))
if err != nil { if err != nil {
return err return err
} }

11
pkg/archive/copy_unix.go Normal file
View file

@ -0,0 +1,11 @@
// +build !windows
package archive
import (
"path/filepath"
)
func normalizePath(path string) string {
return filepath.ToSlash(path)
}

View file

@ -0,0 +1,9 @@
package archive
import (
"path/filepath"
)
func normalizePath(path string) string {
return filepath.FromSlash(path)
}

View file

@ -14,13 +14,14 @@ import (
"strings" "strings"
) )
// FollowSymlinkInScope is a wrapper around evalSymlinksInScope that returns an absolute path // FollowSymlinkInScope is a wrapper around evalSymlinksInScope that returns an
// absolute path. This function handles paths in a platform-agnostic manner.
func FollowSymlinkInScope(path, root string) (string, error) { func FollowSymlinkInScope(path, root string) (string, error) {
path, err := filepath.Abs(path) path, err := filepath.Abs(filepath.FromSlash(path))
if err != nil { if err != nil {
return "", err return "", err
} }
root, err = filepath.Abs(root) root, err = filepath.Abs(filepath.FromSlash(root))
if err != nil { if err != nil {
return "", err return "", err
} }