
add initial work for mount azure file on windows fix review comments full implementation for attach azure file on windows node working azure file mount remove useless functions add a workable implementation about mounting azure file on windows node fix review comments and make the pod creating successful even azure file mount failed fix according to review comments add mount_windows_test add implementation for IsLikelyNotMountPoint func remove mount_windows_test.go temporaly add back unit test for mount_windows.go add normalizeWindowsPath func fix normalizeWindowsPath func issue implment azure disk on windows update bazel BUILD revert validation.go change as it's another PR fix merge issue and compiling issue fix windows compiling issue fix according to review comments fix according to review comments fix cross-build failure fix according to review comments fix test build failure temporalily fix darwin build failure fix azure windows test failure add empty implementation of MakeRShared on windows fix gofmt errors
207 lines
6.3 KiB
Go
207 lines
6.3 KiB
Go
// +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/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"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
|
|
}
|
|
|
|
// empty implementation for mounting azure file
|
|
return os.MkdirAll(target, 0755)
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
// 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)
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
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
|
|
}
|