Skip to content

Commit

Permalink
add 9pfs volume(one 9p mount per fs volume)
Browse files Browse the repository at this point in the history
Signed-off-by: Yanqiang Miao <[email protected]>
  • Loading branch information
Yanqiang Miao committed Aug 25, 2018
1 parent bbacea5 commit 66a9e12
Show file tree
Hide file tree
Showing 29 changed files with 318 additions and 72 deletions.
2 changes: 1 addition & 1 deletion cmd/virtlet/virtlet.go
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,7 @@ func runTapManager(config *v1.VirtletConfig) {
glog.Errorf("FD server returned error: %v", err)
os.Exit(1)
}
if err := libvirttools.ChownForEmulator(*config.FDServerSocketPath); err != nil {
if err := libvirttools.ChownForEmulator(*config.FDServerSocketPath, false); err != nil {
glog.Warningf("Couldn't set tapmanager socket permissions: %v", err)
}
for {
Expand Down
5 changes: 5 additions & 0 deletions deploy/data/virtlet-ds.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,7 @@ spec:
mountPath: /dev
- mountPath: /var/lib/virtlet
name: virtlet
mountPropagation: Bidirectional
- mountPath: /var/lib/libvirt
name: libvirt
- mountPath: /var/run/libvirt
Expand Down Expand Up @@ -222,10 +223,14 @@ spec:
volumeMounts:
- mountPath: /var/lib/virtlet
name: virtlet
mountPropagation: HostToContainer
- mountPath: /var/lib/libvirt
name: libvirt
- name: vms-log
mountPath: /var/log/vms
- mountPath: /var/lib/kubelet/pods
name: k8s-pods-dir
mountPropagation: HostToContainer
- name: dev
mountPath: /dev
- name: modules
Expand Down
4 changes: 4 additions & 0 deletions pkg/flexvolume/flexvolume.go
Original file line number Diff line number Diff line change
Expand Up @@ -269,6 +269,10 @@ func formatResult(fields map[string]interface{}, err error) string {
// means that no partition number was specified.
func GetFlexvolumeInfo(dir string) (string, int, error) {
dataFile := filepath.Join(dir, flexvolumeDataFile)
if _, err := os.Stat(dataFile); os.IsNotExist(err) {
return "", 0, err
}

var opts map[string]interface{}
if err := utils.ReadJSON(dataFile, &opts); err != nil {
return "", 0, fmt.Errorf("can't read flexvolume data file %q: %v", dataFile, err)
Expand Down
3 changes: 3 additions & 0 deletions pkg/libvirttools/TestContainerLifecycle.out.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,7 @@
UserDataOverwrite: false
UserDataScript: ""
VCPUCount: 1
VirtletChown9pfsMounts: false
PodAnnotations:
hello: world
virt: let
Expand Down Expand Up @@ -176,6 +177,7 @@
UserDataOverwrite: false
UserDataScript: ""
VCPUCount: 1
VirtletChown9pfsMounts: false
PodAnnotations:
hello: world
virt: let
Expand Down Expand Up @@ -225,6 +227,7 @@
UserDataOverwrite: false
UserDataScript: ""
VCPUCount: 1
VirtletChown9pfsMounts: false
PodAnnotations:
hello: world
virt: let
Expand Down
1 change: 1 addition & 0 deletions pkg/libvirttools/TestDomainForcedShutdown.out.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,7 @@
UserDataOverwrite: false
UserDataScript: ""
VCPUCount: 1
VirtletChown9pfsMounts: false
PodAnnotations:
hello: world
virt: let
Expand Down
12 changes: 6 additions & 6 deletions pkg/libvirttools/ceph_flexvolume.go
Original file line number Diff line number Diff line change
Expand Up @@ -93,24 +93,24 @@ func (v *cephVolume) UUID() string {
return v.opts.UUID
}

func (v *cephVolume) Setup() (*libvirtxml.DomainDisk, error) {
func (v *cephVolume) Setup() (*libvirtxml.DomainDisk, *libvirtxml.DomainFilesystem, error) {
ipPortPair := strings.Split(v.opts.Monitor, ":")
if len(ipPortPair) != 2 {
return nil, fmt.Errorf("invalid format of ceph monitor setting: %s. Expected ip:port", v.opts.Monitor)
return nil, nil, fmt.Errorf("invalid format of ceph monitor setting: %s. Expected ip:port", v.opts.Monitor)
}

secret, err := v.owner.DomainConnection().DefineSecret(v.secretDef())
if err != nil {
return nil, fmt.Errorf("error defining ceph secret: %v", err)
return nil, nil, fmt.Errorf("error defining ceph secret: %v", err)
}

key, err := base64.StdEncoding.DecodeString(v.opts.Secret)
if err != nil {
return nil, fmt.Errorf("error decoding ceph secret: %v", err)
return nil, nil, fmt.Errorf("error decoding ceph secret: %v", err)
}

if err := secret.SetValue([]byte(key)); err != nil {
return nil, fmt.Errorf("error setting value of secret %q: %v", v.secretUsageName(), err)
return nil, nil, fmt.Errorf("error setting value of secret %q: %v", v.secretUsageName(), err)
}

return &libvirtxml.DomainDisk{
Expand All @@ -135,7 +135,7 @@ func (v *cephVolume) Setup() (*libvirtxml.DomainDisk, error) {
},
},
},
}, nil
}, nil, nil
}

func (v *cephVolume) Teardown() error {
Expand Down
81 changes: 57 additions & 24 deletions pkg/libvirttools/cloudinit.go
Original file line number Diff line number Diff line change
Expand Up @@ -467,43 +467,76 @@ func (g *CloudInitGenerator) generateMounts(volumeMap diskPathMap) ([]interface{
var mountScriptLines []string
for _, m := range g.config.Mounts {
// Skip file based mounts (including secrets and config maps).
if isRegularFile(m.HostPath) {
if isRegularFile(m.HostPath) ||
strings.Contains(m.HostPath, "kubernetes.io~secret") ||
strings.Contains(m.HostPath, "kubernetes.io~configmap") {
continue
}

uuid, part, err := flexvolume.GetFlexvolumeInfo(m.HostPath)
mountInfo, mountScriptLine, err := generateFlexvolumeMounts(volumeMap, m)
if err != nil {
glog.Errorf("Can't mount directory %q to %q inside the VM: can't get flexvolume uuid: %v", m.HostPath, m.ContainerPath, err)
continue
}
dpath, found := volumeMap[uuid]
if !found {
glog.Errorf("Can't mount directory %q to %q inside the VM: no device found for flexvolume uuid %q", m.HostPath, m.ContainerPath, uuid)
continue
}
if part < 0 {
part = 1
}
devPath := dpath.devPath
mountDevSuffix := ""
if part != 0 {
devPath += fmt.Sprintf("-part%d", part)
mountDevSuffix += strconv.Itoa(part)
if !os.IsNotExist(err) {
glog.Errorf("Can't mount directory %q to %q inside the VM: %v", m.HostPath, m.ContainerPath, err)
continue
}

// Fs based volume
mountInfo, mountScriptLine, err = generateFsBasedVolumeMounts(m)
if err != nil {
glog.Errorf("Can't mount directory %q to %q inside the VM: %v", m.HostPath, m.ContainerPath, err)
continue
}
}
r = append(r, []interface{}{devPath, m.ContainerPath})
mountScriptLines = append(
mountScriptLines,
// TODO: do better job at escaping m.ContainerPath
fmt.Sprintf("if ! mountpoint '%s'; then mkdir -p '%s' && mount /dev/`ls %s`%s '%s'; fi",
m.ContainerPath, m.ContainerPath, dpath.sysfsPath, mountDevSuffix, m.ContainerPath))

r = append(r, mountInfo)
mountScriptLines = append(mountScriptLines, mountScriptLine)
}

mountScript := ""
if len(mountScriptLines) != 0 {
mountScript = fmt.Sprintf("#!/bin/sh\n%s\n", strings.Join(mountScriptLines, "\n"))
}
return r, mountScript
}

func generateFlexvolumeMounts(volumeMap diskPathMap, mount types.VMMount) ([]interface{}, string, error) {
uuid, part, err := flexvolume.GetFlexvolumeInfo(mount.HostPath)
if err != nil {
// If the error is NotExist, return the original error
if os.IsNotExist(err) {
return nil, "", err
}
err = fmt.Errorf("can't get flexvolume uuid: %v", err)
return nil, "", err
}
dpath, found := volumeMap[uuid]
if !found {
err = fmt.Errorf("no device found for flexvolume uuid %q", uuid)
return nil, "", err
}
if part < 0 {
part = 1
}
devPath := dpath.devPath
mountDevSuffix := ""
if part != 0 {
devPath += fmt.Sprintf("-part%d", part)
mountDevSuffix += strconv.Itoa(part)
}
// TODO: do better job at escaping mount.ContainerPath
mountScriptLine := fmt.Sprintf("if ! mountpoint '%s'; then mkdir -p '%s' && mount /dev/`ls %s`%s '%s'; fi",
mount.ContainerPath, mount.ContainerPath, dpath.sysfsPath, mountDevSuffix, mount.ContainerPath)
return []interface{}{devPath, mount.ContainerPath}, mountScriptLine, nil
}

func generateFsBasedVolumeMounts(mount types.VMMount) ([]interface{}, string, error) {
mountTag := path.Base(mount.ContainerPath)
fsMountScript := fmt.Sprintf("if ! mountpoint '%s'; then mkdir -p '%s' && mount -t 9p -o trans=virtio %s '%s'; fi",
mount.ContainerPath, mount.ContainerPath, mountTag, mount.ContainerPath)
r := []interface{}{mountTag, mount.ContainerPath, "9p", "trans=virtio"}
return r, fsMountScript, nil
}

type writeFilesUpdater struct {
entries []interface{}
mounts []types.VMMount
Expand Down
4 changes: 2 additions & 2 deletions pkg/libvirttools/config_volumesource.go
Original file line number Diff line number Diff line change
Expand Up @@ -51,8 +51,8 @@ func (v *configVolume) cloudInitGenerator() *CloudInitGenerator {
return NewCloudInitGenerator(v.config, configIsoDir)
}

func (v *configVolume) Setup() (*libvirtxml.DomainDisk, error) {
return v.cloudInitGenerator().DiskDef(), nil
func (v *configVolume) Setup() (*libvirtxml.DomainDisk, *libvirtxml.DomainFilesystem, error) {
return v.cloudInitGenerator().DiskDef(), nil, nil
}

func (v *configVolume) WriteImage(volumeMap diskPathMap) error {
Expand Down
1 change: 1 addition & 0 deletions pkg/libvirttools/defaultvolumesrc.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ func GetDefaultVolumeSource() VMVolumeSource {
return CombineVMVolumeSources(
GetRootVolume,
ScanFlexVolumes,
GetFileSystemVolumes,
// XXX: GetConfigVolume must go last because it
// doesn't produce correct name for cdrom devices
GetConfigVolume)
Expand Down
32 changes: 20 additions & 12 deletions pkg/libvirttools/disklist.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,14 +32,16 @@ type diskItem struct {
volume VMVolume
}

func (di *diskItem) setup(config *types.VMConfig) (*libvirtxml.DomainDisk, error) {
diskDef, err := di.volume.Setup()
func (di *diskItem) setup(config *types.VMConfig) (*libvirtxml.DomainDisk, *libvirtxml.DomainFilesystem, error) {
diskDef, fsDef, err := di.volume.Setup()
if err != nil {
return nil, err
return nil, nil, err
}
if diskDef != nil {
diskDef.Target = di.driver.target()
diskDef.Address = di.driver.address()
}
diskDef.Target = di.driver.target()
diskDef.Address = di.driver.address()
return diskDef, nil
return diskDef, fsDef, nil
}

type diskList struct {
Expand Down Expand Up @@ -72,23 +74,29 @@ func newDiskList(config *types.VMConfig, source VMVolumeSource, owner volumeOwne
}

// setup performs the setup procedure on each volume in the diskList
// and returns a list of libvirtxml DomainDisk structs
func (dl *diskList) setup() ([]libvirtxml.DomainDisk, error) {
// and returns a list of libvirtxml DomainDisk and domainFileSystems structs
func (dl *diskList) setup() ([]libvirtxml.DomainDisk, []libvirtxml.DomainFilesystem, error) {
var domainDisks []libvirtxml.DomainDisk
var domainFileSystems []libvirtxml.DomainFilesystem
for n, item := range dl.items {
diskDef, err := item.setup(dl.config)
diskDef, fsDef, err := item.setup(dl.config)
if err != nil {
// try to tear down volumes that were already set up
for _, item := range dl.items[:n] {
if err := item.volume.Teardown(); err != nil {
glog.Warningf("Failed to tear down a volume on error: %v", err)
}
}
return nil, err
return nil, nil, err
}
if diskDef != nil {
domainDisks = append(domainDisks, *diskDef)
}
if fsDef != nil {
domainFileSystems = append(domainFileSystems, *fsDef)
}
domainDisks = append(domainDisks, *diskDef)
}
return domainDisks, nil
return domainDisks, domainFileSystems, nil
}

// writeImages writes images for volumes that are based on generated
Expand Down
27 changes: 24 additions & 3 deletions pkg/libvirttools/fileownership.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,8 +20,11 @@ import (
"fmt"
"os"
"os/user"
"path/filepath"
"strconv"
"sync"

"github.com/golang/glog"
)

const (
Expand All @@ -35,7 +38,7 @@ var emulatorUser struct {
}

// ChownForEmulator makes a file or directory owned by the emulator user.
func ChownForEmulator(filePath string) error {
func ChownForEmulator(filePath string, recursive bool) error {
emulatorUser.Lock()
defer emulatorUser.Unlock()
if !emulatorUser.initialized {
Expand All @@ -52,8 +55,26 @@ func ChownForEmulator(filePath string) error {
return fmt.Errorf("bad gid %q for user %q: %v", u.Gid, emulatorUserName, err)
}
}
if err := os.Chown(filePath, emulatorUser.uid, emulatorUser.gid); err != nil {
return fmt.Errorf("can't set the owner of tapmanager socket: %v", err)

chown := os.Chown
if recursive {
chown = ChownR
}
if err := chown(filePath, emulatorUser.uid, emulatorUser.gid); err != nil {
return fmt.Errorf("can't set the owner of %q: %v", filePath, err)
}
return nil
}

// ChownR makes a file or directory owned by the emulator user recursively.
func ChownR(path string, uid, gid int) error {
return filepath.Walk(path, func(name string, info os.FileInfo, err error) error {
if err == nil {
err = os.Chown(name, uid, gid)
if err != nil {
glog.Warningf("Failed to change the owner of %q: %v", name, err)
}
}
return err
})
}
52 changes: 52 additions & 0 deletions pkg/libvirttools/filesystem_volumesource.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
/*
Copyright 2018 Mirantis
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 libvirttools

import (
"fmt"
"os"
"path"
"strings"

"github.com/Mirantis/virtlet/pkg/metadata/types"
)

// GetFileSystemVolumes using prepared by kubelet volumes and contained in pod sandbox
// annotations prepares volumes to be passed to libvirt as a DomainFileSystem definitions.
func GetFileSystemVolumes(config *types.VMConfig, owner volumeOwner) ([]VMVolume, error) {
volumePoolPath := supportedStoragePools[owner.VolumePoolName()]
if _, err := os.Stat(volumePoolPath); err != nil {
return nil, err
}

var fsVolumes []VMVolume
for index, mount := range config.Mounts {
if isRegularFile(mount.HostPath) ||
strings.Contains(mount.HostPath, flexvolumeSubdir) ||
strings.Contains(mount.HostPath, "kubernetes.io~secret") ||
strings.Contains(mount.HostPath, "kubernetes.io~configmap") {
continue
}

// `Index` is used to avoid causing conflicts as multiple host paths can have the same `path.Base`
volumeDirName := fmt.Sprintf("virtlet_%s_%s_%d", config.DomainUUID, path.Base(mount.HostPath), index)
volumeMountPoint := path.Join(volumePoolPath, volumeDirName)
fsVolumes = append(fsVolumes, &filesystemVolume{mount: mount, volumeMountPoint: volumeMountPoint})
}

return fsVolumes, nil
}
Loading

0 comments on commit 66a9e12

Please sign in to comment.