mirror of
https://github.com/coder/coder.git
synced 2025-07-10 23:53:15 +00:00
* feat: Add stage to build logs This adds a stage property to logs, and refactors the job logs cliui. It also adds tests to the cliui for build logs! * Fix comments
203 lines
6.2 KiB
Go
203 lines
6.2 KiB
Go
package cli
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/briandowns/spinner"
|
|
"github.com/manifoldco/promptui"
|
|
"github.com/spf13/cobra"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/cli/cliui"
|
|
"github.com/coder/coder/coderd/database"
|
|
"github.com/coder/coder/codersdk"
|
|
"github.com/coder/coder/provisionerd"
|
|
"github.com/coder/coder/provisionersdk"
|
|
)
|
|
|
|
func projectCreate() *cobra.Command {
|
|
var (
|
|
yes bool
|
|
directory string
|
|
provisioner string
|
|
)
|
|
cmd := &cobra.Command{
|
|
Use: "create [name]",
|
|
Short: "Create a project from the current directory",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
client, err := createClient(cmd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
organization, err := currentOrganization(cmd, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var projectName string
|
|
if len(args) == 0 {
|
|
projectName = filepath.Base(directory)
|
|
} else {
|
|
projectName = args[0]
|
|
}
|
|
_, err = client.ProjectByName(cmd.Context(), organization.ID, projectName)
|
|
if err == nil {
|
|
return xerrors.Errorf("A project already exists named %q!", projectName)
|
|
}
|
|
|
|
spin := spinner.New(spinner.CharSets[5], 100*time.Millisecond)
|
|
spin.Writer = cmd.OutOrStdout()
|
|
spin.Suffix = cliui.Styles.Keyword.Render(" Uploading current directory...")
|
|
spin.Start()
|
|
defer spin.Stop()
|
|
archive, err := provisionersdk.Tar(directory)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
resp, err := client.Upload(cmd.Context(), codersdk.ContentTypeTar, archive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
spin.Stop()
|
|
|
|
spin = spinner.New(spinner.CharSets[5], 100*time.Millisecond)
|
|
spin.Writer = cmd.OutOrStdout()
|
|
spin.Suffix = cliui.Styles.Keyword.Render("Something")
|
|
job, parameters, err := createValidProjectVersion(cmd, client, organization, database.ProvisionerType(provisioner), resp.Hash)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !yes {
|
|
_, err = cliui.Prompt(cmd, cliui.PromptOptions{
|
|
Text: "Create project?",
|
|
IsConfirm: true,
|
|
Default: "yes",
|
|
})
|
|
if err != nil {
|
|
if errors.Is(err, promptui.ErrAbort) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
}
|
|
|
|
_, err = client.CreateProject(cmd.Context(), organization.ID, codersdk.CreateProjectRequest{
|
|
Name: projectName,
|
|
VersionID: job.ID,
|
|
ParameterValues: parameters,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, _ = fmt.Fprintf(cmd.OutOrStdout(), "The %s project has been created!\n", projectName)
|
|
return nil
|
|
},
|
|
}
|
|
currentDirectory, _ := os.Getwd()
|
|
cmd.Flags().StringVarP(&directory, "directory", "d", currentDirectory, "Specify the directory to create from")
|
|
cmd.Flags().StringVarP(&provisioner, "provisioner", "p", "terraform", "Customize the provisioner backend")
|
|
// This is for testing!
|
|
err := cmd.Flags().MarkHidden("provisioner")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
cmd.Flags().BoolVarP(&yes, "yes", "y", false, "Bypass prompts")
|
|
return cmd
|
|
}
|
|
|
|
func createValidProjectVersion(cmd *cobra.Command, client *codersdk.Client, organization codersdk.Organization, provisioner database.ProvisionerType, hash string, parameters ...codersdk.CreateParameterRequest) (*codersdk.ProjectVersion, []codersdk.CreateParameterRequest, error) {
|
|
before := time.Now()
|
|
version, err := client.CreateProjectVersion(cmd.Context(), organization.ID, codersdk.CreateProjectVersionRequest{
|
|
StorageMethod: database.ProvisionerStorageMethodFile,
|
|
StorageSource: hash,
|
|
Provisioner: provisioner,
|
|
ParameterValues: parameters,
|
|
})
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
err = cliui.ProvisionerJob(cmd, cliui.ProvisionerJobOptions{
|
|
Fetch: func() (codersdk.ProvisionerJob, error) {
|
|
version, err := client.ProjectVersion(cmd.Context(), version.ID)
|
|
return version.Job, err
|
|
},
|
|
Cancel: func() error {
|
|
return client.CancelProjectVersion(cmd.Context(), version.ID)
|
|
},
|
|
Logs: func() (<-chan codersdk.ProvisionerJobLog, error) {
|
|
return client.ProjectVersionLogsAfter(cmd.Context(), version.ID, before)
|
|
},
|
|
})
|
|
if err != nil {
|
|
if !provisionerd.IsMissingParameterError(err.Error()) {
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
version, err = client.ProjectVersion(cmd.Context(), version.ID)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
parameterSchemas, err := client.ProjectVersionSchema(cmd.Context(), version.ID)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
parameterValues, err := client.ProjectVersionParameters(cmd.Context(), version.ID)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
if provisionerd.IsMissingParameterError(version.Job.Error) {
|
|
valuesBySchemaID := map[string]codersdk.ProjectVersionParameter{}
|
|
for _, parameterValue := range parameterValues {
|
|
valuesBySchemaID[parameterValue.SchemaID.String()] = parameterValue
|
|
}
|
|
sort.Slice(parameterSchemas, func(i, j int) bool {
|
|
return parameterSchemas[i].Name < parameterSchemas[j].Name
|
|
})
|
|
missingSchemas := make([]codersdk.ProjectVersionParameterSchema, 0)
|
|
for _, parameterSchema := range parameterSchemas {
|
|
_, ok := valuesBySchemaID[parameterSchema.ID.String()]
|
|
if ok {
|
|
continue
|
|
}
|
|
missingSchemas = append(missingSchemas, parameterSchema)
|
|
}
|
|
_, _ = fmt.Fprintln(cmd.OutOrStdout(), cliui.Styles.Paragraph.Render("This project has required variables! They are scoped to the project, and not viewable after being set.")+"\r\n")
|
|
for _, parameterSchema := range missingSchemas {
|
|
value, err := cliui.ParameterSchema(cmd, parameterSchema)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
parameters = append(parameters, codersdk.CreateParameterRequest{
|
|
Name: parameterSchema.Name,
|
|
SourceValue: value,
|
|
SourceScheme: database.ParameterSourceSchemeData,
|
|
DestinationScheme: parameterSchema.DefaultDestinationScheme,
|
|
})
|
|
_, _ = fmt.Fprintln(cmd.OutOrStdout())
|
|
}
|
|
return createValidProjectVersion(cmd, client, organization, provisioner, hash, parameters...)
|
|
}
|
|
|
|
if version.Job.Status != codersdk.ProvisionerJobSucceeded {
|
|
return nil, nil, xerrors.New(version.Job.Error)
|
|
}
|
|
|
|
_, _ = fmt.Fprintf(cmd.OutOrStdout(), cliui.Styles.Checkmark.String()+" Successfully imported project source!\n")
|
|
|
|
resources, err := client.ProjectVersionResources(cmd.Context(), version.ID)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
return &version, parameters, displayProjectVersionInfo(cmd, resources)
|
|
}
|