2019-01-12 04:58:27 +00:00
|
|
|
// +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"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"k8s.io/klog"
|
2019-12-12 01:27:03 +00:00
|
|
|
utilexec "k8s.io/utils/exec"
|
2019-04-07 17:07:55 +00:00
|
|
|
"k8s.io/utils/keymutex"
|
|
|
|
utilpath "k8s.io/utils/path"
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// 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,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
// acquire lock for smb mount
|
|
|
|
var getSMBMountMutex = keymutex.NewHashed(0)
|
|
|
|
|
2019-03-04 01:22:32 +00:00
|
|
|
// Mount : mounts source to target with given options.
|
|
|
|
// currently only supports cifs(smb), bind mount(for disk)
|
2019-01-12 04:58:27 +00:00
|
|
|
func (mounter *Mounter) Mount(source string, target string, fstype string, options []string) error {
|
2019-09-27 21:51:53 +00:00
|
|
|
target = NormalizeWindowsPath(target)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
if source == "tmpfs" {
|
2019-03-04 01:22:32 +00:00
|
|
|
klog.V(3).Infof("mounting source (%q), target (%q), with options (%q)", source, target, options)
|
2019-01-12 04:58:27 +00:00
|
|
|
return os.MkdirAll(target, 0755)
|
|
|
|
}
|
|
|
|
|
|
|
|
parentDir := filepath.Dir(target)
|
|
|
|
if err := os.MkdirAll(parentDir, 0755); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-03-04 01:22:32 +00:00
|
|
|
klog.V(4).Infof("mount options(%q) source:%q, target:%q, fstype:%q, begin to mount",
|
2019-01-12 04:58:27 +00:00
|
|
|
options, source, target, fstype)
|
2019-03-04 01:22:32 +00:00
|
|
|
bindSource := source
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// tell it's going to mount azure disk or azure file according to options
|
2019-09-27 21:51:53 +00:00
|
|
|
if bind, _, _ := MakeBindOpts(options); bind {
|
2019-01-12 04:58:27 +00:00
|
|
|
// mount azure disk
|
2019-09-27 21:51:53 +00:00
|
|
|
bindSource = NormalizeWindowsPath(source)
|
2019-01-12 04:58:27 +00:00
|
|
|
} else {
|
|
|
|
if len(options) < 2 {
|
2019-03-04 01:22:32 +00:00
|
|
|
klog.Warningf("mount options(%q) command number(%d) less than 2, source:%q, target:%q, skip mounting",
|
2019-01-12 04:58:27 +00:00
|
|
|
options, len(options), source, target)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// currently only cifs mount is supported
|
|
|
|
if strings.ToLower(fstype) != "cifs" {
|
2019-03-04 01:22:32 +00:00
|
|
|
return fmt.Errorf("only cifs mount is supported now, fstype: %q, mounting source (%q), target (%q), with options (%q)", fstype, source, target, options)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
// lock smb mount for the same source
|
|
|
|
getSMBMountMutex.LockKey(source)
|
|
|
|
defer getSMBMountMutex.UnlockKey(source)
|
|
|
|
|
2019-03-04 01:22:32 +00:00
|
|
|
if output, err := newSMBMapping(options[0], options[1], source); err != nil {
|
|
|
|
if isSMBMappingExist(source) {
|
|
|
|
klog.V(2).Infof("SMB Mapping(%s) already exists, now begin to remove and remount", source)
|
|
|
|
if output, err := removeSMBMapping(source); err != nil {
|
|
|
|
return fmt.Errorf("Remove-SmbGlobalMapping failed: %v, output: %q", err, output)
|
|
|
|
}
|
|
|
|
if output, err := newSMBMapping(options[0], options[1], source); err != nil {
|
|
|
|
return fmt.Errorf("New-SmbGlobalMapping remount failed: %v, output: %q", err, output)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("New-SmbGlobalMapping failed: %v, output: %q", err, output)
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if output, err := exec.Command("cmd", "/c", "mklink", "/D", target, bindSource).CombinedOutput(); err != nil {
|
|
|
|
klog.Errorf("mklink failed: %v, source(%q) target(%q) output: %q", err, bindSource, target, string(output))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-03-04 01:22:32 +00:00
|
|
|
// do the SMB mount with username, password, remotepath
|
|
|
|
// return (output, error)
|
|
|
|
func newSMBMapping(username, password, remotepath string) (string, error) {
|
|
|
|
if username == "" || password == "" || remotepath == "" {
|
|
|
|
return "", fmt.Errorf("invalid parameter(username: %s, password: %s, remoteapth: %s)", username, password, remotepath)
|
|
|
|
}
|
|
|
|
|
|
|
|
// use PowerShell Environment Variables to store user input string to prevent command line injection
|
|
|
|
// https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_environment_variables?view=powershell-5.1
|
|
|
|
cmdLine := `$PWord = ConvertTo-SecureString -String $Env:smbpassword -AsPlainText -Force` +
|
|
|
|
`;$Credential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList $Env:smbuser, $PWord` +
|
|
|
|
`;New-SmbGlobalMapping -RemotePath $Env:smbremotepath -Credential $Credential`
|
|
|
|
cmd := exec.Command("powershell", "/c", cmdLine)
|
|
|
|
cmd.Env = append(os.Environ(),
|
|
|
|
fmt.Sprintf("smbuser=%s", username),
|
|
|
|
fmt.Sprintf("smbpassword=%s", password),
|
|
|
|
fmt.Sprintf("smbremotepath=%s", remotepath))
|
|
|
|
|
|
|
|
output, err := cmd.CombinedOutput()
|
|
|
|
return string(output), err
|
|
|
|
}
|
|
|
|
|
|
|
|
// check whether remotepath is already mounted
|
|
|
|
func isSMBMappingExist(remotepath string) bool {
|
|
|
|
cmd := exec.Command("powershell", "/c", `Get-SmbGlobalMapping -RemotePath $Env:smbremotepath`)
|
|
|
|
cmd.Env = append(os.Environ(), fmt.Sprintf("smbremotepath=%s", remotepath))
|
|
|
|
_, err := cmd.CombinedOutput()
|
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove SMB mapping
|
|
|
|
func removeSMBMapping(remotepath string) (string, error) {
|
|
|
|
cmd := exec.Command("powershell", "/c", `Remove-SmbGlobalMapping -RemotePath $Env:smbremotepath -Force`)
|
|
|
|
cmd.Env = append(os.Environ(), fmt.Sprintf("smbremotepath=%s", remotepath))
|
|
|
|
output, err := cmd.CombinedOutput()
|
|
|
|
return string(output), err
|
|
|
|
}
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
// Unmount unmounts the target.
|
|
|
|
func (mounter *Mounter) Unmount(target string) error {
|
|
|
|
klog.V(4).Infof("azureMount: Unmount target (%q)", target)
|
2019-09-27 21:51:53 +00:00
|
|
|
target = NormalizeWindowsPath(target)
|
2019-01-12 04:58:27 +00:00
|
|
|
if output, err := exec.Command("cmd", "/c", "rmdir", target).CombinedOutput(); err != nil {
|
|
|
|
klog.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
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {
|
|
|
|
target, err := os.Readlink(file)
|
|
|
|
if err != nil {
|
|
|
|
return true, fmt.Errorf("readlink error: %v", err)
|
|
|
|
}
|
2019-09-27 21:51:53 +00:00
|
|
|
exists, err := utilpath.Exists(utilpath.CheckFollowSymlink, target)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return true, err
|
|
|
|
}
|
|
|
|
return !exists, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// GetMountRefs : empty implementation here since there is no place to query all mount points on Windows
|
|
|
|
func (mounter *Mounter) GetMountRefs(pathname string) ([]string, error) {
|
|
|
|
windowsPath := NormalizeWindowsPath(pathname)
|
|
|
|
pathExists, pathErr := PathExists(windowsPath)
|
|
|
|
if !pathExists {
|
|
|
|
return []string{}, nil
|
|
|
|
} else if IsCorruptedMnt(pathErr) {
|
|
|
|
klog.Warningf("GetMountRefs found corrupted mount at %s, treating as unmounted path", windowsPath)
|
|
|
|
return []string{}, nil
|
|
|
|
} else if pathErr != nil {
|
|
|
|
return nil, fmt.Errorf("error checking path %s: %v", windowsPath, pathErr)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-09-27 21:51:53 +00:00
|
|
|
return []string{pathname}, nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (mounter *SafeFormatAndMount) formatAndMount(source string, target string, fstype string, options []string) error {
|
|
|
|
// Try to mount the disk
|
|
|
|
klog.V(4).Infof("Attempting to formatAndMount disk: %s %s %s", fstype, source, target)
|
|
|
|
|
|
|
|
if err := ValidateDiskNumber(source); err != nil {
|
|
|
|
klog.Errorf("diskMount: formatAndMount failed, err: %v", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(fstype) == 0 {
|
|
|
|
// Use 'NTFS' as the default
|
|
|
|
fstype = "NTFS"
|
|
|
|
}
|
|
|
|
|
|
|
|
// format disk if it is unformatted(raw)
|
|
|
|
cmd := fmt.Sprintf("Get-Disk -Number %s | Where partitionstyle -eq 'raw' | Initialize-Disk -PartitionStyle MBR -PassThru"+
|
|
|
|
" | New-Partition -AssignDriveLetter -UseMaximumSize | Format-Volume -FileSystem %s -Confirm:$false", source, fstype)
|
2019-12-12 01:27:03 +00:00
|
|
|
if output, err := mounter.Exec.Command("powershell", "/c", cmd).CombinedOutput(); err != nil {
|
2019-01-12 04:58:27 +00:00
|
|
|
return fmt.Errorf("diskMount: format disk failed, error: %v, output: %q", err, string(output))
|
|
|
|
}
|
|
|
|
klog.V(4).Infof("diskMount: Disk successfully formatted, disk: %q, fstype: %q", source, fstype)
|
|
|
|
|
|
|
|
driveLetter, err := getDriveLetterByDiskNumber(source, mounter.Exec)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
driverPath := driveLetter + ":"
|
2019-09-27 21:51:53 +00:00
|
|
|
target = NormalizeWindowsPath(target)
|
2019-01-12 04:58:27 +00:00
|
|
|
klog.V(4).Infof("Attempting to formatAndMount disk: %s %s %s", fstype, driverPath, target)
|
2019-12-12 01:27:03 +00:00
|
|
|
if output, err := mounter.Exec.Command("cmd", "/c", "mklink", "/D", target, driverPath).CombinedOutput(); err != nil {
|
2019-01-12 04:58:27 +00:00
|
|
|
klog.Errorf("mklink failed: %v, output: %q", err, string(output))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get drive letter according to windows disk number
|
2019-12-12 01:27:03 +00:00
|
|
|
func getDriveLetterByDiskNumber(diskNum string, exec utilexec.Interface) (string, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
cmd := fmt.Sprintf("(Get-Partition -DiskNumber %s).DriveLetter", diskNum)
|
2019-12-12 01:27:03 +00:00
|
|
|
output, err := exec.Command("powershell", "/c", cmd).CombinedOutput()
|
2019-01-12 04:58:27 +00:00
|
|
|
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
|
|
|
|
}
|