Clean cmd + VM image pruning

This commit is contained in:
AlexSSD7 2023-08-30 09:43:41 +01:00
commit 8666a21d0b
4 changed files with 97 additions and 8 deletions

41
cmd/clean.go Normal file
View file

@ -0,0 +1,41 @@
package cmd
import (
"bufio"
"fmt"
"log/slog"
"os"
"strings"
"github.com/spf13/cobra"
)
var cleanCmd = &cobra.Command{
Use: "clean",
Short: "Remove all downloaded VM images.",
Run: func(cmd *cobra.Command, args []string) {
store := createStore()
fmt.Fprintf(os.Stderr, "Will delete all VM images in the data directory. Proceed? (y/n) > ")
reader := bufio.NewReader(os.Stdin)
answer, err := reader.ReadBytes('\n')
if err != nil {
slog.Error("Failed to read answer", "error", err.Error())
os.Exit(1)
}
if strings.ToLower(string(answer)) != "y\n" {
fmt.Fprintf(os.Stderr, "Aborted.\n")
os.Exit(2)
}
deleted, err := store.CleanImages(false)
if err != nil {
slog.Error("Failed to clean images", "error", err.Error())
os.Exit(1)
}
slog.Info("Successful VM image cleanup", "deleted", deleted)
},
}

View file

@ -41,6 +41,7 @@ func init() {
rootCmd.AddCommand(lsCmd)
rootCmd.AddCommand(runCmd)
rootCmd.AddCommand(shellCmd)
rootCmd.AddCommand(cleanCmd)
rootCmd.PersistentFlags().BoolVar(&vmDebugFlag, "vmdebug", false, "Enables the VM debug mode. This will open an accessible VM monitor. You can log in with root user and no password.")
rootCmd.PersistentFlags().BoolVar(&unrestrictedNetworkingFlag, "unrestricted-networking", false, "Enables unrestricted networking. This will allow the VM to connect to the internet.")

View file

@ -54,14 +54,19 @@ func doUSBRootCheck() {
}
}
func runVM(passthroughArg string, fn func(context.Context, *vm.VM, *vm.FileManager) int, forwardPortsRules []vm.PortForwardingRule, unrestrictedNetworking bool) int {
func createStore() *storage.Storage {
store, err := storage.NewStorage(slog.With("caller", "storage"), dataDirFlag)
if err != nil {
slog.Error("Failed to create Linsk data storage", "error", err.Error(), "data-dir", dataDirFlag)
os.Exit(1)
}
_, err = store.ValidateImageHashOrDownload()
return store
}
func runVM(passthroughArg string, fn func(context.Context, *vm.VM, *vm.FileManager) int, forwardPortsRules []vm.PortForwardingRule, unrestrictedNetworking bool) int {
store := createStore()
_, err := store.ValidateImageHashOrDownload()
if err != nil {
slog.Error("Failed to validate image hash or download image", "error", err.Error())
os.Exit(1)
@ -92,7 +97,6 @@ func runVM(passthroughArg string, fn func(context.Context, *vm.VM, *vm.FileManag
ShowDisplay: vmDebugFlag,
}
// TODO: Alpine image should be downloaded from somewhere.
vi, err := vm.NewVM(slog.Default().With("caller", "vm"), vmCfg)
if err != nil {
slog.Error("Failed to create vm instance", "error", err.Error())

View file

@ -11,6 +11,7 @@ import (
"net/http"
"os"
"path/filepath"
"strings"
"github.com/dustin/go-humanize"
"github.com/pkg/errors"
@ -41,6 +42,35 @@ func NewStorage(logger *slog.Logger, dataDir string) (*Storage, error) {
}, nil
}
func (s *Storage) CleanImages(retainLatest bool) (int, error) {
dirEntries, err := os.ReadDir(s.path)
if err != nil {
return 0, errors.Wrap(err, "read dir")
}
localImagePath := s.GetLocalImagePath()
var deleted int
for _, entry := range dirEntries {
fullPath := filepath.Join(s.path, entry.Name())
if !strings.HasSuffix(fullPath, ".qcow2") {
continue
}
if !(retainLatest && fullPath == localImagePath) {
s.logger.Warn("Removing VM image", "path", fullPath)
err = os.Remove(fullPath)
if err != nil {
return deleted, errors.Wrapf(err, "remove vm image (path '%v')", fullPath)
}
deleted++
}
}
return deleted, nil
}
func (s *Storage) GetLocalImagePath() string {
return filepath.Join(s.path, hex.EncodeToString(imageHash[:])+".qcow2")
}
@ -143,16 +173,29 @@ func (s *Storage) ValidateImageHash() error {
}
func (s *Storage) ValidateImageHashOrDownload() (bool, error) {
var downloaded bool
err := s.ValidateImageHash()
if err == nil {
return false, nil
if err != nil {
if errors.Is(err, os.ErrNotExist) {
err = s.DownloadImage()
if err != nil {
return false, errors.Wrap(err, "download iamge")
}
downloaded = true
} else {
return false, errors.Wrap(err, "validate image hash")
}
}
if errors.Is(err, os.ErrNotExist) {
return true, errors.Wrap(s.DownloadImage(), "download image")
deletedImagesCount, err := s.CleanImages(true)
if err != nil {
s.logger.Warn("Failed to prune old VM images", "error", err, "deleted", deletedImagesCount)
} else {
s.logger.Info("Pruned old VM images", "deleted", deletedImagesCount)
}
return false, err
return downloaded, err
}
func copyWithProgress(dst io.Writer, src io.Reader, blockSize int, length int64, report func(int, float64)) (int, error) {