123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239 |
- /*
- Copyright 2015 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 (
- "errors"
- "os"
- "path/filepath"
- "sync"
- "k8s.io/klog"
- )
- // FakeMounter implements mount.Interface for tests.
- type FakeMounter struct {
- MountPoints []MountPoint
- Log []FakeAction
- Filesystem map[string]FileType
- // Error to return for a path when calling IsLikelyNotMountPoint
- MountCheckErrors map[string]error
- // Some tests run things in parallel, make sure the mounter does not produce
- // any golang's DATA RACE warnings.
- mutex sync.Mutex
- }
- var _ Interface = &FakeMounter{}
- // Values for FakeAction.Action
- const FakeActionMount = "mount"
- const FakeActionUnmount = "unmount"
- // FakeAction objects are logged every time a fake mount or unmount is called.
- type FakeAction struct {
- Action string // "mount" or "unmount"
- Target string // applies to both mount and unmount actions
- Source string // applies only to "mount" actions
- FSType string // applies only to "mount" actions
- }
- func (f *FakeMounter) ResetLog() {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- f.Log = []FakeAction{}
- }
- func (f *FakeMounter) Mount(source string, target string, fstype string, options []string) error {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- opts := []string{}
- for _, option := range options {
- // find 'bind' option
- if option == "bind" {
- // This is a bind-mount. In order to mimic linux behaviour, we must
- // use the original device of the bind-mount as the real source.
- // E.g. when mounted /dev/sda like this:
- // $ mount /dev/sda /mnt/test
- // $ mount -o bind /mnt/test /mnt/bound
- // then /proc/mount contains:
- // /dev/sda /mnt/test
- // /dev/sda /mnt/bound
- // (and not /mnt/test /mnt/bound)
- // I.e. we must use /dev/sda as source instead of /mnt/test in the
- // bind mount.
- for _, mnt := range f.MountPoints {
- if source == mnt.Path {
- source = mnt.Device
- break
- }
- }
- }
- // reuse MountPoint.Opts field to mark mount as readonly
- opts = append(opts, option)
- }
- // If target is a symlink, get its absolute path
- absTarget, err := filepath.EvalSymlinks(target)
- if err != nil {
- absTarget = target
- }
- f.MountPoints = append(f.MountPoints, MountPoint{Device: source, Path: absTarget, Type: fstype, Opts: opts})
- klog.V(5).Infof("Fake mounter: mounted %s to %s", source, absTarget)
- f.Log = append(f.Log, FakeAction{Action: FakeActionMount, Target: absTarget, Source: source, FSType: fstype})
- return nil
- }
- func (f *FakeMounter) Unmount(target string) error {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- // If target is a symlink, get its absolute path
- absTarget, err := filepath.EvalSymlinks(target)
- if err != nil {
- absTarget = target
- }
- newMountpoints := []MountPoint{}
- for _, mp := range f.MountPoints {
- if mp.Path == absTarget {
- klog.V(5).Infof("Fake mounter: unmounted %s from %s", mp.Device, absTarget)
- // Don't copy it to newMountpoints
- continue
- }
- newMountpoints = append(newMountpoints, MountPoint{Device: mp.Device, Path: mp.Path, Type: mp.Type})
- }
- f.MountPoints = newMountpoints
- f.Log = append(f.Log, FakeAction{Action: FakeActionUnmount, Target: absTarget})
- delete(f.MountCheckErrors, target)
- return nil
- }
- func (f *FakeMounter) List() ([]MountPoint, error) {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- return f.MountPoints, nil
- }
- func (f *FakeMounter) IsMountPointMatch(mp MountPoint, dir string) bool {
- return mp.Path == dir
- }
- func (f *FakeMounter) IsLikelyNotMountPoint(file string) (bool, error) {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- err := f.MountCheckErrors[file]
- if err != nil {
- return false, err
- }
- _, err = os.Stat(file)
- if err != nil {
- return true, err
- }
- // If file is a symlink, get its absolute path
- absFile, err := filepath.EvalSymlinks(file)
- if err != nil {
- absFile = file
- }
- for _, mp := range f.MountPoints {
- if mp.Path == absFile {
- klog.V(5).Infof("isLikelyNotMountPoint for %s: mounted %s, false", file, mp.Path)
- return false, nil
- }
- }
- klog.V(5).Infof("isLikelyNotMountPoint for %s: true", file)
- return true, nil
- }
- func (f *FakeMounter) DeviceOpened(pathname string) (bool, error) {
- f.mutex.Lock()
- defer f.mutex.Unlock()
- for _, mp := range f.MountPoints {
- if mp.Device == pathname {
- return true, nil
- }
- }
- return false, nil
- }
- func (f *FakeMounter) PathIsDevice(pathname string) (bool, error) {
- return true, nil
- }
- func (f *FakeMounter) GetDeviceNameFromMount(mountPath, pluginMountDir string) (string, error) {
- return getDeviceNameFromMount(f, mountPath, pluginMountDir)
- }
- func (f *FakeMounter) MakeRShared(path string) error {
- return nil
- }
- func (f *FakeMounter) GetFileType(pathname string) (FileType, error) {
- if t, ok := f.Filesystem[pathname]; ok {
- return t, nil
- }
- return FileType("Directory"), nil
- }
- func (f *FakeMounter) MakeDir(pathname string) error {
- return nil
- }
- func (f *FakeMounter) MakeFile(pathname string) error {
- return nil
- }
- func (f *FakeMounter) ExistsPath(pathname string) (bool, error) {
- if _, ok := f.Filesystem[pathname]; ok {
- return true, nil
- }
- return false, nil
- }
- func (f *FakeMounter) EvalHostSymlinks(pathname string) (string, error) {
- return pathname, nil
- }
- func (f *FakeMounter) GetMountRefs(pathname string) ([]string, error) {
- realpath, err := filepath.EvalSymlinks(pathname)
- if err != nil {
- // Ignore error in FakeMounter, because we actually didn't create files.
- realpath = pathname
- }
- return getMountRefsByDev(f, realpath)
- }
- func (f *FakeMounter) GetFSGroup(pathname string) (int64, error) {
- return -1, errors.New("GetFSGroup not implemented")
- }
- func (f *FakeMounter) GetSELinuxSupport(pathname string) (bool, error) {
- return false, errors.New("GetSELinuxSupport not implemented")
- }
- func (f *FakeMounter) GetMode(pathname string) (os.FileMode, error) {
- return 0, errors.New("not implemented")
- }
|