mirror of
https://github.com/coder/coder.git
synced 2025-07-21 01:28:49 +00:00
feat(cli): allow direct tar upload in template update/create (#5720)
This commit is contained in:
@ -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(¶meterFile, "parameter-file", "", "", "Specify a file path with parameter values.")
|
cmd.Flags().StringVarP(¶meterFile, "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 {
|
||||||
|
@ -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",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -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(¶meterFile, "parameter-file", "", "", "Specify a file path with parameter values.")
|
cmd.Flags().StringVarP(¶meterFile, "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")
|
||||||
|
@ -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) {
|
||||||
|
@ -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.
|
||||||
|
@ -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.
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user