package system import ( "path" "path/filepath" "strings" "github.com/pkg/errors" ) // DefaultPathEnvUnix is unix style list of directories to search for // executables. Each directory is separated from the next by a colon // ':' character . const DefaultPathEnvUnix = "/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin" // DefaultPathEnvWindows is windows style list of directories to search for // executables. Each directory is separated from the next by a colon // ';' character . const DefaultPathEnvWindows = "c:\\Windows\\System32;c:\\Windows" func DefaultPathEnv(os string) string { if os == "windows" { return DefaultPathEnvWindows } return DefaultPathEnvUnix } // NormalizePath cleans the path based on the operating system the path is meant for. // It takes into account a potential parent path, and will join the path to the parent // if the path is relative. Additionally, it will apply the folliwing rules: // - always return an absolute path // - always strip drive letters for Windows paths // - optionally keep the trailing slashes on paths // - paths are returned using forward slashes func NormalizePath(parent, newPath, inputOS string, keepSlash bool) (string, error) { if inputOS == "" { inputOS = "linux" } newPath = toSlash(newPath, inputOS) parent = toSlash(parent, inputOS) origPath := newPath if parent == "" { parent = "/" } var err error parent, err = CheckSystemDriveAndRemoveDriveLetter(parent, inputOS) if err != nil { return "", errors.Wrap(err, "removing drive letter") } if !IsAbs(parent, inputOS) { parent = path.Join("/", parent) } if newPath == "" { // New workdir is empty. Use the "current" workdir. It should already // be an absolute path. newPath = parent } newPath, err = CheckSystemDriveAndRemoveDriveLetter(newPath, inputOS) if err != nil { return "", errors.Wrap(err, "removing drive letter") } if !IsAbs(newPath, inputOS) { // The new WD is relative. Join it to the previous WD. newPath = path.Join(parent, newPath) } if keepSlash { if strings.HasSuffix(origPath, "/") && !strings.HasSuffix(newPath, "/") { newPath += "/" } else if strings.HasSuffix(origPath, "/.") { if newPath != "/" { newPath += "/" } newPath += "." } } return toSlash(newPath, inputOS), nil } func toSlash(inputPath, inputOS string) string { separator := "/" if inputOS == "windows" { separator = "\\" } return strings.Replace(inputPath, separator, "/", -1) } func fromSlash(inputPath, inputOS string) string { separator := "/" if inputOS == "windows" { separator = "\\" } return strings.Replace(inputPath, "/", separator, -1) } // NormalizeWorkdir will return a normalized version of the new workdir, given // the currently configured workdir and the desired new workdir. When setting a // new relative workdir, it will be joined to the previous workdir or default to // the root folder. // On Windows we remove the drive letter and convert the path delimiter to "\". // Paths that begin with os.PathSeparator are considered absolute even on Windows. func NormalizeWorkdir(current, wd string, inputOS string) (string, error) { if inputOS == "" { inputOS = "linux" } wd, err := NormalizePath(current, wd, inputOS, false) if err != nil { return "", errors.Wrap(err, "normalizing working directory") } // Make sure we use the platform specific path separator. HCS does not like forward // slashes in CWD. return fromSlash(wd, inputOS), nil } // IsAbs returns a boolean value indicating whether or not the path // is absolute. On Linux, this is just a wrapper for filepath.IsAbs(). // On Windows, we strip away the drive letter (if any), clean the path, // and check whether or not the path starts with a filepath.Separator. // This function is meant to check if a path is absolute, in the context // of a COPY, ADD or WORKDIR, which have their root set in the mount point // of the writable layer we are mutating. The filepath.IsAbs() function on // Windows will not work in these scenatios, as it will return true for paths // that: // - Begin with drive letter (DOS style paths) // - Are volume paths \\?\Volume{UUID} // - Are UNC paths func IsAbs(pth, inputOS string) bool { if inputOS == "" { inputOS = "linux" } cleanedPath, err := CheckSystemDriveAndRemoveDriveLetter(pth, inputOS) if err != nil { return false } cleanedPath = toSlash(cleanedPath, inputOS) // We stripped any potential drive letter and converted any backslashes to // forward slashes. We can safely use path.IsAbs() for both Windows and Linux. return path.IsAbs(cleanedPath) } // CheckSystemDriveAndRemoveDriveLetter verifies and manipulates a Windows path. // For linux, this is a no-op. // // This is used, for example, when validating a user provided path in docker cp. // If a drive letter is supplied, it must be the system drive. The drive letter // is always removed. It also converts any backslash to forward slash. The conversion // to OS specific separator should happen as late as possible (ie: before passing the // value to the function that will actually use it). Paths are parsed and code paths are // triggered starting with the client and all the way down to calling into the runtime // environment. The client may run on a foreign OS from the one the build will be triggered // (Windows clients connecting to Linux or vice versa). // Keeping the file separator consistent until the last moment is desirable. // // We need the Windows path without the drive letter so that it can ultimately be concatenated with // a Windows long-path which doesn't support drive-letters. Examples: // C: --> Fail // C:somepath --> somepath // This is a relative path to the CWD set for that drive letter // C:\ --> \ // a --> a // /a --> \a // d:\ --> Fail // // UNC paths can refer to multiple types of paths. From local filesystem paths, // to remote filesystems like SMB or named pipes. // There is no sane way to support this without adding a lot of complexity // which I am not sure is worth it. // \\.\C$\a --> Fail func CheckSystemDriveAndRemoveDriveLetter(path string, inputOS string) (string, error) { if inputOS == "" { inputOS = "linux" } if inputOS != "windows" { return path, nil } if len(path) == 2 && string(path[1]) == ":" { return "", errors.Errorf("No relative path specified in %q", path) } // UNC paths should error out if len(path) >= 2 && toSlash(path[:2], inputOS) == "//" { return "", errors.Errorf("UNC paths are not supported") } parts := strings.SplitN(path, ":", 2) // Path does not have a drive letter. Just return it. if len(parts) < 2 { return toSlash(filepath.Clean(path), inputOS), nil } // We expect all paths to be in C: if !strings.EqualFold(parts[0], "c") { return "", errors.New("The specified path is not on the system drive (C:)") } // A path of the form F:somepath, is a path that is relative CWD set for a particular // drive letter. See: // https://learn.microsoft.com/en-us/windows/win32/fileio/naming-a-file#fully-qualified-vs-relative-paths // // C:\>mkdir F:somepath // C:\>dir F:\ // Volume in drive F is New Volume // Volume Serial Number is 86E5-AB64 // // Directory of F:\ // // 11/27/2022 02:22 PM