
This change splits the definition of pkg/cri/os.ResolveSymbolicLink by platform (windows/!windows), and switches to an alternate implementation for Windows. This aims to fix the issue described in containerd/containerd#5405. The previous implementation which just called filepath.EvalSymlinks has historically had issues on Windows. One of these issues we were able to fix in Go, but EvalSymlinks's behavior is not well specified on Windows, and there could easily be more issues in the future, so it seems prudent to move to a separate implementation for Windows. The new implementation uses the Windows GetFinalPathNameByHandle API, which takes a handle to an open file or directory and some flags, and returns the "real" name for the object. See comments in the code for details on the implementation. I have tested this change with a variety of mounts and everything seems to work as expected. Functions that make incorrect assumptions on what a Windows path can look like may have some trouble with the \\?\ path syntax. For instance EvalSymlinks fails when given a \\?\UNC\ path. For this reason, the resolvePath implementation modifies the returned path to translate to the more common form (\\?\UNC\server\share -> \\server\share). Signed-off-by: Kevin Parsons <kevpar@microsoft.com>
90 lines
2.5 KiB
Go
90 lines
2.5 KiB
Go
/*
|
|
Copyright The containerd Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package os
|
|
|
|
import (
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
|
|
"github.com/moby/sys/symlink"
|
|
)
|
|
|
|
// OS collects system level operations that need to be mocked out
|
|
// during tests.
|
|
type OS interface {
|
|
MkdirAll(path string, perm os.FileMode) error
|
|
RemoveAll(path string) error
|
|
Stat(name string) (os.FileInfo, error)
|
|
ResolveSymbolicLink(name string) (string, error)
|
|
FollowSymlinkInScope(path, scope string) (string, error)
|
|
CopyFile(src, dest string, perm os.FileMode) error
|
|
WriteFile(filename string, data []byte, perm os.FileMode) error
|
|
Hostname() (string, error)
|
|
}
|
|
|
|
// RealOS is used to dispatch the real system level operations.
|
|
type RealOS struct{}
|
|
|
|
// MkdirAll will call os.MkdirAll to create a directory.
|
|
func (RealOS) MkdirAll(path string, perm os.FileMode) error {
|
|
return os.MkdirAll(path, perm)
|
|
}
|
|
|
|
// RemoveAll will call os.RemoveAll to remove the path and its children.
|
|
func (RealOS) RemoveAll(path string) error {
|
|
return os.RemoveAll(path)
|
|
}
|
|
|
|
// Stat will call os.Stat to get the status of the given file.
|
|
func (RealOS) Stat(name string) (os.FileInfo, error) {
|
|
return os.Stat(name)
|
|
}
|
|
|
|
// FollowSymlinkInScope will call symlink.FollowSymlinkInScope.
|
|
func (RealOS) FollowSymlinkInScope(path, scope string) (string, error) {
|
|
return symlink.FollowSymlinkInScope(path, scope)
|
|
}
|
|
|
|
// CopyFile will copy src file to dest file
|
|
func (RealOS) CopyFile(src, dest string, perm os.FileMode) error {
|
|
in, err := os.Open(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer in.Close()
|
|
|
|
out, err := os.OpenFile(dest, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, perm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer out.Close()
|
|
|
|
_, err = io.Copy(out, in)
|
|
return err
|
|
}
|
|
|
|
// WriteFile will call ioutil.WriteFile to write data into a file.
|
|
func (RealOS) WriteFile(filename string, data []byte, perm os.FileMode) error {
|
|
return ioutil.WriteFile(filename, data, perm)
|
|
}
|
|
|
|
// Hostname will call os.Hostname to get the hostname of the host.
|
|
func (RealOS) Hostname() (string, error) {
|
|
return os.Hostname()
|
|
}
|