From 43a32225cdcaf8c996281cd6da9875c89b2ebdf2 Mon Sep 17 00:00:00 2001 From: Debarshi Ray Date: Fri, 25 Jun 2021 20:34:14 +0200 Subject: [PATCH] cmd/run: Optimize 'enter' and 'run' in the non-fallback case Currently, the 'enter' command involves two extra invocations of 'podman exec' to detect if the user's chosen shell and current working directory are present inside the toolbox container. Each invocation is sufficiently expensive to add a noticeable overhead to the 'enter' and 'run' commands. Moreover, file system operations being inherently racy, it's always better to detect errors and handle them instead of trying to pre-emptively avoid them. Therefore, this shuffles the code around to attempt the non-fallback invocation, and then handle the errors by attempting a series of fallbacks for the command and the current working directory. https://github.com/containers/toolbox/pull/813 --- src/cmd/run.go | 128 +++++++++++++++++++++++++++++-------------------- 1 file changed, 75 insertions(+), 53 deletions(-) diff --git a/src/cmd/run.go b/src/cmd/run.go index cde80b381..12c9616b5 100644 --- a/src/cmd/run.go +++ b/src/cmd/run.go @@ -36,6 +36,9 @@ var ( distro string release string } + + runFallbackCommands = [][]string{{"/bin/bash", "-l"}} + runFallbackWorkDirs = []string{"" /* $HOME */} ) var runCmd = &cobra.Command{ @@ -265,29 +268,14 @@ func runCommand(container string, logrus.Debugf("Container %s is initialized", container) - if _, err := isCommandPresent(container, command[0]); err != nil { - if fallbackToBash { - fmt.Fprintf(os.Stderr, - "Error: command %s not found in container %s\n", - command[0], - container) - fmt.Fprintf(os.Stderr, "Using /bin/bash instead.\n") - - command = []string{"/bin/bash", "-l"} - } else { - return fmt.Errorf("command %s not found in container %s", command[0], container) - } + if err := runCommandWithFallbacks(container, command, emitEscapeSequence, fallbackToBash); err != nil { + return err } - if pathPresent, _ := isPathPresent(container, workingDirectory); !pathPresent { - fmt.Fprintf(os.Stderr, "Error: path %s not found in container %s\n", - workingDirectory, container) - fmt.Fprintf(os.Stderr, "Using %s instead.\n", - currentUser.HomeDir) - - workingDirectory = currentUser.HomeDir - } + return nil +} +func runCommandWithFallbacks(container string, command []string, emitEscapeSequence, fallbackToBash bool) error { logrus.Debug("Checking if 'podman exec' supports disabling the detach keys") var detachKeysSupported bool @@ -299,48 +287,82 @@ func runCommand(container string, envOptions := utils.GetEnvOptionsForPreservedVariables() - execArgs := constructExecArgs(container, command, detachKeysSupported, envOptions, workingDirectory) + runFallbackCommandsIndex := 0 + runFallbackWorkDirsIndex := 0 + workDir := workingDirectory - if emitEscapeSequence { - fmt.Printf("\033]777;container;push;%s;toolbox;%s\033\\", container, currentUser.Uid) - } + for { + execArgs := constructExecArgs(container, command, detachKeysSupported, envOptions, workDir) - logrus.Debugf("Running in container %s:", container) - logrus.Debug("podman") - for _, arg := range execArgs { - logrus.Debugf("%s", arg) - } + if emitEscapeSequence { + fmt.Printf("\033]777;container;push;%s;toolbox;%s\033\\", container, currentUser.Uid) + } - exitCode, err := shell.RunWithExitCode("podman", os.Stdin, os.Stdout, nil, execArgs...) + logrus.Debugf("Running in container %s:", container) + logrus.Debug("podman") + for _, arg := range execArgs { + logrus.Debugf("%s", arg) + } - if emitEscapeSequence { - fmt.Printf("\033]777;container;pop;;;%s\033\\", currentUser.Uid) - } + exitCode, err := shell.RunWithExitCode("podman", os.Stdin, os.Stdout, nil, execArgs...) - switch exitCode { - case 0: - if err != nil { - panic("unexpected error: 'podman exec' finished successfully") - } - case 125: - err = fmt.Errorf("failed to invoke 'podman exec' in container %s", container) - case 126: - err = fmt.Errorf("failed to invoke command %s in container %s", command[0], container) - case 127: - if pathPresent, _ := isPathPresent(container, workingDirectory); !pathPresent { - err = fmt.Errorf("directory %s not found in container %s", workingDirectory, container) - } else { - err = fmt.Errorf("command %s not found in container %s", command[0], container) + if emitEscapeSequence { + fmt.Printf("\033]777;container;pop;;;%s\033\\", currentUser.Uid) } - default: - err = nil - } - if err != nil { - return err + switch exitCode { + case 0: + if err != nil { + panic("unexpected error: 'podman exec' finished successfully") + } + return nil + case 125: + err = fmt.Errorf("failed to invoke 'podman exec' in container %s", container) + return err + case 126: + err = fmt.Errorf("failed to invoke command %s in container %s", command[0], container) + return err + case 127: + if pathPresent, _ := isPathPresent(container, workDir); !pathPresent { + if runFallbackWorkDirsIndex < len(runFallbackWorkDirs) { + fmt.Fprintf(os.Stderr, + "Error: directory %s not found in container %s\n", + workDir, + container) + + workDir = runFallbackWorkDirs[runFallbackWorkDirsIndex] + if workDir == "" { + workDir = currentUser.HomeDir + } + + fmt.Fprintf(os.Stderr, "Using %s instead.\n", workDir) + runFallbackWorkDirsIndex++ + } else { + err = fmt.Errorf("directory %s not found in container %s", workDir, container) + return err + } + } else { + if fallbackToBash && runFallbackCommandsIndex < len(runFallbackCommands) { + fmt.Fprintf(os.Stderr, + "Error: command %s not found in container %s\n", + command[0], + container) + + command = runFallbackCommands[runFallbackCommandsIndex] + fmt.Fprintf(os.Stderr, "Using %s instead.\n", command) + + runFallbackCommandsIndex++ + } else { + err = fmt.Errorf("command %s not found in container %s", command[0], container) + return err + } + } + default: + return nil + } } - return nil + panic("code should not be reached") } func runHelp(cmd *cobra.Command, args []string) {