2015-11-12 19:55:17 +00:00
|
|
|
package container
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
|
2016-09-06 18:18:12 +00:00
|
|
|
"github.com/docker/docker/api/types"
|
2015-11-12 19:55:17 +00:00
|
|
|
"github.com/docker/docker/pkg/archive"
|
2016-04-27 02:26:12 +00:00
|
|
|
"github.com/docker/docker/pkg/system"
|
2015-11-12 19:55:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ResolvePath resolves the given path in the container to a resource on the
|
|
|
|
// host. Returns a resolved path (absolute path to the resource on the host),
|
|
|
|
// the absolute path to the resource relative to the container's rootfs, and
|
2016-03-26 14:06:45 +00:00
|
|
|
// an error if the path points to outside the container's rootfs.
|
2015-11-12 19:55:17 +00:00
|
|
|
func (container *Container) ResolvePath(path string) (resolvedPath, absPath string, err error) {
|
2016-04-27 02:26:12 +00:00
|
|
|
// Check if a drive letter supplied, it must be the system drive. No-op except on Windows
|
2017-08-04 00:22:00 +00:00
|
|
|
path, err = system.CheckSystemDriveAndRemoveDriveLetter(path, container.BaseFS)
|
2016-04-27 02:26:12 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
|
2015-11-12 19:55:17 +00:00
|
|
|
// Consider the given path as an absolute path in the container.
|
2017-08-04 00:22:00 +00:00
|
|
|
absPath = archive.PreserveTrailingDotOrSeparator(
|
|
|
|
container.BaseFS.Join(string(container.BaseFS.Separator()), path),
|
|
|
|
path,
|
|
|
|
container.BaseFS.Separator())
|
2015-11-12 19:55:17 +00:00
|
|
|
|
|
|
|
// Split the absPath into its Directory and Base components. We will
|
|
|
|
// resolve the dir in the scope of the container then append the base.
|
2017-08-04 00:22:00 +00:00
|
|
|
dirPath, basePath := container.BaseFS.Split(absPath)
|
2015-11-12 19:55:17 +00:00
|
|
|
|
|
|
|
resolvedDirPath, err := container.GetResourcePath(dirPath)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolvedDirPath will have been cleaned (no trailing path separators) so
|
|
|
|
// we can manually join it with the base path element.
|
2017-08-04 00:22:00 +00:00
|
|
|
resolvedPath = resolvedDirPath + string(container.BaseFS.Separator()) + basePath
|
2015-11-12 19:55:17 +00:00
|
|
|
return resolvedPath, absPath, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// StatPath is the unexported version of StatPath. Locks and mounts should
|
|
|
|
// be acquired before calling this method and the given path should be fully
|
|
|
|
// resolved to a path on the host corresponding to the given absolute path
|
|
|
|
// inside the container.
|
|
|
|
func (container *Container) StatPath(resolvedPath, absPath string) (stat *types.ContainerPathStat, err error) {
|
2017-08-04 00:22:00 +00:00
|
|
|
driver := container.BaseFS
|
|
|
|
|
|
|
|
lstat, err := driver.Lstat(resolvedPath)
|
2015-11-12 19:55:17 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var linkTarget string
|
|
|
|
if lstat.Mode()&os.ModeSymlink != 0 {
|
|
|
|
// Fully evaluate the symlink in the scope of the container rootfs.
|
|
|
|
hostPath, err := container.GetResourcePath(absPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-08-04 00:22:00 +00:00
|
|
|
linkTarget, err = driver.Rel(driver.Path(), hostPath)
|
2015-11-12 19:55:17 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make it an absolute path.
|
2017-08-04 00:22:00 +00:00
|
|
|
linkTarget = driver.Join(string(driver.Separator()), linkTarget)
|
2015-11-12 19:55:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return &types.ContainerPathStat{
|
2017-08-04 00:22:00 +00:00
|
|
|
Name: driver.Base(absPath),
|
2015-11-12 19:55:17 +00:00
|
|
|
Size: lstat.Size(),
|
|
|
|
Mode: lstat.Mode(),
|
|
|
|
Mtime: lstat.ModTime(),
|
|
|
|
LinkTarget: linkTarget,
|
|
|
|
}, nil
|
|
|
|
}
|