From ef22d1edf486a7c844ffa1df53da1322ccdf2264 Mon Sep 17 00:00:00 2001 From: Sankalp Acharya Date: Thu, 13 Apr 2023 20:10:59 +0530 Subject: [PATCH 1/2] Add mounts option to checkpointctl show This patch adds `--mounts` option for the checkpointctl show command. The extended output for the same would look like: checkpointctl show checkpoint.tar.gz --mounts Overview of Mounts +--------------------+--------+---------------------------+ | DESTINATION | TYPE | SOURCE | +--------------------+--------+---------------------------+ | /proc | proc | proc | | /dev | tmpfs | tmpfs | | /sys | sysfs | sysfs | | /dev/pts | devpts | devpts | | /dev/mqueue | mqueue | mqueue | | /etc/hostname | bind | ../userdata/hostname | | /run/.containerenv | bind | ../userdata/.containerenv | | /etc/resolv.conf | bind | ../userdata/resolv.conf | | /etc/hosts | bind | ../userdata/hosts | | /dev/shm | bind | ../userdata/shm | | /sys/fs/cgroup | cgroup | cgroup | +--------------------+--------+---------------------------+ Closes #40 Signed-off-by: Sankalp Acharya --- checkpointctl.go | 7 ++++ container.go | 88 +++++++++++++++++++++++++++++++----------------- 2 files changed, 64 insertions(+), 31 deletions(-) diff --git a/checkpointctl.go b/checkpointctl.go index f92d6cb7..a15e0206 100644 --- a/checkpointctl.go +++ b/checkpointctl.go @@ -14,6 +14,7 @@ var ( name string version string printStats bool + showMounts bool ) func main() { @@ -48,6 +49,12 @@ func setupShow() *cobra.Command { false, "Print checkpointing statistics if available", ) + flags.BoolVar( + &showMounts, + "mounts", + false, + "Print overview about mounts used in the checkpoints", + ) return cmd } diff --git a/container.go b/container.go index a11be114..292bb3dc 100644 --- a/container.go +++ b/container.go @@ -9,6 +9,7 @@ import ( "fmt" "os" "path/filepath" + "strings" "time" metadata "github.com/checkpoint-restore/checkpointctl/lib" @@ -147,41 +148,58 @@ func showContainerCheckpoint(checkpointDirectory string) error { table.Append(row) table.Render() - if !printStats { - return nil + if showMounts { + table = tablewriter.NewWriter(os.Stdout) + table.SetHeader([]string{ + "Destination", + "Type", + "Source", + }) + // Get overview of mounts from spec.dump + for _, data := range specDump.Mounts { + table.Append([]string{ + data.Destination, + data.Type, + shortenPath(data.Source), + }) + } + fmt.Println("\nOverview of Mounts") + table.Render() } - cpDir, err := os.Open(checkpointDirectory) - if err != nil { - return err - } - defer cpDir.Close() + if printStats { + cpDir, err := os.Open(checkpointDirectory) + if err != nil { + return err + } + defer cpDir.Close() - // Get dump statistics with crit - dumpStatistics, err := crit.GetDumpStats(cpDir.Name()) - if err != nil { - return fmt.Errorf("unable to display checkpointing statistics: %w", err) - } + // Get dump statistics with crit + dumpStatistics, err := crit.GetDumpStats(cpDir.Name()) + if err != nil { + return fmt.Errorf("unable to display checkpointing statistics: %w", err) + } - table = tablewriter.NewWriter(os.Stdout) - table.SetHeader([]string{ - "Freezing Time", - "Frozen Time", - "Memdump Time", - "Memwrite Time", - "Pages Scanned", - "Pages Written", - }) - table.Append([]string{ - fmt.Sprintf("%d us", dumpStatistics.GetFreezingTime()), - fmt.Sprintf("%d us", dumpStatistics.GetFrozenTime()), - fmt.Sprintf("%d us", dumpStatistics.GetMemdumpTime()), - fmt.Sprintf("%d us", dumpStatistics.GetMemwriteTime()), - fmt.Sprintf("%d", dumpStatistics.GetPagesScanned()), - fmt.Sprintf("%d", dumpStatistics.GetPagesWritten()), - }) - fmt.Println("CRIU dump statistics") - table.Render() + table = tablewriter.NewWriter(os.Stdout) + table.SetHeader([]string{ + "Freezing Time", + "Frozen Time", + "Memdump Time", + "Memwrite Time", + "Pages Scanned", + "Pages Written", + }) + table.Append([]string{ + fmt.Sprintf("%d us", dumpStatistics.GetFreezingTime()), + fmt.Sprintf("%d us", dumpStatistics.GetFrozenTime()), + fmt.Sprintf("%d us", dumpStatistics.GetMemdumpTime()), + fmt.Sprintf("%d us", dumpStatistics.GetMemwriteTime()), + fmt.Sprintf("%d", dumpStatistics.GetPagesScanned()), + fmt.Sprintf("%d", dumpStatistics.GetPagesWritten()), + }) + fmt.Println("\nCRIU dump statistics") + table.Render() + } return nil } @@ -206,3 +224,11 @@ func getCheckpointSize(path string) (size int64, err error) { return dirSize(dir) } + +func shortenPath(path string) string { + parts := strings.Split(path, string(filepath.Separator)) + if len(parts) <= 2 { + return path + } + return filepath.Join("..", filepath.Join(parts[len(parts)-2:]...)) +} From d7c5e4b765c0db82e6c4c5e4306d1a14fd919d7e Mon Sep 17 00:00:00 2001 From: Sankalp Acharya Date: Sat, 15 Apr 2023 06:22:01 +0530 Subject: [PATCH 2/2] Add test for `--mounts` This patch adds test for the `--mounts` option for checkpointctl. Also adds '--full-paths' option to display mounts with full paths. Signed-off-by: Sankalp Acharya --- checkpointctl.go | 12 +++++++++++- container.go | 7 ++++++- test/checkpointctl.bats | 35 +++++++++++++++++++++++++++++++++++ test/spec.dump | 16 +++++++++++++++- 4 files changed, 67 insertions(+), 3 deletions(-) diff --git a/checkpointctl.go b/checkpointctl.go index a15e0206..0b062367 100644 --- a/checkpointctl.go +++ b/checkpointctl.go @@ -15,6 +15,7 @@ var ( version string printStats bool showMounts bool + fullPaths bool ) func main() { @@ -55,11 +56,21 @@ func setupShow() *cobra.Command { false, "Print overview about mounts used in the checkpoints", ) + flags.BoolVar( + &fullPaths, + "full-paths", + false, + "Display mounts with full paths", + ) return cmd } func show(cmd *cobra.Command, args []string) error { + if fullPaths && !showMounts { + return fmt.Errorf("Cannot use --full-paths without --mounts option") + } + input := args[0] tar, err := os.Stat(input) if err != nil { @@ -81,6 +92,5 @@ func show(cmd *cobra.Command, args []string) error { if err := archive.UntarPath(input, dir); err != nil { return fmt.Errorf("unpacking of checkpoint archive %s failed: %w", input, err) } - return showContainerCheckpoint(dir) } diff --git a/container.go b/container.go index 292bb3dc..0143c285 100644 --- a/container.go +++ b/container.go @@ -160,7 +160,12 @@ func showContainerCheckpoint(checkpointDirectory string) error { table.Append([]string{ data.Destination, data.Type, - shortenPath(data.Source), + func() string { + if fullPaths { + return data.Source + } + return shortenPath(data.Source) + }(), }) } fmt.Println("\nOverview of Mounts") diff --git a/test/checkpointctl.bats b/test/checkpointctl.bats index 63baa17c..5fcaedbe 100644 --- a/test/checkpointctl.bats +++ b/test/checkpointctl.bats @@ -136,6 +136,41 @@ function teardown() { [[ ${lines[10]} == *"446571 us"* ]] } +@test "Run checkpointctl show with tar file and --mounts and valid spec.dump" { + cp test/config.dump "$TEST_TMP_DIR1" + cp test/spec.dump "$TEST_TMP_DIR1" + mkdir "$TEST_TMP_DIR1"/checkpoint + ( cd "$TEST_TMP_DIR1" && tar cf "$TEST_TMP_DIR2"/test.tar . ) + checkpointctl show "$TEST_TMP_DIR2"/test.tar --mounts + [ "$status" -eq 0 ] + [[ ${lines[6]} == *"Overview of Mounts"* ]] + [[ ${lines[8]} == *"DESTINATION"* ]] + [[ ${lines[10]} == *"/proc"* ]] +} + +@test "Run checkpointctl show with tar file and --mounts and --full-paths and valid spec.dump" { + cp test/config.dump "$TEST_TMP_DIR1" + cp test/spec.dump "$TEST_TMP_DIR1" + mkdir "$TEST_TMP_DIR1"/checkpoint + ( cd "$TEST_TMP_DIR1" && tar cf "$TEST_TMP_DIR2"/test.tar . ) + checkpointctl show "$TEST_TMP_DIR2"/test.tar --mounts --full-paths + [ "$status" -eq 0 ] + [[ ${lines[6]} == *"Overview of Mounts"* ]] + [[ ${lines[8]} == *"DESTINATION"* ]] + [[ ${lines[10]} == *"/proc"* ]] +} + + +@test "Run checkpointctl show with tar file and missing --mounts and --full-paths" { + cp test/config.dump "$TEST_TMP_DIR1" + cp test/spec.dump "$TEST_TMP_DIR1" + mkdir "$TEST_TMP_DIR1"/checkpoint + ( cd "$TEST_TMP_DIR1" && tar cf "$TEST_TMP_DIR2"/test.tar . ) + checkpointctl show "$TEST_TMP_DIR2"/test.tar --full-paths + [ "$status" -eq 1 ] + [[ ${lines[0]} == *"Error: Cannot use --full-paths without --mounts option"* ]] +} + @test "Run checkpointctl show with tar file with valid config.dump and valid spec.dump (CRI-O) and no checkpoint directory" { cp test/config.dump "$TEST_TMP_DIR1" cp test/spec.dump.cri-o "$TEST_TMP_DIR1"/spec.dump diff --git a/test/spec.dump b/test/spec.dump index 237795b8..d7c94dfc 100644 --- a/test/spec.dump +++ b/test/spec.dump @@ -1,3 +1,17 @@ { -"annotations": {"io.container.manager": "libpod"} + "mounts": [ + { + "destination": "/proc", + "type": "proc", + "source": "proc" + }, + { + "destination": "/etc/hostname", + "type": "bind", + "source": "/run/containers/storage/overlay-containers/d5eee7931a29b2d6bf51469e3ab7284bb22a9e6dad073277e30e2a29256efc84/userdata/hostname" + } + ], + "annotations": { + "io.container.manager": "libpod" + } }