2017-08-23 20:06:50 -04:00
|
|
|
package driver
|
|
|
|
|
|
|
|
import (
|
2018-10-31 17:42:24 -04:00
|
|
|
"fmt"
|
2020-08-24 10:54:30 -04:00
|
|
|
"path"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
2020-02-14 11:42:29 -05:00
|
|
|
|
2017-08-23 20:06:50 -04:00
|
|
|
"github.com/vmware/govmomi/object"
|
2020-08-24 10:54:30 -04:00
|
|
|
"github.com/vmware/govmomi/property"
|
2017-08-24 16:18:14 -04:00
|
|
|
"github.com/vmware/govmomi/vim25/mo"
|
2018-02-01 06:47:09 -05:00
|
|
|
"github.com/vmware/govmomi/vim25/soap"
|
2018-10-31 17:42:24 -04:00
|
|
|
"github.com/vmware/govmomi/vim25/types"
|
2017-08-23 20:06:50 -04:00
|
|
|
)
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
type Datastore interface {
|
|
|
|
Info(params ...string) (*mo.Datastore, error)
|
|
|
|
FileExists(path string) bool
|
2020-10-23 14:04:27 -04:00
|
|
|
DirExists(path string) bool
|
2020-08-31 04:34:41 -04:00
|
|
|
Name() string
|
|
|
|
ResolvePath(path string) string
|
|
|
|
UploadFile(src, dst, host string, setHost bool) error
|
|
|
|
Delete(path string) error
|
|
|
|
MakeDirectory(path string) error
|
|
|
|
Reference() types.ManagedObjectReference
|
|
|
|
}
|
|
|
|
|
|
|
|
type DatastoreDriver struct {
|
2018-02-01 06:47:09 -05:00
|
|
|
ds *object.Datastore
|
2020-08-31 04:34:41 -04:00
|
|
|
driver *VCenterDriver
|
2017-08-23 20:06:50 -04:00
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (d *VCenterDriver) NewDatastore(ref *types.ManagedObjectReference) Datastore {
|
|
|
|
return &DatastoreDriver{
|
2018-02-01 06:47:09 -05:00
|
|
|
ds: object.NewDatastore(d.client.Client, *ref),
|
2017-08-24 16:18:14 -04:00
|
|
|
driver: d,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-20 19:03:47 -04:00
|
|
|
// If name is an empty string, then resolve host's one
|
2020-08-31 04:34:41 -04:00
|
|
|
func (d *VCenterDriver) FindDatastore(name string, host string) (Datastore, error) {
|
2018-03-20 19:03:47 -04:00
|
|
|
if name == "" {
|
|
|
|
h, err := d.FindHost(host)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
i, err := h.Info("datastore")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(i.Datastore) > 1 {
|
|
|
|
return nil, fmt.Errorf("Host has multiple datastores. Specify it explicitly")
|
|
|
|
}
|
|
|
|
|
|
|
|
ds := d.NewDatastore(&i.Datastore[0])
|
|
|
|
inf, err := ds.Info("name")
|
2019-07-08 10:48:37 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-03-20 19:03:47 -04:00
|
|
|
name = inf.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
ds, err := d.finder.Datastore(d.ctx, name)
|
2018-01-24 08:35:04 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-03-20 19:03:47 -04:00
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
return &DatastoreDriver{
|
2018-01-24 08:35:04 -05:00
|
|
|
ds: ds,
|
|
|
|
driver: d,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (d *VCenterDriver) GetDatastoreName(id string) (string, error) {
|
2020-08-24 10:54:30 -04:00
|
|
|
obj := types.ManagedObjectReference{
|
|
|
|
Type: "Datastore",
|
|
|
|
Value: id,
|
|
|
|
}
|
|
|
|
pc := property.DefaultCollector(d.vimClient)
|
|
|
|
var me mo.ManagedEntity
|
|
|
|
|
|
|
|
err := pc.RetrieveOne(d.ctx, obj, []string{"name"}, &me)
|
|
|
|
if err != nil {
|
|
|
|
return id, err
|
|
|
|
}
|
|
|
|
return me.Name, nil
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) Info(params ...string) (*mo.Datastore, error) {
|
2017-08-24 16:18:14 -04:00
|
|
|
var p []string
|
|
|
|
if len(params) == 0 {
|
|
|
|
p = []string{"*"}
|
|
|
|
} else {
|
|
|
|
p = params
|
|
|
|
}
|
|
|
|
var info mo.Datastore
|
|
|
|
err := ds.ds.Properties(ds.driver.ctx, ds.ds.Reference(), p, &info)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &info, nil
|
|
|
|
}
|
2018-01-24 08:35:04 -05:00
|
|
|
|
2020-10-23 14:04:27 -04:00
|
|
|
func (ds *DatastoreDriver) DirExists(filepath string) bool {
|
|
|
|
_, err := ds.ds.Stat(ds.driver.ctx, filepath)
|
|
|
|
if _, ok := err.(object.DatastoreNoSuchDirectoryError); ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) FileExists(path string) bool {
|
2018-01-24 08:35:04 -05:00
|
|
|
_, err := ds.ds.Stat(ds.driver.ctx, path)
|
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) Name() string {
|
2018-01-24 08:35:04 -05:00
|
|
|
return ds.ds.Name()
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) Reference() types.ManagedObjectReference {
|
|
|
|
return ds.ds.Reference()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ds *DatastoreDriver) ResolvePath(path string) string {
|
2018-01-24 08:35:04 -05:00
|
|
|
return ds.ds.Path(path)
|
|
|
|
}
|
2018-02-01 06:47:09 -05:00
|
|
|
|
2020-08-24 10:54:30 -04:00
|
|
|
// The file ID isn't available via the API, so we use DatastoreBrowser to search
|
2020-08-31 04:34:41 -04:00
|
|
|
func (d *VCenterDriver) GetDatastoreFilePath(datastoreID, dir, filename string) (string, error) {
|
2020-08-24 10:54:30 -04:00
|
|
|
ref := types.ManagedObjectReference{Type: "Datastore", Value: datastoreID}
|
|
|
|
ds := object.NewDatastore(d.vimClient, ref)
|
|
|
|
|
|
|
|
b, err := ds.Browser(d.ctx)
|
|
|
|
if err != nil {
|
|
|
|
return filename, err
|
|
|
|
}
|
|
|
|
ext := path.Ext(filename)
|
|
|
|
pat := strings.Replace(filename, ext, "*"+ext, 1)
|
|
|
|
spec := types.HostDatastoreBrowserSearchSpec{
|
|
|
|
MatchPattern: []string{pat},
|
|
|
|
}
|
|
|
|
|
|
|
|
task, err := b.SearchDatastore(d.ctx, dir, &spec)
|
|
|
|
if err != nil {
|
|
|
|
return filename, err
|
|
|
|
}
|
|
|
|
|
|
|
|
info, err := task.WaitForResult(d.ctx, nil)
|
|
|
|
if err != nil {
|
|
|
|
return filename, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res, ok := info.Result.(types.HostDatastoreBrowserSearchResults)
|
|
|
|
if !ok {
|
|
|
|
return filename, fmt.Errorf("search(%s) result type=%T", pat, info.Result)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(res.File) != 1 {
|
|
|
|
return filename, fmt.Errorf("search(%s) result files=%d", pat, len(res.File))
|
|
|
|
}
|
|
|
|
return res.File[0].GetFileInfo().Path, nil
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) UploadFile(src, dst, host string, setHost bool) error {
|
2018-02-01 06:47:09 -05:00
|
|
|
p := soap.DefaultUpload
|
2019-01-04 16:55:04 -05:00
|
|
|
ctx := ds.driver.ctx
|
2020-05-05 14:34:15 -04:00
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
if setHost && host != "" {
|
2020-05-05 14:34:15 -04:00
|
|
|
h, err := ds.driver.FindHost(host)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
ctx = ds.ds.HostContext(ctx, h.host)
|
|
|
|
}
|
|
|
|
|
2018-11-13 12:22:19 -05:00
|
|
|
return ds.ds.UploadFile(ctx, src, dst, &p)
|
2018-02-01 06:47:09 -05:00
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) Delete(path string) error {
|
2018-02-01 06:47:09 -05:00
|
|
|
dc, err := ds.driver.finder.Datacenter(ds.driver.ctx, ds.ds.DatacenterPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fm := ds.ds.NewFileManager(dc, false)
|
|
|
|
return fm.Delete(ds.driver.ctx, path)
|
|
|
|
}
|
|
|
|
|
2020-08-31 04:34:41 -04:00
|
|
|
func (ds *DatastoreDriver) MakeDirectory(path string) error {
|
2019-01-04 09:07:09 -05:00
|
|
|
dc, err := ds.driver.finder.Datacenter(ds.driver.ctx, ds.ds.DatacenterPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fm := ds.ds.NewFileManager(dc, false)
|
|
|
|
return fm.FileManager.MakeDirectory(ds.driver.ctx, path, dc, true)
|
|
|
|
}
|
|
|
|
|
2018-02-01 06:47:09 -05:00
|
|
|
// Cuts out the datastore prefix
|
|
|
|
// Example: "[datastore1] file.ext" --> "file.ext"
|
|
|
|
func RemoveDatastorePrefix(path string) string {
|
|
|
|
res := object.DatastorePath{}
|
|
|
|
if hadPrefix := res.FromString(path); hadPrefix {
|
|
|
|
return res.Path
|
|
|
|
} else {
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
}
|
2020-08-24 10:54:30 -04:00
|
|
|
|
|
|
|
type DatastoreIsoPath struct {
|
|
|
|
path string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DatastoreIsoPath) Validate() bool {
|
|
|
|
// Matches:
|
|
|
|
// [datastore] /dir/subdir/file
|
|
|
|
// [datastore] dir/subdir/file
|
|
|
|
// [] /dir/subdir/file
|
2020-09-01 04:11:48 -04:00
|
|
|
// [data-store] /dir/subdir/file
|
|
|
|
// dir/subdir/file or dir/subdir/file
|
|
|
|
matched, _ := regexp.MatchString(`^\s*(\[[^\[\]\/]*\])?\s*[^\[\]]+\s*$`, d.path)
|
2020-08-24 10:54:30 -04:00
|
|
|
return matched
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DatastoreIsoPath) GetFilePath() string {
|
|
|
|
filePath := d.path
|
|
|
|
parts := strings.Split(d.path, "]")
|
|
|
|
if len(parts) > 1 {
|
|
|
|
// removes datastore name from path
|
|
|
|
filePath = parts[1]
|
2020-09-01 04:11:48 -04:00
|
|
|
filePath = strings.TrimSpace(filePath)
|
2020-08-24 10:54:30 -04:00
|
|
|
}
|
|
|
|
return filePath
|
|
|
|
}
|