2014-11-23 15:47:25 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2014-11-23 15:47:25 +00:00
|
|
|
|
|
|
|
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 volume
|
|
|
|
|
|
|
|
import (
|
2015-09-07 18:44:49 +00:00
|
|
|
"fmt"
|
2014-11-23 15:47:25 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
2015-08-13 19:01:50 +00:00
|
|
|
client "k8s.io/kubernetes/pkg/client/unversioned"
|
2015-04-10 16:54:01 +00:00
|
|
|
"k8s.io/kubernetes/pkg/cloudprovider"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/types"
|
|
|
|
"k8s.io/kubernetes/pkg/util"
|
2015-09-14 09:51:40 +00:00
|
|
|
"k8s.io/kubernetes/pkg/util/io"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/util/mount"
|
2014-11-23 15:47:25 +00:00
|
|
|
)
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
// fakeVolumeHost is useful for testing volume plugins.
|
|
|
|
type fakeVolumeHost struct {
|
2015-03-07 21:38:50 +00:00
|
|
|
rootDir string
|
|
|
|
kubeClient client.Interface
|
2015-03-19 05:18:31 +00:00
|
|
|
pluginMgr VolumePluginMgr
|
2015-04-10 16:54:01 +00:00
|
|
|
cloud cloudprovider.Interface
|
2015-09-14 09:51:40 +00:00
|
|
|
mounter mount.Interface
|
|
|
|
writer io.Writer
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func NewFakeVolumeHost(rootDir string, kubeClient client.Interface, plugins []VolumePlugin) *fakeVolumeHost {
|
2015-04-10 16:54:01 +00:00
|
|
|
host := &fakeVolumeHost{rootDir: rootDir, kubeClient: kubeClient, cloud: nil}
|
2015-09-14 09:51:40 +00:00
|
|
|
host.mounter = &mount.FakeMounter{}
|
|
|
|
host.writer = &io.StdWriter{}
|
2015-03-07 21:38:50 +00:00
|
|
|
host.pluginMgr.InitPlugins(plugins, host)
|
|
|
|
return host
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (f *fakeVolumeHost) GetPluginDir(podUID string) string {
|
2015-03-07 21:38:50 +00:00
|
|
|
return path.Join(f.rootDir, "plugins", podUID)
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (f *fakeVolumeHost) GetPodVolumeDir(podUID types.UID, pluginName, volumeName string) string {
|
2015-03-07 21:38:50 +00:00
|
|
|
return path.Join(f.rootDir, "pods", string(podUID), "volumes", pluginName, volumeName)
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (f *fakeVolumeHost) GetPodPluginDir(podUID types.UID, pluginName string) string {
|
2015-03-07 21:38:50 +00:00
|
|
|
return path.Join(f.rootDir, "pods", string(podUID), "plugins", pluginName)
|
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (f *fakeVolumeHost) GetKubeClient() client.Interface {
|
2015-03-07 21:38:50 +00:00
|
|
|
return f.kubeClient
|
|
|
|
}
|
|
|
|
|
2015-04-10 16:54:01 +00:00
|
|
|
func (f *fakeVolumeHost) GetCloudProvider() cloudprovider.Interface {
|
|
|
|
return f.cloud
|
|
|
|
}
|
|
|
|
|
2015-09-14 09:51:40 +00:00
|
|
|
func (f *fakeVolumeHost) GetMounter() mount.Interface {
|
|
|
|
return f.mounter
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fakeVolumeHost) GetWriter() io.Writer {
|
|
|
|
return f.writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fakeVolumeHost) NewWrapperBuilder(spec *Spec, pod *api.Pod, opts VolumeOptions) (Builder, error) {
|
2015-03-07 21:38:50 +00:00
|
|
|
plug, err := f.pluginMgr.FindPluginBySpec(spec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-09-14 09:51:40 +00:00
|
|
|
return plug.NewBuilder(spec, pod, opts)
|
2015-03-07 21:38:50 +00:00
|
|
|
}
|
|
|
|
|
2015-09-14 09:51:40 +00:00
|
|
|
func (f *fakeVolumeHost) NewWrapperCleaner(spec *Spec, podUID types.UID) (Cleaner, error) {
|
2015-03-07 21:38:50 +00:00
|
|
|
plug, err := f.pluginMgr.FindPluginBySpec(spec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-09-14 09:51:40 +00:00
|
|
|
return plug.NewCleaner(spec.Name(), podUID)
|
2015-02-18 01:26:41 +00:00
|
|
|
}
|
|
|
|
|
2015-11-24 22:48:41 +00:00
|
|
|
// Returns the hostname of the host kubelet is running on
|
|
|
|
func (f *fakeVolumeHost) GetHostName() string {
|
|
|
|
return "fakeHostName"
|
|
|
|
}
|
|
|
|
|
2015-08-31 13:44:37 +00:00
|
|
|
func ProbeVolumePlugins(config VolumeConfig) []VolumePlugin {
|
|
|
|
if _, ok := config.OtherAttributes["fake-property"]; ok {
|
|
|
|
return []VolumePlugin{
|
|
|
|
&FakeVolumePlugin{
|
|
|
|
PluginName: "fake-plugin",
|
|
|
|
Host: nil,
|
|
|
|
// SomeFakeProperty: config.OtherAttributes["fake-property"] -- string, may require parsing by plugin
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return []VolumePlugin{&FakeVolumePlugin{PluginName: "fake-plugin"}}
|
|
|
|
}
|
|
|
|
|
2015-03-12 19:37:02 +00:00
|
|
|
// FakeVolumePlugin is useful for testing. It tries to be a fully compliant
|
2014-11-23 15:47:25 +00:00
|
|
|
// plugin, but all it does is make empty directories.
|
|
|
|
// Use as:
|
|
|
|
// volume.RegisterPlugin(&FakePlugin{"fake-name"})
|
2015-03-19 05:18:31 +00:00
|
|
|
type FakeVolumePlugin struct {
|
2014-11-23 15:47:25 +00:00
|
|
|
PluginName string
|
2015-03-19 05:18:31 +00:00
|
|
|
Host VolumeHost
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
var _ VolumePlugin = &FakeVolumePlugin{}
|
2015-05-29 20:32:44 +00:00
|
|
|
var _ RecyclableVolumePlugin = &FakeVolumePlugin{}
|
2014-11-23 15:47:25 +00:00
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (plugin *FakeVolumePlugin) Init(host VolumeHost) {
|
2014-11-23 15:47:25 +00:00
|
|
|
plugin.Host = host
|
|
|
|
}
|
|
|
|
|
2015-03-19 05:18:31 +00:00
|
|
|
func (plugin *FakeVolumePlugin) Name() string {
|
2014-11-23 15:47:25 +00:00
|
|
|
return plugin.PluginName
|
|
|
|
}
|
|
|
|
|
2015-04-14 16:29:33 +00:00
|
|
|
func (plugin *FakeVolumePlugin) CanSupport(spec *Spec) bool {
|
2015-08-12 19:11:03 +00:00
|
|
|
// TODO: maybe pattern-match on spec.Name() to decide?
|
2014-11-23 15:47:25 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-09-14 09:51:40 +00:00
|
|
|
func (plugin *FakeVolumePlugin) NewBuilder(spec *Spec, pod *api.Pod, opts VolumeOptions) (Builder, error) {
|
2015-12-04 20:40:01 +00:00
|
|
|
return &FakeVolume{pod.UID, spec.Name(), plugin, MetricsNil{}}, nil
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-09-14 09:51:40 +00:00
|
|
|
func (plugin *FakeVolumePlugin) NewCleaner(volName string, podUID types.UID) (Cleaner, error) {
|
2015-12-04 20:40:01 +00:00
|
|
|
return &FakeVolume{podUID, volName, plugin, MetricsNil{}}, nil
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-05-29 20:32:44 +00:00
|
|
|
func (plugin *FakeVolumePlugin) NewRecycler(spec *Spec) (Recycler, error) {
|
2015-12-04 20:40:01 +00:00
|
|
|
return &fakeRecycler{"/attributesTransferredFromSpec", MetricsNil{}}, nil
|
2015-05-29 20:32:44 +00:00
|
|
|
}
|
|
|
|
|
2015-09-07 16:11:37 +00:00
|
|
|
func (plugin *FakeVolumePlugin) NewDeleter(spec *Spec) (Deleter, error) {
|
2015-12-04 20:40:01 +00:00
|
|
|
return &FakeDeleter{"/attributesTransferredFromSpec", MetricsNil{}}, nil
|
2015-09-07 16:11:37 +00:00
|
|
|
}
|
|
|
|
|
2015-05-18 20:22:30 +00:00
|
|
|
func (plugin *FakeVolumePlugin) GetAccessModes() []api.PersistentVolumeAccessMode {
|
|
|
|
return []api.PersistentVolumeAccessMode{}
|
2015-03-12 19:37:02 +00:00
|
|
|
}
|
|
|
|
|
2014-11-23 15:47:25 +00:00
|
|
|
type FakeVolume struct {
|
|
|
|
PodUID types.UID
|
|
|
|
VolName string
|
2015-03-19 05:18:31 +00:00
|
|
|
Plugin *FakeVolumePlugin
|
2015-12-04 20:40:01 +00:00
|
|
|
MetricsNil
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
2015-10-30 20:25:36 +00:00
|
|
|
func (_ *FakeVolume) GetAttributes() Attributes {
|
|
|
|
return Attributes{
|
|
|
|
ReadOnly: false,
|
|
|
|
Managed: true,
|
|
|
|
SupportsOwnershipManagement: true,
|
|
|
|
SupportsSELinux: true,
|
|
|
|
}
|
2015-10-20 18:49:39 +00:00
|
|
|
}
|
|
|
|
|
2014-11-23 15:47:25 +00:00
|
|
|
func (fv *FakeVolume) SetUp() error {
|
2015-03-07 21:38:50 +00:00
|
|
|
return fv.SetUpAt(fv.GetPath())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FakeVolume) SetUpAt(dir string) error {
|
|
|
|
return os.MkdirAll(dir, 0750)
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FakeVolume) GetPath() string {
|
2015-03-24 14:39:51 +00:00
|
|
|
return path.Join(fv.Plugin.Host.GetPodVolumeDir(fv.PodUID, util.EscapeQualifiedNameForDisk(fv.Plugin.PluginName), fv.VolName))
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FakeVolume) TearDown() error {
|
2015-03-07 21:38:50 +00:00
|
|
|
return fv.TearDownAt(fv.GetPath())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FakeVolume) TearDownAt(dir string) error {
|
|
|
|
return os.RemoveAll(dir)
|
2014-11-23 15:47:25 +00:00
|
|
|
}
|
2015-05-29 20:32:44 +00:00
|
|
|
|
2015-09-07 18:44:49 +00:00
|
|
|
type fakeRecycler struct {
|
2015-05-29 20:32:44 +00:00
|
|
|
path string
|
2015-12-04 20:40:01 +00:00
|
|
|
MetricsNil
|
2015-05-29 20:32:44 +00:00
|
|
|
}
|
|
|
|
|
2015-09-07 18:44:49 +00:00
|
|
|
func (fr *fakeRecycler) Recycle() error {
|
2015-05-29 20:32:44 +00:00
|
|
|
// nil is success, else error
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-09-07 18:44:49 +00:00
|
|
|
func (fr *fakeRecycler) GetPath() string {
|
2015-05-29 20:32:44 +00:00
|
|
|
return fr.path
|
|
|
|
}
|
2015-09-07 18:44:49 +00:00
|
|
|
|
2015-09-03 03:14:26 +00:00
|
|
|
func NewFakeRecycler(spec *Spec, host VolumeHost, config VolumeConfig) (Recycler, error) {
|
2015-09-07 18:44:49 +00:00
|
|
|
if spec.PersistentVolume == nil || spec.PersistentVolume.Spec.HostPath == nil {
|
|
|
|
return nil, fmt.Errorf("fakeRecycler only supports spec.PersistentVolume.Spec.HostPath")
|
|
|
|
}
|
|
|
|
return &fakeRecycler{
|
|
|
|
path: spec.PersistentVolume.Spec.HostPath.Path,
|
|
|
|
}, nil
|
|
|
|
}
|
2015-09-07 16:11:37 +00:00
|
|
|
|
|
|
|
type FakeDeleter struct {
|
|
|
|
path string
|
2015-12-04 20:40:01 +00:00
|
|
|
MetricsNil
|
2015-09-07 16:11:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (fd *FakeDeleter) Delete() error {
|
|
|
|
// nil is success, else error
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fd *FakeDeleter) GetPath() string {
|
|
|
|
return fd.path
|
|
|
|
}
|