github.com/containerd/Containerd@v1.4.13/snapshots/devmapper/pool_device_test.go (about)

     1  // +build linux
     2  
     3  /*
     4     Copyright The containerd Authors.
     5  
     6     Licensed under the Apache License, Version 2.0 (the "License");
     7     you may not use this file except in compliance with the License.
     8     You may obtain a copy of the License at
     9  
    10         http://www.apache.org/licenses/LICENSE-2.0
    11  
    12     Unless required by applicable law or agreed to in writing, software
    13     distributed under the License is distributed on an "AS IS" BASIS,
    14     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    15     See the License for the specific language governing permissions and
    16     limitations under the License.
    17  */
    18  
    19  package devmapper
    20  
    21  import (
    22  	"context"
    23  	"fmt"
    24  	"io/ioutil"
    25  	"os"
    26  	"os/exec"
    27  	"path/filepath"
    28  	"testing"
    29  	"time"
    30  
    31  	"github.com/containerd/containerd/mount"
    32  	"github.com/containerd/containerd/pkg/testutil"
    33  	"github.com/containerd/containerd/snapshots/devmapper/dmsetup"
    34  	"github.com/containerd/containerd/snapshots/devmapper/losetup"
    35  	"github.com/docker/go-units"
    36  	"github.com/sirupsen/logrus"
    37  	"gotest.tools/v3/assert"
    38  )
    39  
    40  const (
    41  	thinDevice1 = "thin-1"
    42  	thinDevice2 = "thin-2"
    43  	snapDevice1 = "snap-1"
    44  	device1Size = 100000
    45  	device2Size = 200000
    46  	testsPrefix = "devmapper-snapshotter-tests-"
    47  )
    48  
    49  // TestPoolDevice runs integration tests for pool device.
    50  // The following scenario implemented:
    51  // - Create pool device with name 'test-pool-device'
    52  // - Create two thin volumes 'thin-1' and 'thin-2'
    53  // - Write ext4 file system on 'thin-1' and make sure it'errs moutable
    54  // - Write v1 test file on 'thin-1' volume
    55  // - Take 'thin-1' snapshot 'snap-1'
    56  // - Change v1 file to v2 on 'thin-1'
    57  // - Mount 'snap-1' and make sure test file is v1
    58  // - Unmount volumes and remove all devices
    59  func TestPoolDevice(t *testing.T) {
    60  	testutil.RequiresRoot(t)
    61  
    62  	logrus.SetLevel(logrus.DebugLevel)
    63  	ctx := context.Background()
    64  
    65  	tempDir, err := ioutil.TempDir("", "pool-device-test-")
    66  	assert.NilError(t, err, "couldn't get temp directory for testing")
    67  
    68  	_, loopDataDevice := createLoopbackDevice(t, tempDir)
    69  	_, loopMetaDevice := createLoopbackDevice(t, tempDir)
    70  
    71  	poolName := fmt.Sprintf("test-pool-device-%d", time.Now().Nanosecond())
    72  	err = dmsetup.CreatePool(poolName, loopDataDevice, loopMetaDevice, 64*1024/dmsetup.SectorSize)
    73  	assert.NilError(t, err, "failed to create pool %q", poolName)
    74  
    75  	defer func() {
    76  		// Detach loop devices and remove images
    77  		err := losetup.DetachLoopDevice(loopDataDevice, loopMetaDevice)
    78  		assert.NilError(t, err)
    79  
    80  		err = os.RemoveAll(tempDir)
    81  		assert.NilError(t, err, "couldn't cleanup temp directory")
    82  	}()
    83  
    84  	config := &Config{
    85  		PoolName:           poolName,
    86  		RootPath:           tempDir,
    87  		BaseImageSize:      "16mb",
    88  		BaseImageSizeBytes: 16 * 1024 * 1024,
    89  	}
    90  
    91  	pool, err := NewPoolDevice(ctx, config)
    92  	assert.NilError(t, err, "can't create device pool")
    93  	assert.Assert(t, pool != nil)
    94  
    95  	defer func() {
    96  		err := pool.RemovePool(ctx)
    97  		assert.NilError(t, err, "can't close device pool")
    98  	}()
    99  
   100  	// Create thin devices
   101  	t.Run("CreateThinDevice", func(t *testing.T) {
   102  		testCreateThinDevice(t, pool)
   103  	})
   104  
   105  	// Make ext4 filesystem on 'thin-1'
   106  	t.Run("MakeFileSystem", func(t *testing.T) {
   107  		testMakeFileSystem(t, pool)
   108  	})
   109  
   110  	// Mount 'thin-1' and write v1 test file on 'thin-1' device
   111  	err = mount.WithTempMount(ctx, getMounts(thinDevice1), func(thin1MountPath string) error {
   112  		// Write v1 test file on 'thin-1' device
   113  		thin1TestFilePath := filepath.Join(thin1MountPath, "TEST")
   114  		err := ioutil.WriteFile(thin1TestFilePath, []byte("test file (v1)"), 0700)
   115  		assert.NilError(t, err, "failed to write test file v1 on '%s' volume", thinDevice1)
   116  
   117  		return nil
   118  	})
   119  
   120  	// Take snapshot of 'thin-1'
   121  	t.Run("CreateSnapshotDevice", func(t *testing.T) {
   122  		testCreateSnapshot(t, pool)
   123  	})
   124  
   125  	// Update TEST file on 'thin-1' to v2
   126  	err = mount.WithTempMount(ctx, getMounts(thinDevice1), func(thin1MountPath string) error {
   127  		thin1TestFilePath := filepath.Join(thin1MountPath, "TEST")
   128  		err = ioutil.WriteFile(thin1TestFilePath, []byte("test file (v2)"), 0700)
   129  		assert.NilError(t, err, "failed to write test file v2 on 'thin-1' volume after taking snapshot")
   130  
   131  		return nil
   132  	})
   133  
   134  	assert.NilError(t, err)
   135  
   136  	// Mount 'snap-1' and make sure TEST file is v1
   137  	err = mount.WithTempMount(ctx, getMounts(snapDevice1), func(snap1MountPath string) error {
   138  		// Read test file from snapshot device and make sure it's v1
   139  		fileData, err := ioutil.ReadFile(filepath.Join(snap1MountPath, "TEST"))
   140  		assert.NilError(t, err, "couldn't read test file from '%s' device", snapDevice1)
   141  		assert.Equal(t, "test file (v1)", string(fileData), "test file content is invalid on snapshot")
   142  
   143  		return nil
   144  	})
   145  
   146  	assert.NilError(t, err)
   147  
   148  	t.Run("DeactivateDevice", func(t *testing.T) {
   149  		testDeactivateThinDevice(t, pool)
   150  	})
   151  
   152  	t.Run("RemoveDevice", func(t *testing.T) {
   153  		testRemoveThinDevice(t, pool)
   154  	})
   155  
   156  	t.Run("rollbackActivate", func(t *testing.T) {
   157  		testCreateThinDevice(t, pool)
   158  
   159  		ctx := context.Background()
   160  
   161  		snapDevice := "snap2"
   162  
   163  		err := pool.CreateSnapshotDevice(ctx, thinDevice1, snapDevice, device1Size)
   164  		assert.NilError(t, err)
   165  
   166  		info, err := pool.metadata.GetDevice(ctx, snapDevice)
   167  		assert.NilError(t, err)
   168  
   169  		// Simulate a case that the device cannot be activated.
   170  		err = pool.DeactivateDevice(ctx, info.Name, false, false)
   171  		assert.NilError(t, err)
   172  
   173  		err = pool.rollbackActivate(ctx, info, err)
   174  		assert.NilError(t, err)
   175  	})
   176  }
   177  
   178  func TestPoolDeviceMarkFaulty(t *testing.T) {
   179  	tempDir, store := createStore(t)
   180  	defer cleanupStore(t, tempDir, store)
   181  
   182  	err := store.AddDevice(testCtx, &DeviceInfo{Name: "1", State: Unknown})
   183  	assert.NilError(t, err)
   184  
   185  	// Note: do not use 'Activated' here because pool.ensureDeviceStates() will
   186  	// try to activate the real dm device, which will fail on a faked device.
   187  	err = store.AddDevice(testCtx, &DeviceInfo{Name: "2", State: Deactivated})
   188  	assert.NilError(t, err)
   189  
   190  	pool := &PoolDevice{metadata: store}
   191  	err = pool.ensureDeviceStates(testCtx)
   192  	assert.NilError(t, err)
   193  
   194  	called := 0
   195  	err = pool.metadata.WalkDevices(testCtx, func(info *DeviceInfo) error {
   196  		called++
   197  
   198  		switch called {
   199  		case 1:
   200  			assert.Equal(t, Faulty, info.State)
   201  			assert.Equal(t, "1", info.Name)
   202  		case 2:
   203  			assert.Equal(t, Deactivated, info.State)
   204  			assert.Equal(t, "2", info.Name)
   205  		default:
   206  			t.Error("unexpected walk call")
   207  		}
   208  
   209  		return nil
   210  	})
   211  	assert.NilError(t, err)
   212  	assert.Equal(t, 2, called)
   213  }
   214  
   215  func testCreateThinDevice(t *testing.T, pool *PoolDevice) {
   216  	ctx := context.Background()
   217  
   218  	err := pool.CreateThinDevice(ctx, thinDevice1, device1Size)
   219  	assert.NilError(t, err, "can't create first thin device")
   220  
   221  	err = pool.CreateThinDevice(ctx, thinDevice1, device1Size)
   222  	assert.Assert(t, err != nil, "device pool allows duplicated device names")
   223  
   224  	err = pool.CreateThinDevice(ctx, thinDevice2, device2Size)
   225  	assert.NilError(t, err, "can't create second thin device")
   226  
   227  	deviceInfo1, err := pool.metadata.GetDevice(ctx, thinDevice1)
   228  	assert.NilError(t, err)
   229  
   230  	deviceInfo2, err := pool.metadata.GetDevice(ctx, thinDevice2)
   231  	assert.NilError(t, err)
   232  
   233  	assert.Assert(t, deviceInfo1.DeviceID != deviceInfo2.DeviceID, "assigned device ids should be different")
   234  
   235  	usage, err := pool.GetUsage(thinDevice1)
   236  	assert.NilError(t, err)
   237  	assert.Equal(t, usage, int64(0))
   238  }
   239  
   240  func testMakeFileSystem(t *testing.T, pool *PoolDevice) {
   241  	devicePath := dmsetup.GetFullDevicePath(thinDevice1)
   242  	args := []string{
   243  		devicePath,
   244  		"-E",
   245  		"nodiscard,lazy_itable_init=0,lazy_journal_init=0",
   246  	}
   247  
   248  	output, err := exec.Command("mkfs.ext4", args...).CombinedOutput()
   249  	assert.NilError(t, err, "failed to make filesystem on '%s': %s", thinDevice1, string(output))
   250  
   251  	usage, err := pool.GetUsage(thinDevice1)
   252  	assert.NilError(t, err)
   253  	assert.Assert(t, usage > 0)
   254  }
   255  
   256  func testCreateSnapshot(t *testing.T, pool *PoolDevice) {
   257  	err := pool.CreateSnapshotDevice(context.Background(), thinDevice1, snapDevice1, device1Size)
   258  	assert.NilError(t, err, "failed to create snapshot from '%s' volume", thinDevice1)
   259  }
   260  
   261  func testDeactivateThinDevice(t *testing.T, pool *PoolDevice) {
   262  	deviceList := []string{
   263  		thinDevice2,
   264  		snapDevice1,
   265  	}
   266  
   267  	for _, deviceName := range deviceList {
   268  		assert.Assert(t, pool.IsActivated(deviceName))
   269  
   270  		err := pool.DeactivateDevice(context.Background(), deviceName, false, true)
   271  		assert.NilError(t, err, "failed to remove '%s'", deviceName)
   272  
   273  		assert.Assert(t, !pool.IsActivated(deviceName))
   274  	}
   275  }
   276  
   277  func testRemoveThinDevice(t *testing.T, pool *PoolDevice) {
   278  	err := pool.RemoveDevice(testCtx, thinDevice1)
   279  	assert.NilError(t, err, "should delete thin device from pool")
   280  
   281  	err = pool.RemoveDevice(testCtx, thinDevice2)
   282  	assert.NilError(t, err, "should delete thin device from pool")
   283  }
   284  
   285  func getMounts(thinDeviceName string) []mount.Mount {
   286  	return []mount.Mount{
   287  		{
   288  			Source: dmsetup.GetFullDevicePath(thinDeviceName),
   289  			Type:   "ext4",
   290  		},
   291  	}
   292  }
   293  
   294  func createLoopbackDevice(t *testing.T, dir string) (string, string) {
   295  	file, err := ioutil.TempFile(dir, testsPrefix)
   296  	assert.NilError(t, err)
   297  
   298  	size, err := units.RAMInBytes("128Mb")
   299  	assert.NilError(t, err)
   300  
   301  	err = file.Truncate(size)
   302  	assert.NilError(t, err)
   303  
   304  	err = file.Close()
   305  	assert.NilError(t, err)
   306  
   307  	imagePath := file.Name()
   308  
   309  	loopDevice, err := losetup.AttachLoopDevice(imagePath)
   310  	assert.NilError(t, err)
   311  
   312  	return imagePath, loopDevice
   313  }