mirror of
https://github.com/coder/coder.git
synced 2025-07-12 00:14:10 +00:00
This brings together a bunch of random, partially implemented packages for support of the new(ish) Windows [`conpty`](https://devblogs.microsoft.com/commandline/windows-command-line-introducing-the-windows-pseudo-console-conpty/) API - such that we can leverage the `expect` style of CLI tests, but in a way that works in Linux/OSX `pty`s and Windows `conpty`. These include: - Vendoring the `go-expect` library from Netflix w/ some tweaks to work cross-platform - Vendoring the `pty` cross-platform implementation from [waypoint-plugin-sdk](b55c787a65/internal/pkg/pty
) - Vendoring the `conpty` Windows-specific implementation from [waypoint-plugin-sdk](b55c787a65/internal/pkg/conpty
) - Adjusting the `pty` interface to work with `go-expect` + the cross-plat version There were several limitations with the current packages: - `go-expect` requires the same `os.File` (TTY) for input / output, but `conhost` requires separate file handles - `conpty` does not handle input, only output - The cross-platform `pty` didn't expose the full set of primitives needed for `console` Therefore, the following changes were made: - Handling of `stdin` was added to the `conpty` interface - We weren't using the full extent of the `go-expect` interface, so some portions were removed (ie, exec'ing a process) to simplify our implementation and make it easier to extend cross-platform - Instead of `console` exposing just a `Tty`, it exposes an `InTty` and `OutTty`, to help encapsulate the difference on Windows (on Linux, these point to the same pipe) Future improvements: - The `isatty` implementation doesn't support accurate detection of `conhost` pty's without an associated process. In lieu of a more robust check, I've added a `--force-tty` flag intended for test case use - that forces the CLI to run in tty mode. - It seems the windows implementation doesn't support setting a deadline. This is needed for the expect.Timeout API, but isn't used by us yet. Fixes #241
137 lines
3.7 KiB
Go
137 lines
3.7 KiB
Go
package cli
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"os/user"
|
|
"strings"
|
|
|
|
"github.com/fatih/color"
|
|
"github.com/go-playground/validator/v10"
|
|
"github.com/manifoldco/promptui"
|
|
"github.com/spf13/cobra"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/coderd"
|
|
"github.com/coder/coder/codersdk"
|
|
)
|
|
|
|
func login() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "login <url>",
|
|
Args: cobra.ExactArgs(1),
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
rawURL := args[0]
|
|
|
|
if !strings.HasPrefix(rawURL, "http://") && !strings.HasPrefix(rawURL, "https://") {
|
|
scheme := "https"
|
|
if strings.HasPrefix(rawURL, "localhost") {
|
|
scheme = "http"
|
|
}
|
|
rawURL = fmt.Sprintf("%s://%s", scheme, rawURL)
|
|
}
|
|
serverURL, err := url.Parse(rawURL)
|
|
if err != nil {
|
|
return xerrors.Errorf("parse raw url %q: %w", rawURL, err)
|
|
}
|
|
// Default to HTTPs. Enables simple URLs like: master.cdr.dev
|
|
if serverURL.Scheme == "" {
|
|
serverURL.Scheme = "https"
|
|
}
|
|
|
|
client := codersdk.New(serverURL)
|
|
hasInitialUser, err := client.HasInitialUser(cmd.Context())
|
|
if err != nil {
|
|
return xerrors.Errorf("has initial user: %w", err)
|
|
}
|
|
if !hasInitialUser {
|
|
if !isTTY(cmd) {
|
|
return xerrors.New("the initial user cannot be created in non-interactive mode. use the API")
|
|
}
|
|
_, _ = fmt.Fprintf(cmd.OutOrStdout(), "%s Your Coder deployment hasn't been set up!\n", color.HiBlackString(">"))
|
|
|
|
_, err := prompt(cmd, &promptui.Prompt{
|
|
Label: "Would you like to create the first user?",
|
|
IsConfirm: true,
|
|
Default: "y",
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("create user prompt: %w", err)
|
|
}
|
|
currentUser, err := user.Current()
|
|
if err != nil {
|
|
return xerrors.Errorf("get current user: %w", err)
|
|
}
|
|
username, err := prompt(cmd, &promptui.Prompt{
|
|
Label: "What username would you like?",
|
|
Default: currentUser.Username,
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("pick username prompt: %w", err)
|
|
}
|
|
|
|
organization, err := prompt(cmd, &promptui.Prompt{
|
|
Label: "What is the name of your organization?",
|
|
Default: "acme-corp",
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("pick organization prompt: %w", err)
|
|
}
|
|
|
|
email, err := prompt(cmd, &promptui.Prompt{
|
|
Label: "What's your email?",
|
|
Validate: func(s string) error {
|
|
err := validator.New().Var(s, "email")
|
|
if err != nil {
|
|
return xerrors.New("That's not a valid email address!")
|
|
}
|
|
return err
|
|
},
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("specify email prompt: %w", err)
|
|
}
|
|
|
|
password, err := prompt(cmd, &promptui.Prompt{
|
|
Label: "Enter a password:",
|
|
Mask: '*',
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("specify password prompt: %w", err)
|
|
}
|
|
|
|
_, err = client.CreateInitialUser(cmd.Context(), coderd.CreateInitialUserRequest{
|
|
Email: email,
|
|
Username: username,
|
|
Password: password,
|
|
Organization: organization,
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("create initial user: %w", err)
|
|
}
|
|
resp, err := client.LoginWithPassword(cmd.Context(), coderd.LoginWithPasswordRequest{
|
|
Email: email,
|
|
Password: password,
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("login with password: %w", err)
|
|
}
|
|
config := createConfig(cmd)
|
|
err = config.Session().Write(resp.SessionToken)
|
|
if err != nil {
|
|
return xerrors.Errorf("write session token: %w", err)
|
|
}
|
|
err = config.URL().Write(serverURL.String())
|
|
if err != nil {
|
|
return xerrors.Errorf("write server url: %w", err)
|
|
}
|
|
|
|
_, _ = fmt.Fprintf(cmd.OutOrStdout(), "%s Welcome to Coder, %s! You're authenticated.\n", color.HiBlackString(">"), color.HiCyanString(username))
|
|
return nil
|
|
}
|
|
|
|
return nil
|
|
},
|
|
}
|
|
}
|