feat(cli): allow direct tar upload in template update/create (#5720)

This commit is contained in:
Ammar Bandukwala
2023-01-16 14:32:11 -06:00
committed by GitHub
parent 5f7cce775b
commit 592ce3b118
7 changed files with 175 additions and 76 deletions

View File

@ -9,7 +9,6 @@ import (
"time" "time"
"unicode/utf8" "unicode/utf8"
"github.com/briandowns/spinner"
"github.com/google/uuid" "github.com/google/uuid"
"github.com/spf13/cobra" "github.com/spf13/cobra"
"golang.org/x/xerrors" "golang.org/x/xerrors"
@ -19,16 +18,16 @@ import (
"github.com/coder/coder/coderd/util/ptr" "github.com/coder/coder/coderd/util/ptr"
"github.com/coder/coder/codersdk" "github.com/coder/coder/codersdk"
"github.com/coder/coder/provisionerd" "github.com/coder/coder/provisionerd"
"github.com/coder/coder/provisionersdk"
) )
func templateCreate() *cobra.Command { func templateCreate() *cobra.Command {
var ( var (
directory string
provisioner string provisioner string
provisionerTags []string provisionerTags []string
parameterFile string parameterFile string
defaultTTL time.Duration defaultTTL time.Duration
uploadFlags templateUploadFlags
) )
cmd := &cobra.Command{ cmd := &cobra.Command{
Use: "create [name]", Use: "create [name]",
@ -45,11 +44,9 @@ func templateCreate() *cobra.Command {
return err return err
} }
var templateName string templateName, err := uploadFlags.templateName(args)
if len(args) == 0 { if err != nil {
templateName = filepath.Base(directory) return err
} else {
templateName = args[0]
} }
if utf8.RuneCountInString(templateName) > 31 { if utf8.RuneCountInString(templateName) > 31 {
@ -62,32 +59,11 @@ func templateCreate() *cobra.Command {
} }
// Confirm upload of the directory. // Confirm upload of the directory.
prettyDir := prettyDirectoryPath(directory) resp, err := uploadFlags.upload(cmd, client)
_, err = cliui.Prompt(cmd, cliui.PromptOptions{
Text: fmt.Sprintf("Create and upload %q?", prettyDir),
IsConfirm: true,
Default: cliui.ConfirmYes,
})
if err != nil { if err != nil {
return err return err
} }
spin := spinner.New(spinner.CharSets[5], 100*time.Millisecond)
spin.Writer = cmd.OutOrStdout()
spin.Suffix = cliui.Styles.Keyword.Render(" Uploading directory...")
spin.Start()
defer spin.Stop()
archive, err := provisionersdk.Tar(directory, provisionersdk.TemplateArchiveLimit)
if err != nil {
return err
}
resp, err := client.Upload(cmd.Context(), codersdk.ContentTypeTar, archive)
if err != nil {
return err
}
spin.Stop()
tags, err := ParseProvisionerTags(provisionerTags) tags, err := ParseProvisionerTags(provisionerTags)
if err != nil { if err != nil {
return err return err
@ -105,6 +81,7 @@ func templateCreate() *cobra.Command {
return err return err
} }
if !uploadFlags.stdin() {
_, err = cliui.Prompt(cmd, cliui.PromptOptions{ _, err = cliui.Prompt(cmd, cliui.PromptOptions{
Text: "Confirm create?", Text: "Confirm create?",
IsConfirm: true, IsConfirm: true,
@ -112,6 +89,7 @@ func templateCreate() *cobra.Command {
if err != nil { if err != nil {
return err return err
} }
}
createReq := codersdk.CreateTemplateRequest{ createReq := codersdk.CreateTemplateRequest{
Name: templateName, Name: templateName,
@ -134,12 +112,11 @@ func templateCreate() *cobra.Command {
return nil return nil
}, },
} }
currentDirectory, _ := os.Getwd()
cmd.Flags().StringVarP(&directory, "directory", "d", currentDirectory, "Specify the directory to create from")
cmd.Flags().StringVarP(&provisioner, "test.provisioner", "", "terraform", "Customize the provisioner backend")
cmd.Flags().StringVarP(&parameterFile, "parameter-file", "", "", "Specify a file path with parameter values.") cmd.Flags().StringVarP(&parameterFile, "parameter-file", "", "", "Specify a file path with parameter values.")
cmd.Flags().StringArrayVarP(&provisionerTags, "provisioner-tag", "", []string{}, "Specify a set of tags to target provisioner daemons.") cmd.Flags().StringArrayVarP(&provisionerTags, "provisioner-tag", "", []string{}, "Specify a set of tags to target provisioner daemons.")
cmd.Flags().DurationVarP(&defaultTTL, "default-ttl", "", 24*time.Hour, "Specify a default TTL for workspaces created from this template.") cmd.Flags().DurationVarP(&defaultTTL, "default-ttl", "", 24*time.Hour, "Specify a default TTL for workspaces created from this template.")
uploadFlags.register(cmd.Flags())
cmd.Flags().StringVarP(&provisioner, "test.provisioner", "", "terraform", "Customize the provisioner backend")
// This is for testing! // This is for testing!
err := cmd.Flags().MarkHidden("test.provisioner") err := cmd.Flags().MarkHidden("test.provisioner")
if err != nil { if err != nil {

View File

@ -1,6 +1,7 @@
package cli_test package cli_test
import ( import (
"bytes"
"os" "os"
"testing" "testing"
@ -69,7 +70,7 @@ func TestTemplateCreate(t *testing.T) {
match string match string
write string write string
}{ }{
{match: "Create and upload", write: "yes"}, {match: "Upload", write: "yes"},
{match: "compute.main"}, {match: "compute.main"},
{match: "smith (linux, i386)"}, {match: "smith (linux, i386)"},
{match: "Confirm create?", write: "yes"}, {match: "Confirm create?", write: "yes"},
@ -84,6 +85,38 @@ func TestTemplateCreate(t *testing.T) {
require.NoError(t, <-execDone) require.NoError(t, <-execDone)
}) })
t.Run("CreateStdin", func(t *testing.T) {
t.Parallel()
client := coderdtest.New(t, &coderdtest.Options{IncludeProvisionerDaemon: true})
coderdtest.CreateFirstUser(t, client)
source, err := echo.Tar(&echo.Responses{
Parse: echo.ParseComplete,
ProvisionApply: provisionCompleteWithAgent,
})
require.NoError(t, err)
args := []string{
"templates",
"create",
"my-template",
"--directory", "-",
"--test.provisioner", string(database.ProvisionerTypeEcho),
"--default-ttl", "24h",
}
cmd, root := clitest.New(t, args...)
clitest.SetupConfig(t, client, root)
pty := ptytest.New(t)
cmd.SetIn(bytes.NewReader(source))
cmd.SetOut(pty.Output())
execDone := make(chan error)
go func() {
execDone <- cmd.Execute()
}()
require.NoError(t, <-execDone)
})
t.Run("WithParameter", func(t *testing.T) { t.Run("WithParameter", func(t *testing.T) {
t.Parallel() t.Parallel()
client := coderdtest.New(t, &coderdtest.Options{IncludeProvisionerDaemon: true}) client := coderdtest.New(t, &coderdtest.Options{IncludeProvisionerDaemon: true})
@ -108,7 +141,7 @@ func TestTemplateCreate(t *testing.T) {
match string match string
write string write string
}{ }{
{match: "Create and upload", write: "yes"}, {match: "Upload", write: "yes"},
{match: "Enter a value:", write: "bananas"}, {match: "Enter a value:", write: "bananas"},
{match: "Confirm create?", write: "yes"}, {match: "Confirm create?", write: "yes"},
} }
@ -148,7 +181,7 @@ func TestTemplateCreate(t *testing.T) {
match string match string
write string write string
}{ }{
{match: "Create and upload", write: "yes"}, {match: "Upload", write: "yes"},
{match: "Confirm create?", write: "yes"}, {match: "Confirm create?", write: "yes"},
} }
for _, m := range matches { for _, m := range matches {
@ -188,7 +221,7 @@ func TestTemplateCreate(t *testing.T) {
write string write string
}{ }{
{ {
match: "Create and upload", match: "Upload",
write: "yes", write: "yes",
}, },
{ {

View File

@ -2,12 +2,14 @@ package cli
import ( import (
"fmt" "fmt"
"io"
"os" "os"
"path/filepath" "path/filepath"
"time" "time"
"github.com/briandowns/spinner" "github.com/briandowns/spinner"
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/spf13/pflag"
"golang.org/x/xerrors" "golang.org/x/xerrors"
"github.com/coder/coder/cli/cliui" "github.com/coder/coder/cli/cliui"
@ -16,14 +18,81 @@ import (
"github.com/coder/coder/provisionersdk" "github.com/coder/coder/provisionersdk"
) )
// templateUploadFlags is shared by `templates create` and `templates push`.
type templateUploadFlags struct {
directory string
}
func (pf *templateUploadFlags) register(f *pflag.FlagSet) {
currentDirectory, _ := os.Getwd()
f.StringVarP(&pf.directory, "directory", "d", currentDirectory, "Specify the directory to create from, use '-' to read tar from stdin")
}
func (pf *templateUploadFlags) stdin() bool {
return pf.directory == "-"
}
func (pf *templateUploadFlags) upload(cmd *cobra.Command, client *codersdk.Client) (*codersdk.UploadResponse, error) {
var (
content []byte
err error
)
if pf.stdin() {
content, err = io.ReadAll(cmd.InOrStdin())
} else {
prettyDir := prettyDirectoryPath(pf.directory)
_, err = cliui.Prompt(cmd, cliui.PromptOptions{
Text: fmt.Sprintf("Upload %q?", prettyDir),
IsConfirm: true,
Default: cliui.ConfirmYes,
})
if err != nil {
return nil, err
}
content, err = provisionersdk.Tar(pf.directory, provisionersdk.TemplateArchiveLimit)
}
if err != nil {
return nil, xerrors.Errorf("read tar: %w", err)
}
spin := spinner.New(spinner.CharSets[5], 100*time.Millisecond)
spin.Writer = cmd.OutOrStdout()
spin.Suffix = cliui.Styles.Keyword.Render(" Uploading directory...")
spin.Start()
defer spin.Stop()
resp, err := client.Upload(cmd.Context(), codersdk.ContentTypeTar, content)
if err != nil {
return nil, xerrors.Errorf("upload: %w", err)
}
return &resp, nil
}
func (pf *templateUploadFlags) templateName(args []string) (string, error) {
if pf.stdin() {
// Can't infer name from directory if none provided.
if len(args) == 0 {
return "", xerrors.New("template name argument must be provided")
}
return args[0], nil
}
name := filepath.Base(pf.directory)
if len(args) > 0 {
name = args[0]
}
return name, nil
}
func templatePush() *cobra.Command { func templatePush() *cobra.Command {
var ( var (
directory string
versionName string versionName string
provisioner string provisioner string
parameterFile string parameterFile string
alwaysPrompt bool alwaysPrompt bool
provisionerTags []string provisionerTags []string
uploadFlags templateUploadFlags
) )
cmd := &cobra.Command{ cmd := &cobra.Command{
@ -40,9 +109,9 @@ func templatePush() *cobra.Command {
return err return err
} }
name := filepath.Base(directory) name, err := uploadFlags.templateName(args)
if len(args) > 0 { if err != nil {
name = args[0] return err
} }
template, err := client.TemplateByName(cmd.Context(), organization.ID, name) template, err := client.TemplateByName(cmd.Context(), organization.ID, name)
@ -50,32 +119,11 @@ func templatePush() *cobra.Command {
return err return err
} }
// Confirm upload of the directory. resp, err := uploadFlags.upload(cmd, client)
prettyDir := prettyDirectoryPath(directory)
_, err = cliui.Prompt(cmd, cliui.PromptOptions{
Text: fmt.Sprintf("Upload %q?", prettyDir),
IsConfirm: true,
Default: cliui.ConfirmYes,
})
if err != nil { if err != nil {
return err return err
} }
spin := spinner.New(spinner.CharSets[5], 100*time.Millisecond)
spin.Writer = cmd.OutOrStdout()
spin.Suffix = cliui.Styles.Keyword.Render(" Uploading directory...")
spin.Start()
defer spin.Stop()
content, err := provisionersdk.Tar(directory, provisionersdk.TemplateArchiveLimit)
if err != nil {
return err
}
resp, err := client.Upload(cmd.Context(), codersdk.ContentTypeTar, content)
if err != nil {
return err
}
spin.Stop()
tags, err := ParseProvisionerTags(provisionerTags) tags, err := ParseProvisionerTags(provisionerTags)
if err != nil { if err != nil {
return err return err
@ -112,13 +160,12 @@ func templatePush() *cobra.Command {
}, },
} }
currentDirectory, _ := os.Getwd()
cmd.Flags().StringVarP(&directory, "directory", "d", currentDirectory, "Specify the directory to create from")
cmd.Flags().StringVarP(&provisioner, "test.provisioner", "", "terraform", "Customize the provisioner backend") cmd.Flags().StringVarP(&provisioner, "test.provisioner", "", "terraform", "Customize the provisioner backend")
cmd.Flags().StringVarP(&parameterFile, "parameter-file", "", "", "Specify a file path with parameter values.") cmd.Flags().StringVarP(&parameterFile, "parameter-file", "", "", "Specify a file path with parameter values.")
cmd.Flags().StringVarP(&versionName, "name", "", "", "Specify a name for the new template version. It will be automatically generated if not provided.") cmd.Flags().StringVarP(&versionName, "name", "", "", "Specify a name for the new template version. It will be automatically generated if not provided.")
cmd.Flags().StringArrayVarP(&provisionerTags, "provisioner-tag", "", []string{}, "Specify a set of tags to target provisioner daemons.") cmd.Flags().StringArrayVarP(&provisionerTags, "provisioner-tag", "", []string{}, "Specify a set of tags to target provisioner daemons.")
cmd.Flags().BoolVar(&alwaysPrompt, "always-prompt", false, "Always prompt all parameters. Does not pull parameter values from active template version") cmd.Flags().BoolVar(&alwaysPrompt, "always-prompt", false, "Always prompt all parameters. Does not pull parameter values from active template version")
uploadFlags.register(cmd.Flags())
cliui.AllowSkipPrompt(cmd) cliui.AllowSkipPrompt(cmd)
// This is for testing! // This is for testing!
err := cmd.Flags().MarkHidden("test.provisioner") err := cmd.Flags().MarkHidden("test.provisioner")

View File

@ -1,6 +1,7 @@
package cli_test package cli_test
import ( import (
"bytes"
"context" "context"
"path/filepath" "path/filepath"
"testing" "testing"
@ -208,6 +209,47 @@ func TestTemplatePush(t *testing.T) {
assert.Len(t, templateVersions, 2) assert.Len(t, templateVersions, 2)
assert.NotEqual(t, template.ActiveVersionID, templateVersions[1].ID) assert.NotEqual(t, template.ActiveVersionID, templateVersions[1].ID)
}) })
t.Run("Stdin", func(t *testing.T) {
t.Parallel()
client := coderdtest.New(t, &coderdtest.Options{IncludeProvisionerDaemon: true})
user := coderdtest.CreateFirstUser(t, client)
version := coderdtest.CreateTemplateVersion(t, client, user.OrganizationID, nil)
_ = coderdtest.AwaitTemplateVersionJob(t, client, version.ID)
source, err := echo.Tar(&echo.Responses{
Parse: echo.ParseComplete,
ProvisionApply: echo.ProvisionComplete,
})
require.NoError(t, err)
template := coderdtest.CreateTemplate(t, client, user.OrganizationID, version.ID)
cmd, root := clitest.New(
t, "templates", "push", "--directory", "-",
"--test.provisioner", string(database.ProvisionerTypeEcho),
template.Name,
)
clitest.SetupConfig(t, client, root)
pty := ptytest.New(t)
cmd.SetIn(bytes.NewReader(source))
cmd.SetOut(pty.Output())
execDone := make(chan error)
go func() {
execDone <- cmd.Execute()
}()
require.NoError(t, <-execDone)
// Assert that the template version changed.
templateVersions, err := client.TemplateVersionsByTemplate(context.Background(), codersdk.TemplateVersionsByTemplateRequest{
TemplateID: template.ID,
})
require.NoError(t, err)
assert.Len(t, templateVersions, 2)
assert.NotEqual(t, template.ActiveVersionID, templateVersions[1].ID)
})
} }
func latestTemplateVersion(t *testing.T, client *codersdk.Client, templateID uuid.UUID) (codersdk.TemplateVersion, []codersdk.Parameter) { func latestTemplateVersion(t *testing.T, client *codersdk.Client, templateID uuid.UUID) (codersdk.TemplateVersion, []codersdk.Parameter) {

View File

@ -6,8 +6,8 @@ Usage:
Flags: Flags:
--default-ttl duration Specify a default TTL for workspaces created from this --default-ttl duration Specify a default TTL for workspaces created from this
template. (default 24h0m0s) template. (default 24h0m0s)
-d, --directory string Specify the directory to create from (default -d, --directory string Specify the directory to create from, use '-' to read
"/tmp/coder-cli-test-workdir") tar from stdin (default "/tmp/coder-cli-test-workdir")
-h, --help help for create -h, --help help for create
--parameter-file string Specify a file path with parameter values. --parameter-file string Specify a file path with parameter values.
--provisioner-tag stringArray Specify a set of tags to target provisioner daemons. --provisioner-tag stringArray Specify a set of tags to target provisioner daemons.

View File

@ -6,8 +6,8 @@ Usage:
Flags: Flags:
--always-prompt Always prompt all parameters. Does not pull parameter --always-prompt Always prompt all parameters. Does not pull parameter
values from active template version values from active template version
-d, --directory string Specify the directory to create from (default -d, --directory string Specify the directory to create from, use '-' to read
"/tmp/coder-cli-test-workdir") tar from stdin (default "/tmp/coder-cli-test-workdir")
-h, --help help for push -h, --help help for push
--name string Specify a name for the new template version. It will be --name string Specify a name for the new template version. It will be
automatically generated if not provided. automatically generated if not provided.

View File

@ -88,7 +88,7 @@ func (e *echo) Parse(request *proto.Parse_Request, stream proto.DRPCProvisioner_
if err != nil { if err != nil {
if index == 0 { if index == 0 {
// Error if nothing is around to enable failed states. // Error if nothing is around to enable failed states.
return xerrors.New("no state") return xerrors.Errorf("no state: %w", err)
} }
break break
} }