This repository has been archived by the owner on Sep 24, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathsmb_mounter_unix.go
executable file
·166 lines (138 loc) · 4.62 KB
/
smb_mounter_unix.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
//go:build linux || darwin
// +build linux darwin
package smbdriver
import (
"context"
"fmt"
"path/filepath"
"strings"
"time"
"code.cloudfoundry.org/dockerdriver"
"code.cloudfoundry.org/dockerdriver/driverhttp"
"code.cloudfoundry.org/goshims/ioutilshim"
"code.cloudfoundry.org/goshims/osshim"
"code.cloudfoundry.org/lager/v3"
vmo "code.cloudfoundry.org/volume-mount-options"
"code.cloudfoundry.org/volumedriver"
"code.cloudfoundry.org/volumedriver/invoker"
)
type smbMounter struct {
invoker invoker.Invoker
osutil osshim.Os
ioutil ioutilshim.Ioutil
configMask vmo.MountOptsMask
forceNoserverino bool
forceNoDfs bool
}
func NewSmbMounter(invoker invoker.Invoker, osutil osshim.Os, ioutil ioutilshim.Ioutil, configMask vmo.MountOptsMask, forceNoserverino, forceNoDfs bool) volumedriver.Mounter {
return &smbMounter{invoker: invoker, osutil: osutil, ioutil: ioutil, configMask: configMask, forceNoserverino: forceNoserverino, forceNoDfs: forceNoDfs}
}
func (m *smbMounter) Mount(env dockerdriver.Env, source string, target string, opts map[string]interface{}) error {
logger := env.Logger().Session("smb-mount")
logger.Info("start")
defer logger.Info("end")
mountOpts, err := vmo.NewMountOpts(opts, m.configMask)
if err != nil {
logger.Debug("error-parse-entries", lager.Data{
"given_source": source,
"given_target": target,
"given_options": opts,
})
return safeError(err)
}
mountFlags, mountEnvVars := ToKernelMountOptionFlagsAndEnvVars(mountOpts)
mountFlags = fmt.Sprintf("%s,uid=2000,gid=2000", mountFlags)
if m.forceNoserverino {
mountFlags = fmt.Sprintf("%s,noserverino", mountFlags)
}
if m.forceNoDfs {
mountFlags = fmt.Sprintf("%s,nodfs", mountFlags)
}
mountArgs := []string{
"-t", "cifs",
source,
target,
"-o", mountFlags,
"--verbose",
}
logger.Debug("parse-mount", lager.Data{
"given_source": source,
"given_target": target,
"given_options": opts,
"mountArgs": mountArgs,
})
logger.Debug("mount", lager.Data{"params": strings.Join(mountArgs, ",")})
invokeResult := m.invoker.Invoke(env, "mount", mountArgs, mountEnvVars...)
return safeError(invokeResult.Wait())
}
func (m *smbMounter) Unmount(env dockerdriver.Env, target string) error {
logger := env.Logger().Session("smb-umount")
logger.Info("start")
defer logger.Info("end")
invokeResult := m.invoker.Invoke(env, "umount", []string{"-l", target})
err := invokeResult.Wait()
if err != nil {
return safeError(err)
}
return nil
}
func (m *smbMounter) Check(env dockerdriver.Env, name, mountPoint string) bool {
logger := env.Logger().Session("smb-check-mountpoint")
logger.Info("start")
defer logger.Info("end")
ctx, cancel := context.WithDeadline(context.TODO(), time.Now().Add(time.Second*5))
defer cancel()
env = driverhttp.EnvWithContext(ctx, env)
invokeResult := m.invoker.Invoke(env, "mountpoint", []string{"-q", mountPoint})
err := invokeResult.Wait()
if err != nil {
// Note: Created volumes (with no mounts) will be removed
// since VolumeInfo.Mountpoint will be an empty string
logger.Info(fmt.Sprintf("unable to verify volume %s (%s)", name, err.Error()))
return false
}
return true
}
func (m *smbMounter) Purge(env dockerdriver.Env, path string) {
logger := env.Logger().Session("purge")
logger.Info("start")
defer logger.Info("end")
fileInfos, err := m.ioutil.ReadDir(path)
if err != nil {
logger.Error("purge-readdir-failed", err, lager.Data{"path": path})
return
}
for _, fileInfo := range fileInfos {
if fileInfo.IsDir() {
mountDir := filepath.Join(path, fileInfo.Name())
err = m.invoker.Invoke(env, "umount", []string{"-l", "-f", mountDir}).Wait()
if err != nil {
logger.Error("warning-umount-failed", err)
} else {
logger.Info("unmount-successful", lager.Data{"path": mountDir})
}
if err := m.osutil.Remove(mountDir); err != nil {
logger.Error("purge-cannot-remove-directory", err, lager.Data{"name": mountDir, "path": path})
}
logger.Info("remove-directory-successful", lager.Data{"path": mountDir})
}
}
}
func NewSmbVolumeMountMask() (vmo.MountOptsMask, error) {
allowed := []string{"mfsymlinks", "username", "password", "file_mode", "dir_mode", "ro", "domain", "vers", "sec", "version",
"noserverino", "forceuid", "noforceuid", "forcegid", "noforcegid", "nodfs"}
defaultMap := map[string]interface{}{}
return vmo.NewMountOptsMask(
allowed,
defaultMap,
map[string]string{"readonly": "ro", "version": "vers"},
[]string{"source", "mount"},
[]string{"username", "password"},
)
}
func safeError(e error) error {
if e == nil {
return nil
}
return dockerdriver.SafeError{SafeDescription: e.Error()}
}