summaryrefslogtreecommitdiff
path: root/vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go')
-rw-r--r--vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go346
1 files changed, 346 insertions, 0 deletions
diff --git a/vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go b/vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go
new file mode 100644
index 000000000..0c63626ea
--- /dev/null
+++ b/vendor/k8s.io/kubernetes/pkg/util/mount/mount_windows.go
@@ -0,0 +1,346 @@
+// +build windows
+
+/*
+Copyright 2017 The Kubernetes 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 mount
+
+import (
+ "fmt"
+ "os"
+ "os/exec"
+ "path"
+ "path/filepath"
+ "strconv"
+ "strings"
+ "syscall"
+
+ "github.com/golang/glog"
+)
+
+// Mounter provides the default implementation of mount.Interface
+// for the windows platform. This implementation assumes that the
+// kubelet is running in the host's root mount namespace.
+type Mounter struct {
+ mounterPath string
+}
+
+// New returns a mount.Interface for the current system.
+// It provides options to override the default mounter behavior.
+// mounterPath allows using an alternative to `/bin/mount` for mounting.
+func New(mounterPath string) Interface {
+ return &Mounter{
+ mounterPath: mounterPath,
+ }
+}
+
+// Mount : mounts source to target as NTFS with given options.
+func (mounter *Mounter) Mount(source string, target string, fstype string, options []string) error {
+ target = normalizeWindowsPath(target)
+
+ if source == "tmpfs" {
+ glog.V(3).Infof("azureMount: mounting source (%q), target (%q), with options (%q)", source, target, options)
+ return os.MkdirAll(target, 0755)
+ }
+
+ parentDir := filepath.Dir(target)
+ if err := os.MkdirAll(parentDir, 0755); err != nil {
+ return err
+ }
+
+ glog.V(4).Infof("azureMount: mount options(%q) source:%q, target:%q, fstype:%q, begin to mount",
+ options, source, target, fstype)
+ bindSource := ""
+
+ // tell it's going to mount azure disk or azure file according to options
+ if bind, _ := isBind(options); bind {
+ // mount azure disk
+ bindSource = normalizeWindowsPath(source)
+ } else {
+ if len(options) < 2 {
+ glog.Warningf("azureMount: mount options(%q) command number(%d) less than 2, source:%q, target:%q, skip mounting",
+ options, len(options), source, target)
+ return nil
+ }
+
+ // currently only cifs mount is supported
+ if strings.ToLower(fstype) != "cifs" {
+ return fmt.Errorf("azureMount: only cifs mount is supported now, fstype: %q, mounting source (%q), target (%q), with options (%q)", fstype, source, target, options)
+ }
+
+ cmdLine := fmt.Sprintf(`$User = "%s";$PWord = ConvertTo-SecureString -String "%s" -AsPlainText -Force;`+
+ `$Credential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList $User, $PWord`,
+ options[0], options[1])
+
+ bindSource = source
+ cmdLine += fmt.Sprintf(";New-SmbGlobalMapping -RemotePath %s -Credential $Credential", source)
+
+ if output, err := exec.Command("powershell", "/c", cmdLine).CombinedOutput(); err != nil {
+ return fmt.Errorf("azureMount: SmbGlobalMapping failed: %v, only SMB mount is supported now, output: %q", err, string(output))
+ }
+ }
+
+ if output, err := exec.Command("cmd", "/c", "mklink", "/D", target, bindSource).CombinedOutput(); err != nil {
+ glog.Errorf("mklink failed: %v, source(%q) target(%q) output: %q", err, bindSource, target, string(output))
+ return err
+ }
+
+ return nil
+}
+
+// Unmount unmounts the target.
+func (mounter *Mounter) Unmount(target string) error {
+ glog.V(4).Infof("azureMount: Unmount target (%q)", target)
+ target = normalizeWindowsPath(target)
+ if output, err := exec.Command("cmd", "/c", "rmdir", target).CombinedOutput(); err != nil {
+ glog.Errorf("rmdir failed: %v, output: %q", err, string(output))
+ return err
+ }
+ return nil
+}
+
+// GetMountRefs finds all other references to the device(drive) referenced
+// by mountPath; returns a list of paths.
+func GetMountRefs(mounter Interface, mountPath string) ([]string, error) {
+ refs, err := getAllParentLinks(normalizeWindowsPath(mountPath))
+ if err != nil {
+ return nil, err
+ }
+ return refs, nil
+}
+
+// List returns a list of all mounted filesystems. todo
+func (mounter *Mounter) List() ([]MountPoint, error) {
+ return []MountPoint{}, nil
+}
+
+// IsMountPointMatch determines if the mountpoint matches the dir
+func (mounter *Mounter) IsMountPointMatch(mp MountPoint, dir string) bool {
+ return mp.Path == dir
+}
+
+// IsNotMountPoint determines if a directory is a mountpoint.
+func (mounter *Mounter) IsNotMountPoint(dir string) (bool, error) {
+ return IsNotMountPoint(mounter, dir)
+}
+
+// IsLikelyNotMountPoint determines if a directory is not a mountpoint.
+func (mounter *Mounter) IsLikelyNotMountPoint(file string) (bool, error) {
+ stat, err := os.Lstat(file)
+ if err != nil {
+ return true, err
+ }
+ // If current file is a symlink, then it is a mountpoint.
+ if stat.Mode()&os.ModeSymlink != 0 {
+ return false, nil
+ }
+
+ return true, nil
+}
+
+// GetDeviceNameFromMount given a mnt point, find the device
+func (mounter *Mounter) GetDeviceNameFromMount(mountPath, pluginDir string) (string, error) {
+ return getDeviceNameFromMount(mounter, mountPath, pluginDir)
+}
+
+// getDeviceNameFromMount find the device(drive) name in which
+// the mount path reference should match the given plugin directory. In case no mount path reference
+// matches, returns the volume name taken from its given mountPath
+func getDeviceNameFromMount(mounter Interface, mountPath, pluginDir string) (string, error) {
+ refs, err := GetMountRefs(mounter, mountPath)
+ if err != nil {
+ glog.V(4).Infof("GetMountRefs failed for mount path %q: %v", mountPath, err)
+ return "", err
+ }
+ if len(refs) == 0 {
+ return "", fmt.Errorf("directory %s is not mounted", mountPath)
+ }
+ basemountPath := normalizeWindowsPath(path.Join(pluginDir, MountsInGlobalPDPath))
+ for _, ref := range refs {
+ if strings.Contains(ref, basemountPath) {
+ volumeID, err := filepath.Rel(normalizeWindowsPath(basemountPath), ref)
+ if err != nil {
+ glog.Errorf("Failed to get volume id from mount %s - %v", mountPath, err)
+ return "", err
+ }
+ return volumeID, nil
+ }
+ }
+
+ return path.Base(mountPath), nil
+}
+
+// DeviceOpened determines if the device is in use elsewhere
+func (mounter *Mounter) DeviceOpened(pathname string) (bool, error) {
+ return false, nil
+}
+
+// PathIsDevice determines if a path is a device.
+func (mounter *Mounter) PathIsDevice(pathname string) (bool, error) {
+ return false, nil
+}
+
+// MakeRShared checks that given path is on a mount with 'rshared' mount
+// propagation. Empty implementation here.
+func (mounter *Mounter) MakeRShared(path string) error {
+ return nil
+}
+
+// GetFileType checks for sockets/block/character devices
+func (mounter *Mounter) GetFileType(pathname string) (FileType, error) {
+ var pathType FileType
+ info, err := os.Stat(pathname)
+ if os.IsNotExist(err) {
+ return pathType, fmt.Errorf("path %q does not exist", pathname)
+ }
+ // err in call to os.Stat
+ if err != nil {
+ return pathType, err
+ }
+
+ mode := info.Sys().(*syscall.Win32FileAttributeData).FileAttributes
+ switch mode & syscall.S_IFMT {
+ case syscall.S_IFSOCK:
+ return FileTypeSocket, nil
+ case syscall.S_IFBLK:
+ return FileTypeBlockDev, nil
+ case syscall.S_IFCHR:
+ return FileTypeCharDev, nil
+ case syscall.S_IFDIR:
+ return FileTypeDirectory, nil
+ case syscall.S_IFREG:
+ return FileTypeFile, nil
+ }
+
+ return pathType, fmt.Errorf("only recognise file, directory, socket, block device and character device")
+}
+
+// MakeFile creates a new directory
+func (mounter *Mounter) MakeDir(pathname string) error {
+ err := os.MkdirAll(pathname, os.FileMode(0755))
+ if err != nil {
+ if !os.IsExist(err) {
+ return err
+ }
+ }
+ return nil
+}
+
+// MakeFile creates an empty file
+func (mounter *Mounter) MakeFile(pathname string) error {
+ f, err := os.OpenFile(pathname, os.O_CREATE, os.FileMode(0644))
+ defer f.Close()
+ if err != nil {
+ if !os.IsExist(err) {
+ return err
+ }
+ }
+ return nil
+}
+
+// ExistsPath checks whether the path exists
+func (mounter *Mounter) ExistsPath(pathname string) bool {
+ _, err := os.Stat(pathname)
+ if err != nil {
+ return false
+ }
+ return true
+}
+
+func (mounter *SafeFormatAndMount) formatAndMount(source string, target string, fstype string, options []string) error {
+ // Try to mount the disk
+ glog.V(4).Infof("Attempting to formatAndMount disk: %s %s %s", fstype, source, target)
+
+ if err := ValidateDiskNumber(source); err != nil {
+ glog.Errorf("azureMount: formatAndMount failed, err: %v\n", err)
+ return err
+ }
+
+ driveLetter, err := getDriveLetterByDiskNumber(source, mounter.Exec)
+ if err != nil {
+ return err
+ }
+ driverPath := driveLetter + ":"
+ target = normalizeWindowsPath(target)
+ glog.V(4).Infof("Attempting to formatAndMount disk: %s %s %s", fstype, driverPath, target)
+ if output, err := mounter.Exec.Run("cmd", "/c", "mklink", "/D", target, driverPath); err != nil {
+ glog.Errorf("mklink failed: %v, output: %q", err, string(output))
+ return err
+ }
+ return nil
+}
+
+func normalizeWindowsPath(path string) string {
+ normalizedPath := strings.Replace(path, "/", "\\", -1)
+ if strings.HasPrefix(normalizedPath, "\\") {
+ normalizedPath = "c:" + normalizedPath
+ }
+ return normalizedPath
+}
+
+// ValidateDiskNumber : disk number should be a number in [0, 99]
+func ValidateDiskNumber(disk string) error {
+ diskNum, err := strconv.Atoi(disk)
+ if err != nil {
+ return fmt.Errorf("wrong disk number format: %q, err:%v", disk, err)
+ }
+
+ if diskNum < 0 || diskNum > 99 {
+ return fmt.Errorf("disk number out of range: %q", disk)
+ }
+
+ return nil
+}
+
+// Get drive letter according to windows disk number
+func getDriveLetterByDiskNumber(diskNum string, exec Exec) (string, error) {
+ cmd := fmt.Sprintf("(Get-Partition -DiskNumber %s).DriveLetter", diskNum)
+ output, err := exec.Run("powershell", "/c", cmd)
+ if err != nil {
+ return "", fmt.Errorf("azureMount: Get Drive Letter failed: %v, output: %q", err, string(output))
+ }
+ if len(string(output)) < 1 {
+ return "", fmt.Errorf("azureMount: Get Drive Letter failed, output is empty")
+ }
+ return string(output)[:1], nil
+}
+
+// getAllParentLinks walks all symbolic links and return all the parent targets recursively
+func getAllParentLinks(path string) ([]string, error) {
+ const maxIter = 255
+ links := []string{}
+ for {
+ links = append(links, path)
+ if len(links) > maxIter {
+ return links, fmt.Errorf("unexpected length of parent links: %v", links)
+ }
+
+ fi, err := os.Lstat(path)
+ if err != nil {
+ return links, fmt.Errorf("Lstat: %v", err)
+ }
+ if fi.Mode()&os.ModeSymlink == 0 {
+ break
+ }
+
+ path, err = os.Readlink(path)
+ if err != nil {
+ return links, fmt.Errorf("Readlink error: %v", err)
+ }
+ }
+
+ return links, nil
+}