linsk/cmd/run.go

142 lines
4.7 KiB
Go
Raw Normal View History

2023-08-25 16:54:58 +01:00
package cmd
import (
"context"
"fmt"
"log/slog"
2023-08-26 11:27:38 +01:00
"os"
"runtime"
"strings"
2023-08-25 16:54:58 +01:00
"github.com/AlexSSD7/linsk/share"
2023-08-26 09:16:52 +01:00
"github.com/AlexSSD7/linsk/vm"
2023-08-26 16:26:35 +01:00
"github.com/sethvargo/go-password/password"
2023-08-25 16:54:58 +01:00
"github.com/spf13/cobra"
)
var runCmd = &cobra.Command{
2023-08-29 10:37:52 +01:00
Use: "run",
2023-08-30 15:24:25 +01:00
Short: "Start a VM and expose an FTP file share.",
2023-08-29 10:37:52 +01:00
Args: cobra.ExactArgs(3),
2023-08-27 15:30:51 +01:00
Run: func(cmd *cobra.Command, args []string) {
2023-08-25 16:54:58 +01:00
vmMountDevName := args[1]
fsType := args[2]
newBackendFunc := share.GetBackend(shareBackendFlag)
if newBackendFunc == nil {
slog.Error("Unknown file share backend", "type", shareBackendFlag)
2023-08-26 16:26:35 +01:00
os.Exit(1)
}
cfg, err := share.RawUserConfiguration{
ListenIP: shareListenIPFlag,
2023-08-30 15:24:25 +01:00
FTPExtIP: ftpExtIPFlag,
SMBExtMode: smbUseExternAddrFlag,
}.Process(shareBackendFlag, slog.With("caller", "share-config"))
if err != nil {
slog.Error("Failed to process raw configuration", "error", err.Error())
2023-08-30 15:24:25 +01:00
os.Exit(1)
}
backend, vmOpts, err := newBackendFunc(cfg)
if err != nil {
slog.Error("Failed to initialize share backend", "backend", shareBackendFlag, "error", err.Error())
os.Exit(1)
2023-08-29 10:00:12 +01:00
}
2023-09-01 12:40:13 +01:00
if luksFlag && !allowLUKSLowMemoryFlag {
if vmMemAllocFlag < 2048 {
if vmMemAllocFlag != defaultMemAlloc {
slog.Warn("Enforcing minimum LUKS memory allocation. Please add --allow-luks-low-memory to disable this.", "min", vmMemAllocFlag, "specified", vmMemAllocFlag)
}
vmMemAllocFlag = defaultMemAllocLUKS
}
}
os.Exit(runVM(args[0], func(ctx context.Context, i *vm.VM, fm *vm.FileManager, tapCtx *share.NetTapRuntimeContext) int {
2023-08-29 10:59:50 +01:00
slog.Info("Mounting the device", "dev", vmMountDevName, "fs", fsType, "luks", luksFlag)
2023-08-25 19:55:11 +01:00
err := fm.Mount(vmMountDevName, vm.MountOptions{
FSType: fsType,
LUKS: luksFlag,
})
2023-08-25 16:54:58 +01:00
if err != nil {
2023-08-29 10:59:50 +01:00
slog.Error("Failed to mount the disk inside the VM", "error", err.Error())
2023-08-26 11:27:38 +01:00
return 1
2023-08-25 16:54:58 +01:00
}
2023-08-26 16:26:35 +01:00
sharePWD, err := password.Generate(16, 10, 0, false, false)
if err != nil {
2023-08-29 13:29:46 +01:00
slog.Error("Failed to generate ephemeral password for the network file share", "error", err.Error())
2023-08-26 16:26:35 +01:00
return 1
}
2023-09-01 15:15:40 +01:00
lg := slog.With("backend", shareBackendFlag)
shareURI, err := backend.Apply(ctx, sharePWD, &share.VMShareContext{
Instance: i,
FileManager: fm,
NetTapCtx: tapCtx,
})
2023-08-26 16:26:35 +01:00
if err != nil {
2023-09-01 15:15:40 +01:00
lg.Error("Failed to apply (start) file share backend", "error", err.Error())
2023-08-26 16:26:35 +01:00
return 1
}
2023-09-01 15:15:40 +01:00
lg.Info("Started the network share successfully")
2023-08-26 16:26:35 +01:00
2023-08-31 19:46:13 +01:00
fmt.Fprintf(os.Stderr, "===========================\n[Network File Share Config]\nThe network file share was started. Please use the credentials below to connect to the file server.\n\nType: "+strings.ToUpper(shareBackendFlag)+"\nURL: %v\nUsername: linsk\nPassword: %v\n===========================\n", shareURI, sharePWD)
2023-08-29 13:29:46 +01:00
2023-09-01 14:40:17 +01:00
ctxWait := true
if debugShellFlag {
slog.Warn("Starting a debug VM shell")
err := runVMShell(ctx, i)
if err != nil {
slog.Error("Failed to run VM shell", "error", err.Error())
} else {
ctxWait = false
}
}
if ctxWait {
<-ctx.Done()
}
2023-08-26 11:27:38 +01:00
return 0
}, vmOpts.Ports, unrestrictedNetworkingFlag, vmOpts.EnableTap))
2023-08-25 16:54:58 +01:00
},
}
2023-08-25 19:55:11 +01:00
2023-09-01 12:40:13 +01:00
var (
luksFlag bool
allowLUKSLowMemoryFlag bool
shareListenIPFlag string
ftpExtIPFlag string
shareBackendFlag string
smbUseExternAddrFlag bool
2023-09-01 14:40:17 +01:00
debugShellFlag bool
2023-09-01 12:40:13 +01:00
)
2023-08-25 19:55:11 +01:00
func init() {
2023-08-30 15:24:25 +01:00
runCmd.Flags().BoolVarP(&luksFlag, "luks", "l", false, "Use cryptsetup to open a LUKS volume (password will be prompted).")
2023-09-01 12:40:13 +01:00
runCmd.Flags().BoolVar(&allowLUKSLowMemoryFlag, "allow-luks-low-memory", false, "Allow VM memory allocation lower than 2048 MiB when LUKS is enabled.")
2023-09-01 14:40:17 +01:00
runCmd.Flags().BoolVar(&debugShellFlag, "debug-shell", false, "Start a VM shell when the network file share is active.")
var defaultShareType string
switch runtime.GOOS {
case "windows":
defaultShareType = "smb"
2023-09-01 15:25:35 +01:00
case "darwin":
defaultShareType = "afp"
default:
defaultShareType = "ftp"
}
runCmd.Flags().StringVar(&shareBackendFlag, "share-backend", defaultShareType, "Specifies the file share backend to use. The default value is OS-specific.")
runCmd.Flags().StringVar(&shareListenIPFlag, "share-listen", share.GetDefaultListenIPStr(), "Specifies the IP to bind the network share port to. NOTE: For FTP, changing the bind address is not enough to connect remotely. You should also specify --ftp-extip.")
runCmd.Flags().StringVar(&ftpExtIPFlag, "ftp-extip", share.GetDefaultListenIPStr(), "Specifies the external IP the FTP server should advertise.")
2023-08-31 20:17:55 +01:00
runCmd.Flags().BoolVar(&smbUseExternAddrFlag, "smb-extern", share.IsSMBExtModeDefault(), "Specifies whether Linsk emulate external networking for the VM's SMB server. This is the default for Windows as there is no way to specify ports in Windows SMB client.")
2023-08-25 19:55:11 +01:00
}