mirror of
https://github.com/coder/coder.git
synced 2025-07-13 21:36:50 +00:00
fix: change coder start to be a no-op if workspace is started
Fixes #11380
This commit is contained in:
37
cli/start.go
37
cli/start.go
@ -30,18 +30,33 @@ func (r *RootCmd) start() *clibase.Cmd {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
var build codersdk.WorkspaceBuild
|
||||||
build, err := startWorkspace(inv, client, workspace, parameterFlags, WorkspaceStart)
|
switch workspace.LatestBuild.Status {
|
||||||
// It's possible for a workspace build to fail due to the template requiring starting
|
case codersdk.WorkspaceStatusRunning:
|
||||||
// workspaces with the active version.
|
_, _ = fmt.Fprintf(
|
||||||
if cerr, ok := codersdk.AsError(err); ok && cerr.StatusCode() == http.StatusForbidden {
|
inv.Stdout, "\nThe %s workspace is already running!\n",
|
||||||
_, _ = fmt.Fprintln(inv.Stdout, "Failed to restart with the template version from your last build. Policy may require you to restart with the current active template version.")
|
cliui.Keyword(workspace.Name),
|
||||||
build, err = startWorkspace(inv, client, workspace, parameterFlags, WorkspaceUpdate)
|
)
|
||||||
if err != nil {
|
return nil
|
||||||
return xerrors.Errorf("start workspace with active template version: %w", err)
|
case codersdk.WorkspaceStatusStarting:
|
||||||
|
_, _ = fmt.Fprintf(
|
||||||
|
inv.Stdout, "\nThe %s workspace is already starting.\n",
|
||||||
|
cliui.Keyword(workspace.Name),
|
||||||
|
)
|
||||||
|
build = workspace.LatestBuild
|
||||||
|
default:
|
||||||
|
build, err = startWorkspace(inv, client, workspace, parameterFlags, WorkspaceStart)
|
||||||
|
// It's possible for a workspace build to fail due to the template requiring starting
|
||||||
|
// workspaces with the active version.
|
||||||
|
if cerr, ok := codersdk.AsError(err); ok && cerr.StatusCode() == http.StatusForbidden {
|
||||||
|
_, _ = fmt.Fprintln(inv.Stdout, "Failed to restart with the template version from your last build. Policy may require you to restart with the current active template version.")
|
||||||
|
build, err = startWorkspace(inv, client, workspace, parameterFlags, WorkspaceUpdate)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("start workspace with active template version: %w", err)
|
||||||
|
}
|
||||||
|
} else if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
} else if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
err = cliui.WorkspaceBuild(inv.Context(), inv.Stdout, client, build.ID)
|
err = cliui.WorkspaceBuild(inv.Context(), inv.Stdout, client, build.ID)
|
||||||
|
@ -11,6 +11,7 @@ import (
|
|||||||
"github.com/coder/coder/v2/cli/clitest"
|
"github.com/coder/coder/v2/cli/clitest"
|
||||||
"github.com/coder/coder/v2/coderd/coderdtest"
|
"github.com/coder/coder/v2/coderd/coderdtest"
|
||||||
"github.com/coder/coder/v2/coderd/database"
|
"github.com/coder/coder/v2/coderd/database"
|
||||||
|
"github.com/coder/coder/v2/coderd/database/dbfake"
|
||||||
"github.com/coder/coder/v2/codersdk"
|
"github.com/coder/coder/v2/codersdk"
|
||||||
"github.com/coder/coder/v2/provisioner/echo"
|
"github.com/coder/coder/v2/provisioner/echo"
|
||||||
"github.com/coder/coder/v2/provisionersdk/proto"
|
"github.com/coder/coder/v2/provisionersdk/proto"
|
||||||
@ -109,6 +110,9 @@ func TestStart(t *testing.T) {
|
|||||||
template := coderdtest.CreateTemplate(t, client, owner.OrganizationID, version.ID)
|
template := coderdtest.CreateTemplate(t, client, owner.OrganizationID, version.ID)
|
||||||
workspace := coderdtest.CreateWorkspace(t, member, owner.OrganizationID, template.ID)
|
workspace := coderdtest.CreateWorkspace(t, member, owner.OrganizationID, template.ID)
|
||||||
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspace.LatestBuild.ID)
|
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspace.LatestBuild.ID)
|
||||||
|
// Stop the workspace
|
||||||
|
workspaceBuild := coderdtest.CreateWorkspaceBuild(t, client, workspace, database.WorkspaceTransitionStop)
|
||||||
|
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspaceBuild.ID)
|
||||||
|
|
||||||
inv, root := clitest.New(t, "start", workspace.Name, "--build-options")
|
inv, root := clitest.New(t, "start", workspace.Name, "--build-options")
|
||||||
clitest.SetupConfig(t, member, root)
|
clitest.SetupConfig(t, member, root)
|
||||||
@ -160,6 +164,9 @@ func TestStart(t *testing.T) {
|
|||||||
template := coderdtest.CreateTemplate(t, client, owner.OrganizationID, version.ID)
|
template := coderdtest.CreateTemplate(t, client, owner.OrganizationID, version.ID)
|
||||||
workspace := coderdtest.CreateWorkspace(t, member, owner.OrganizationID, template.ID)
|
workspace := coderdtest.CreateWorkspace(t, member, owner.OrganizationID, template.ID)
|
||||||
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspace.LatestBuild.ID)
|
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspace.LatestBuild.ID)
|
||||||
|
// Stop the workspace
|
||||||
|
workspaceBuild := coderdtest.CreateWorkspaceBuild(t, client, workspace, database.WorkspaceTransitionStop)
|
||||||
|
coderdtest.AwaitWorkspaceBuildJobCompleted(t, client, workspaceBuild.ID)
|
||||||
|
|
||||||
inv, root := clitest.New(t, "start", workspace.Name,
|
inv, root := clitest.New(t, "start", workspace.Name,
|
||||||
"--build-option", fmt.Sprintf("%s=%s", ephemeralParameterName, ephemeralParameterValue))
|
"--build-option", fmt.Sprintf("%s=%s", ephemeralParameterName, ephemeralParameterValue))
|
||||||
@ -374,3 +381,61 @@ func TestStartAutoUpdate(t *testing.T) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestStart_AlreadyRunning(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
ctx := testutil.Context(t, testutil.WaitShort)
|
||||||
|
|
||||||
|
client, db := coderdtest.NewWithDatabase(t, nil)
|
||||||
|
owner := coderdtest.CreateFirstUser(t, client)
|
||||||
|
memberClient, member := coderdtest.CreateAnotherUser(t, client, owner.OrganizationID)
|
||||||
|
r := dbfake.WorkspaceBuild(t, db, database.Workspace{
|
||||||
|
OwnerID: member.ID,
|
||||||
|
OrganizationID: owner.OrganizationID,
|
||||||
|
}).Do()
|
||||||
|
|
||||||
|
inv, root := clitest.New(t, "start", r.Workspace.Name)
|
||||||
|
clitest.SetupConfig(t, memberClient, root)
|
||||||
|
doneChan := make(chan struct{})
|
||||||
|
pty := ptytest.New(t).Attach(inv)
|
||||||
|
go func() {
|
||||||
|
defer close(doneChan)
|
||||||
|
err := inv.Run()
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}()
|
||||||
|
|
||||||
|
pty.ExpectMatch("workspace is already running")
|
||||||
|
_ = testutil.RequireRecvCtx(ctx, t, doneChan)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStart_Starting(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
ctx := testutil.Context(t, testutil.WaitShort)
|
||||||
|
|
||||||
|
client, db := coderdtest.NewWithDatabase(t, nil)
|
||||||
|
owner := coderdtest.CreateFirstUser(t, client)
|
||||||
|
memberClient, member := coderdtest.CreateAnotherUser(t, client, owner.OrganizationID)
|
||||||
|
r := dbfake.WorkspaceBuild(t, db, database.Workspace{
|
||||||
|
OwnerID: member.ID,
|
||||||
|
OrganizationID: owner.OrganizationID,
|
||||||
|
}).
|
||||||
|
Starting().
|
||||||
|
Do()
|
||||||
|
|
||||||
|
inv, root := clitest.New(t, "start", r.Workspace.Name)
|
||||||
|
clitest.SetupConfig(t, memberClient, root)
|
||||||
|
doneChan := make(chan struct{})
|
||||||
|
pty := ptytest.New(t).Attach(inv)
|
||||||
|
go func() {
|
||||||
|
defer close(doneChan)
|
||||||
|
err := inv.Run()
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}()
|
||||||
|
|
||||||
|
pty.ExpectMatch("workspace is already starting")
|
||||||
|
|
||||||
|
_ = dbfake.JobComplete(t, db, r.Build.JobID).Do()
|
||||||
|
pty.ExpectMatch("workspace has been started")
|
||||||
|
|
||||||
|
_ = testutil.RequireRecvCtx(ctx, t, doneChan)
|
||||||
|
}
|
||||||
|
@ -5,6 +5,7 @@ import (
|
|||||||
"database/sql"
|
"database/sql"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/sqlc-dev/pqtype"
|
"github.com/sqlc-dev/pqtype"
|
||||||
@ -47,6 +48,11 @@ type WorkspaceBuildBuilder struct {
|
|||||||
resources []*sdkproto.Resource
|
resources []*sdkproto.Resource
|
||||||
params []database.WorkspaceBuildParameter
|
params []database.WorkspaceBuildParameter
|
||||||
agentToken string
|
agentToken string
|
||||||
|
dispo workspaceBuildDisposition
|
||||||
|
}
|
||||||
|
|
||||||
|
type workspaceBuildDisposition struct {
|
||||||
|
starting bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// WorkspaceBuild generates a workspace build for the provided workspace.
|
// WorkspaceBuild generates a workspace build for the provided workspace.
|
||||||
@ -100,6 +106,12 @@ func (b WorkspaceBuildBuilder) WithAgent(mutations ...func([]*sdkproto.Agent) []
|
|||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b WorkspaceBuildBuilder) Starting() WorkspaceBuildBuilder {
|
||||||
|
//nolint: revive // returns modified struct
|
||||||
|
b.dispo.starting = true
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
// Do generates all the resources associated with a workspace build.
|
// Do generates all the resources associated with a workspace build.
|
||||||
// Template and TemplateVersion will be optionally populated if no
|
// Template and TemplateVersion will be optionally populated if no
|
||||||
// TemplateID is set on the provided workspace.
|
// TemplateID is set on the provided workspace.
|
||||||
@ -166,20 +178,43 @@ func (b WorkspaceBuildBuilder) Do() WorkspaceResponse {
|
|||||||
})
|
})
|
||||||
require.NoError(b.t, err, "insert job")
|
require.NoError(b.t, err, "insert job")
|
||||||
|
|
||||||
err = b.db.UpdateProvisionerJobWithCompleteByID(ownerCtx, database.UpdateProvisionerJobWithCompleteByIDParams{
|
if b.dispo.starting {
|
||||||
ID: job.ID,
|
// might need to do this multiple times if we got a template version
|
||||||
UpdatedAt: dbtime.Now(),
|
// import job as well
|
||||||
Error: sql.NullString{},
|
for {
|
||||||
ErrorCode: sql.NullString{},
|
j, err := b.db.AcquireProvisionerJob(ownerCtx, database.AcquireProvisionerJobParams{
|
||||||
CompletedAt: sql.NullTime{
|
StartedAt: sql.NullTime{
|
||||||
Time: dbtime.Now(),
|
Time: dbtime.Now(),
|
||||||
Valid: true,
|
Valid: true,
|
||||||
},
|
},
|
||||||
})
|
WorkerID: uuid.NullUUID{
|
||||||
require.NoError(b.t, err, "complete job")
|
UUID: uuid.New(),
|
||||||
|
Valid: true,
|
||||||
|
},
|
||||||
|
Types: []database.ProvisionerType{database.ProvisionerTypeEcho},
|
||||||
|
Tags: nil,
|
||||||
|
})
|
||||||
|
require.NoError(b.t, err, "acquire starting job")
|
||||||
|
if j.ID == job.ID {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
err = b.db.UpdateProvisionerJobWithCompleteByID(ownerCtx, database.UpdateProvisionerJobWithCompleteByIDParams{
|
||||||
|
ID: job.ID,
|
||||||
|
UpdatedAt: dbtime.Now(),
|
||||||
|
Error: sql.NullString{},
|
||||||
|
ErrorCode: sql.NullString{},
|
||||||
|
CompletedAt: sql.NullTime{
|
||||||
|
Time: dbtime.Now(),
|
||||||
|
Valid: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(b.t, err, "complete job")
|
||||||
|
ProvisionerJobResources(b.t, b.db, job.ID, b.seed.Transition, b.resources...).Do()
|
||||||
|
}
|
||||||
|
|
||||||
resp.Build = dbgen.WorkspaceBuild(b.t, b.db, b.seed)
|
resp.Build = dbgen.WorkspaceBuild(b.t, b.db, b.seed)
|
||||||
ProvisionerJobResources(b.t, b.db, job.ID, b.seed.Transition, b.resources...).Do()
|
|
||||||
|
|
||||||
for i := range b.params {
|
for i := range b.params {
|
||||||
b.params[i].WorkspaceBuildID = resp.Build.ID
|
b.params[i].WorkspaceBuildID = resp.Build.ID
|
||||||
@ -340,6 +375,40 @@ func (t TemplateVersionBuilder) Do() TemplateVersionResponse {
|
|||||||
return resp
|
return resp
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type JobCompleteBuilder struct {
|
||||||
|
t testing.TB
|
||||||
|
db database.Store
|
||||||
|
jobID uuid.UUID
|
||||||
|
}
|
||||||
|
|
||||||
|
type JobCompleteResponse struct {
|
||||||
|
CompletedAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
func JobComplete(t testing.TB, db database.Store, jobID uuid.UUID) JobCompleteBuilder {
|
||||||
|
return JobCompleteBuilder{
|
||||||
|
t: t,
|
||||||
|
db: db,
|
||||||
|
jobID: jobID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b JobCompleteBuilder) Do() JobCompleteResponse {
|
||||||
|
r := JobCompleteResponse{CompletedAt: dbtime.Now()}
|
||||||
|
err := b.db.UpdateProvisionerJobWithCompleteByID(ownerCtx, database.UpdateProvisionerJobWithCompleteByIDParams{
|
||||||
|
ID: b.jobID,
|
||||||
|
UpdatedAt: r.CompletedAt,
|
||||||
|
Error: sql.NullString{},
|
||||||
|
ErrorCode: sql.NullString{},
|
||||||
|
CompletedAt: sql.NullTime{
|
||||||
|
Time: r.CompletedAt,
|
||||||
|
Valid: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(b.t, err, "complete job")
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
func must[V any](v V, err error) V {
|
func must[V any](v V, err error) V {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
|
Reference in New Issue
Block a user