Skip to content

Commit

Permalink
fix: add option to use builtin SSH client, not enabled by default
Browse files Browse the repository at this point in the history
On some windows machines the builtin SSH library doesn't play well with DevPod. Originally we've tried to fix this by using our own SSH client. A couple of users reported this change broke their setup. We're rolling back the default on windows and instead offer it as a workaround via the `USE_BUILTIN_SSH` option. Updated readme for troubleshooting
  • Loading branch information
pascalbreuninger committed Jul 15, 2024
1 parent bd240fb commit 43fa3a9
Show file tree
Hide file tree
Showing 6 changed files with 72 additions and 23 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,4 @@ release
env
devpod-provider-ssh
e2e/bin
.idea
34 changes: 32 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,44 @@ This repository hosts the default SSH provider configuration used in DevPod.
To add this SSH provider from the CLI, use the `provider add` command. For example:

```shell
# be sure to set $CURRENT_VERSION to an appropriate release tag from this repo
devpod provider add https://github.com/loft-sh/devpod-provider-ssh/releases/download/$CURRENT_VERSION/provider.yaml
devpod provider add ssh
```

## Compatibility

We only support Linux machine as remote hosts.

### Windows

There are known issues with the default windows SSH installation in some setups. If you're unable to connect to your host by default,
try to enable the `USE_BUILTIN_SSH` option
```shell
devpod provider add ssh --option USE_BUILTIN_SSH=true
# or if already installed
devpod provider set-options ssh --option USE_BUILTIN_SSH=true
```

This forces the provider to use the builtin SSH client over the one accessible in your shell.
You will need to add the identities file manually to your SSH config in case it's not the default key:
```ssh
Host my-domain.com
User my-user
IdentityFile ~/.my-dir/my-key
```

## Options

This provider has the following options:

| NAME | REQUIRED | DESCRIPTION | DEFAULT |
|-----------------|----------|------------------------------------------------------------|-------------------|
| HOST | true | The SSH Host to connect to. Example: [email protected] | |
| AGENT_PATH | false | The path where to inject the DevPod agent to. | /tmp/devpod/agent |
| DOCKER_PATH | false | The path of the docker binary. | docker |
| EXTRA_FLAGS | false | Extra flags to pass to the SSH command. | |
| PORT | false | The SSH port to use. | 22 |
| USE_BUILTIN_SSH | false | Use the builtin SSH package. | false |

# Extra

For more detail, see the [DevPod Documentation](https://devpod.sh/docs/managing-providers/what-are-providers).
18 changes: 12 additions & 6 deletions e2e/tests/provider/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,39 +13,42 @@ import (
var _ = ginkgo.Describe("[e2e]: devpod provider ssh test suite", ginkgo.Ordered, func() {
ginkgo.Context("testing /kubeletinfo endpoint", ginkgo.Label("e2e"), ginkgo.Ordered, func() {
ginkgo.It("should fail the init", func() {
cmd := exec.Command("../release/devpod-provider-ssh-linux-amd64", "init")
cmd := exec.Command("../release/devpod-provider-ssh-darwin-arm64", "init")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"COMMAND=ls",
"DOCKER_PATH=docker",
"HOST=localhost",
"PORT=1234",
"USE_BUILTIN_SSH=false",
}...)
err := cmd.Run()
framework.ExpectError(err)
})

ginkgo.It("should run the init", func() {
cmd := exec.Command("../release/devpod-provider-ssh-linux-amd64", "init")
cmd := exec.Command("../release/devpod-provider-ssh-darwin-arm64", "init")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"COMMAND=ls",
"DOCKER_PATH=docker",
"HOST=localhost",
"PORT=22",
"USE_BUILTIN_SSH=false",
}...)
err := cmd.Run()
framework.ExpectNoError(err)
})

ginkgo.It("should run a command", func() {
cmd := exec.Command("../release/devpod-provider-ssh-linux-amd64", "command")
cmd := exec.Command("../release/devpod-provider-ssh-darwin-arm64", "command")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"COMMAND=ls",
"DOCKER_PATH=docker",
"HOST=localhost",
"PORT=22",
"USE_BUILTIN_SSH=false",
}...)
err := cmd.Run()
framework.ExpectNoError(err)
Expand All @@ -56,13 +59,14 @@ var _ = ginkgo.Describe("[e2e]: devpod provider ssh test suite", ginkgo.Ordered,
controlOutput, err := cmd.Output()
framework.ExpectNoError(err)

cmd = exec.Command("../release/devpod-provider-ssh-linux-amd64", "command")
cmd = exec.Command("../release/devpod-provider-ssh-darwin-arm64", "command")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"COMMAND=ls /",
"DOCKER_PATH=docker",
"HOST=localhost",
"PORT=22",
"USE_BUILTIN_SSH=false",
}...)
output, err := cmd.Output()
framework.ExpectNoError(err)
Expand All @@ -81,7 +85,7 @@ line3`)
echo line2
echo line3`)

cmd = exec.Command("../release/devpod-provider-ssh-linux-amd64", "command")
cmd = exec.Command("../release/devpod-provider-ssh-darwin-arm64", "command")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"DOCKER_PATH=docker",
Expand All @@ -90,6 +94,7 @@ echo line2
echo line3`,
"HOST=localhost",
"PORT=22",
"USE_BUILTIN_SSH=false",
}...)
output, err := cmd.CombinedOutput()
framework.ExpectNoError(err)
Expand All @@ -100,13 +105,14 @@ echo line3`,
ginkgo.It("should run a failing command and fail", func() {
controlOutput := []byte("bash: line 1: not-a-command: command not found")

cmd := exec.Command("../release/devpod-provider-ssh-linux-amd64", "command")
cmd := exec.Command("../release/devpod-provider-ssh-darwin-arm64", "command")
cmd.Env = append(cmd.Environ(), []string{
"AGENT_PATH=/tmp/devpod/agent",
"COMMAND=not-a-command",
"DOCKER_PATH=docker",
"HOST=localhost",
"PORT=22",
"USE_BUILTIN_SSH=false",
}...)
output, err := cmd.CombinedOutput()
framework.ExpectError(err)
Expand Down
5 changes: 5 additions & 0 deletions hack/provider/provider.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ optionGroups:
- options:
- PORT
- EXTRA_FLAGS
- USE_BUILTIN_SSH
name: "SSH options"
defaultVisible: false
- options:
Expand Down Expand Up @@ -41,6 +42,10 @@ options:
default: "22"
EXTRA_FLAGS:
description: "Extra flags to pass to the SSH command."
USE_BUILTIN_SSH:
description: "Use the builtin SSH package."
default: false
type: boolean
agent:
inactivityTimeout: ${INACTIVITY_TIMEOUT}
injectGitCredentials: ${INJECT_GIT_CREDENTIALS}
Expand Down
30 changes: 19 additions & 11 deletions pkg/options/options.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,20 +6,22 @@ import (
)

var (
DOCKER_PATH = "DOCKER_PATH"
AGENT_PATH = "AGENT_PATH"
HOST = "HOST"
PORT = "PORT"
EXTRA_FLAGS = "EXTRA_FLAGS"
DOCKER_PATH = "DOCKER_PATH"
AGENT_PATH = "AGENT_PATH"
HOST = "HOST"
PORT = "PORT"
EXTRA_FLAGS = "EXTRA_FLAGS"
USE_BUILTIN_SSH = "USE_BUILTIN_SSH"
)

type Options struct {
DockerPath string
AgentPath string
User string
Host string
Port string
ExtraFlags string
DockerPath string
AgentPath string
User string
Host string
Port string
ExtraFlags string
UseBuiltinSSH bool
}

func FromEnv() (*Options, error) {
Expand Down Expand Up @@ -49,6 +51,12 @@ func FromEnv() (*Options, error) {
return nil, err
}

builtinSSH, err := fromEnvOrError(USE_BUILTIN_SSH)
if err != nil {
return nil, err
}
retOptions.UseBuiltinSSH = builtinSSH == "true"

return retOptions, nil
}

Expand Down
7 changes: 3 additions & 4 deletions pkg/ssh/ssh.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import (
"os/exec"
"path"
"path/filepath"
"runtime"
"strings"
"time"

Expand Down Expand Up @@ -73,7 +72,7 @@ func getSSHCommand(provider *SSHProvider) ([]string, error) {
}

func execSSHCommand(provider *SSHProvider, command string, output io.Writer) error {
if runtime.GOOS == "windows" {
if provider.Config.UseBuiltinSSH {
// get ssh config for host
ctx, cancel := context.WithTimeout(context.Background(), time.Second*5)
defer cancel()
Expand Down Expand Up @@ -178,8 +177,8 @@ func copyCommandToRemote(provider *SSHProvider, command string) (string, error)
return "", err
}
defer func() {
script.Close()
os.Remove(script.Name())
_ = script.Close()
_ = os.Remove(script.Name())
}()

commandToRun, err := getSCPCommand(provider, script.Name())
Expand Down

0 comments on commit 43fa3a9

Please sign in to comment.