mirror of
https://github.com/kemko/nomad.git
synced 2026-01-05 01:45:44 +03:00
139 lines
4.1 KiB
Go
139 lines
4.1 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package client
|
|
|
|
import (
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/hashicorp/nomad/ci"
|
|
hvm "github.com/hashicorp/nomad/client/hostvolumemanager"
|
|
"github.com/hashicorp/nomad/client/state"
|
|
cstructs "github.com/hashicorp/nomad/client/structs"
|
|
"github.com/hashicorp/nomad/helper/testlog"
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
|
"github.com/shoenig/test/must"
|
|
)
|
|
|
|
func TestHostVolume(t *testing.T) {
|
|
ci.Parallel(t)
|
|
|
|
client, cleanup := TestClient(t, nil)
|
|
defer cleanup()
|
|
|
|
memdb := state.NewMemDB(testlog.HCLogger(t))
|
|
client.stateDB = memdb
|
|
|
|
tmp := t.TempDir()
|
|
manager := hvm.NewHostVolumeManager(testlog.HCLogger(t), hvm.Config{
|
|
StateMgr: client.stateDB,
|
|
UpdateNodeVols: client.updateNodeFromHostVol,
|
|
PluginDir: "/no/ext/plugins",
|
|
SharedMountDir: tmp,
|
|
})
|
|
client.hostVolumeManager = manager
|
|
expectDir := filepath.Join(tmp, "test-vol-id")
|
|
|
|
t.Run("happy", func(t *testing.T) {
|
|
|
|
/* create */
|
|
|
|
req := &cstructs.ClientHostVolumeCreateRequest{
|
|
Name: "test-vol-name",
|
|
ID: "test-vol-id",
|
|
PluginID: "mkdir", // real plugin really makes a dir
|
|
}
|
|
var resp cstructs.ClientHostVolumeCreateResponse
|
|
err := client.ClientRPC("HostVolume.Create", req, &resp)
|
|
must.NoError(t, err)
|
|
must.Eq(t, cstructs.ClientHostVolumeCreateResponse{
|
|
HostPath: expectDir,
|
|
CapacityBytes: 0, // "mkdir" always returns zero
|
|
}, resp)
|
|
// technically this is testing "mkdir" more than the RPC
|
|
must.DirExists(t, expectDir)
|
|
// ensure we saved to client state
|
|
vols, err := memdb.GetDynamicHostVolumes()
|
|
must.NoError(t, err)
|
|
must.Len(t, 1, vols)
|
|
expectState := &cstructs.HostVolumeState{
|
|
ID: req.ID,
|
|
CreateReq: req,
|
|
}
|
|
must.Eq(t, expectState, vols[0])
|
|
// and should be fingerprinted
|
|
must.Eq(t, hvm.VolumeMap{
|
|
req.Name: {
|
|
ID: req.ID,
|
|
Name: req.Name,
|
|
Path: expectDir,
|
|
},
|
|
}, client.Node().HostVolumes)
|
|
|
|
/* delete */
|
|
|
|
delReq := &cstructs.ClientHostVolumeDeleteRequest{
|
|
Name: "test-vol-name",
|
|
ID: "test-vol-id",
|
|
PluginID: "mkdir",
|
|
HostPath: expectDir,
|
|
}
|
|
var delResp cstructs.ClientHostVolumeDeleteResponse
|
|
err = client.ClientRPC("HostVolume.Delete", delReq, &delResp)
|
|
must.NoError(t, err)
|
|
must.NotNil(t, delResp)
|
|
// again, actually testing the "mkdir" plugin
|
|
must.DirNotExists(t, expectDir)
|
|
// client state should be deleted
|
|
vols, err = memdb.GetDynamicHostVolumes()
|
|
must.NoError(t, err)
|
|
must.Len(t, 0, vols)
|
|
// and the fingerprint, too
|
|
must.Eq(t, map[string]*structs.ClientHostVolumeConfig{}, client.Node().HostVolumes)
|
|
})
|
|
|
|
t.Run("missing plugin", func(t *testing.T) {
|
|
req := &cstructs.ClientHostVolumeCreateRequest{
|
|
PluginID: "non-existent",
|
|
}
|
|
var resp cstructs.ClientHostVolumeCreateResponse
|
|
err := client.ClientRPC("HostVolume.Create", req, &resp)
|
|
must.EqError(t, err, `no such plugin: "non-existent"`)
|
|
|
|
delReq := &cstructs.ClientHostVolumeDeleteRequest{
|
|
PluginID: "non-existent",
|
|
}
|
|
var delResp cstructs.ClientHostVolumeDeleteResponse
|
|
err = client.ClientRPC("HostVolume.Delete", delReq, &delResp)
|
|
must.EqError(t, err, `no such plugin: "non-existent"`)
|
|
})
|
|
|
|
t.Run("error from plugin", func(t *testing.T) {
|
|
// "mkdir" plugin can't create a directory within a file
|
|
client.hostVolumeManager = hvm.NewHostVolumeManager(testlog.HCLogger(t), hvm.Config{
|
|
StateMgr: client.stateDB,
|
|
UpdateNodeVols: client.updateNodeFromHostVol,
|
|
PluginDir: "/no/ext/plugins",
|
|
SharedMountDir: "host_volume_endpoint_test.go",
|
|
})
|
|
|
|
req := &cstructs.ClientHostVolumeCreateRequest{
|
|
ID: "test-vol-id",
|
|
Name: "test-vol-name",
|
|
PluginID: "mkdir",
|
|
}
|
|
var resp cstructs.ClientHostVolumeCreateResponse
|
|
err := client.ClientRPC("HostVolume.Create", req, &resp)
|
|
must.ErrorContains(t, err, "host_volume_endpoint_test.go/test-vol-id: not a directory")
|
|
|
|
delReq := &cstructs.ClientHostVolumeDeleteRequest{
|
|
ID: "test-vol-id",
|
|
PluginID: "mkdir",
|
|
}
|
|
var delResp cstructs.ClientHostVolumeDeleteResponse
|
|
err = client.ClientRPC("HostVolume.Delete", delReq, &delResp)
|
|
must.ErrorContains(t, err, "host_volume_endpoint_test.go/test-vol-id: not a directory")
|
|
})
|
|
}
|