Skip to content

Commit 3d76662

Browse files
authored
feat: add delay time to prevent premature exit before supply chain is ready when waiting (#652)
1 parent 1667904 commit 3d76662

10 files changed

+50
-25
lines changed

docs/command-reference/tanzu_apps_workload_apply.md

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ tanzu apps workload apply --file workload.yaml
2929
-a, --app name application name the workload is a part of
3030
--build-env "key=value" pair build environment variables represented as a "key=value" pair ("key-" to remove, flag can be used multiple times)
3131
--debug put the workload in debug mode (--debug=false to deactivate)
32+
--delay duration delay set to prevent premature exit before supply chain step completion when waiting/tailing (default 30s)
3233
--dry-run print kubernetes resources to stdout rather than apply them to the cluster, messages normally on stdout will be sent to stderr
3334
-e, --env "key=value" pair environment variables represented as a "key=value" pair ("key-" to remove, flag can be used multiple times)
3435
-f, --file file path file path containing the description of a single workload, other flags are layered on top of this resource. Use value "-" to read from stdin

docs/command-reference/tanzu_apps_workload_create.md

+1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ tanzu apps workload create --file workload.yaml
3131
-a, --app name application name the workload is a part of
3232
--build-env "key=value" pair build environment variables represented as a "key=value" pair ("key-" to remove, flag can be used multiple times)
3333
--debug put the workload in debug mode (--debug=false to deactivate)
34+
--delay duration delay set to prevent premature exit before supply chain step completion when waiting/tailing (default 30s)
3435
--dry-run print kubernetes resources to stdout rather than apply them to the cluster, messages normally on stdout will be sent to stderr
3536
-e, --env "key=value" pair environment variables represented as a "key=value" pair ("key-" to remove, flag can be used multiple times)
3637
-f, --file file path file path containing the description of a single workload, other flags are layered on top of this resource. Use value "-" to read from stdin

pkg/cli-runtime/wait/wait.go

+20-2
Original file line numberDiff line numberDiff line change
@@ -32,12 +32,15 @@ var (
3232

3333
type ConditionFunc = func(client.Object) (bool, error)
3434

35-
func UntilCondition(ctx context.Context, watchClient client.WithWatch, target types.NamespacedName, listType client.ObjectList, condition ConditionFunc) error {
35+
func UntilCondition(ctx context.Context, watchClient client.WithWatch, target types.NamespacedName, listType client.ObjectList, condition ConditionFunc, delayTime time.Duration) error {
36+
readyStatus := false
37+
timer := time.NewTimer(delayTime)
3638
eventWatcher, err := watchClient.Watch(ctx, listType, &client.ListOptions{Namespace: target.Namespace})
3739
if err != nil {
3840
return err
3941
}
4042
defer eventWatcher.Stop()
43+
defer timer.Stop()
4144
for {
4245
select {
4346
case event := <-eventWatcher.ResultChan():
@@ -53,10 +56,25 @@ func UntilCondition(ctx context.Context, watchClient client.WithWatch, target ty
5356
return err
5457
}
5558
if cond {
56-
return nil
59+
// Timer is started/reset to track ready status change.
60+
timer.Reset(delayTime)
61+
readyStatus = true
62+
} else {
63+
// This is to capture 'unknown' state to avoid incorrect exit from tailing
64+
readyStatus = false
5765
}
5866
}
5967
}
68+
case <-timer.C:
69+
// Wait until the delay time is met before stopping the tail. This is done to address the use case where
70+
// the workload apply flows through supply chain steps to rerun, which may result in the workload status
71+
// switching between Ready - unknown - Ready - unknown, and so on.
72+
// The delay timer provides an option to allow the supply chain to parse through the steps before exiting the tail.
73+
if readyStatus {
74+
return nil
75+
} else {
76+
timer.Reset(delayTime)
77+
}
6078
case <-ctx.Done():
6179
return ctx.Err()
6280
}

pkg/cli-runtime/wait/wait_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -133,7 +133,7 @@ func TestUntilReady(t *testing.T) {
133133
done := make(chan error, 1)
134134
defer close(done)
135135
go func() {
136-
done <- UntilCondition(ctx, fakeWithWatcher, types.NamespacedName{Name: test.resource.Name, Namespace: test.resource.Namespace}, &cartov1alpha1.WorkloadList{}, test.condFunc)
136+
done <- UntilCondition(ctx, fakeWithWatcher, types.NamespacedName{Name: test.resource.Name, Namespace: test.resource.Namespace}, &cartov1alpha1.WorkloadList{}, test.condFunc, 0*time.Second)
137137
}()
138138

139139
for _, r := range objs {

pkg/commands/workload.go

+6-3
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,7 @@ type WorkloadOptions struct {
147147

148148
Wait bool
149149
WaitTimeout time.Duration
150+
DelayTime time.Duration
150151
Tail bool
151152
TailTimestamps bool
152153
DryRun bool
@@ -896,18 +897,18 @@ func getStatusChangeWorker(c *cli.Config, workload *cartov1alpha1.Workload) wait
896897
}
897898
}
898899
return false, nil
899-
})
900+
}, 0*time.Second)
900901
})
901902
return worker
902903
}
903904

904-
func getReadyConditionWorker(c *cli.Config, workload *cartov1alpha1.Workload) wait.Worker {
905+
func getReadyConditionWorker(c *cli.Config, workload *cartov1alpha1.Workload, delayTime time.Duration) wait.Worker {
905906
worker := wait.Worker(func(ctx context.Context) error {
906907
clientWithWatch, err := watch.GetWatcher(ctx, c)
907908
if err != nil {
908909
return err
909910
}
910-
return wait.UntilCondition(ctx, clientWithWatch, types.NamespacedName{Name: workload.Name, Namespace: workload.Namespace}, &cartov1alpha1.WorkloadList{}, cartov1alpha1.WorkloadReadyConditionFunc)
911+
return wait.UntilCondition(ctx, clientWithWatch, types.NamespacedName{Name: workload.Name, Namespace: workload.Namespace}, &cartov1alpha1.WorkloadList{}, cartov1alpha1.WorkloadReadyConditionFunc, delayTime)
911912
})
912913

913914
return worker
@@ -1018,6 +1019,8 @@ func (opts *WorkloadOptions) DefineFlags(ctx context.Context, c *cli.Config, cmd
10181019
cmd.MarkFlagFilename(cli.StripDash(flags.FilePathFlagName), ".yaml", ".yml")
10191020
cmd.Flags().BoolVar(&opts.DryRun, cli.StripDash(flags.DryRunFlagName), false, "print kubernetes resources to stdout rather than apply them to the cluster, messages normally on stdout will be sent to stderr")
10201021
cmd.Flags().BoolVarP(&opts.Yes, cli.StripDash(flags.YesFlagName), "y", false, "accept all prompts")
1022+
cmd.Flags().DurationVar(&opts.DelayTime, cli.StripDash(flags.DelayTimeFlagName), 30*time.Second, "delay set to prevent premature exit before supply chain step completion when waiting/tailing")
1023+
cmd.RegisterFlagCompletionFunc(cli.StripDash(flags.DelayTimeFlagName), completion.SuggestDurationUnits(ctx, completion.CommonDurationUnits))
10211024
}
10221025

10231026
func (opts *WorkloadOptions) DefineEnvVars(ctx context.Context, c *cli.Config, cmd *cobra.Command) {

pkg/commands/workload_apply.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -225,7 +225,7 @@ func (opts *WorkloadApplyOptions) Exec(ctx context.Context, c *cli.Config) error
225225
}
226226
}
227227

228-
workers = append(workers, getReadyConditionWorker(c, workload))
228+
workers = append(workers, getReadyConditionWorker(c, workload, opts.DelayTime))
229229

230230
if anyTail {
231231
workers = append(workers, getTailWorker(c, workload, opts.TailTimestamps))

pkg/commands/workload_apply_test.go

+12-12
Original file line numberDiff line numberDiff line change
@@ -350,7 +350,7 @@ status:
350350
{
351351
Name: "create - output yaml with wait",
352352
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch,
353-
flags.OutputFlagName, printer.OutputFormatYaml, flags.WaitFlagName, flags.YesFlagName},
353+
flags.OutputFlagName, printer.OutputFormatYaml, flags.WaitFlagName, flags.YesFlagName, flags.DelayTimeFlagName, "0ns"},
354354
GivenObjects: givenNamespaceDefault,
355355
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
356356
workload := &cartov1alpha1.Workload{
@@ -936,7 +936,7 @@ To get status: "tanzu apps workload get my-workload"
936936
{
937937
Name: "wait with timeout error",
938938
Skip: runtm.GOOS == "windows",
939-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.WaitTimeoutFlagName, "1ns"},
939+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.WaitTimeoutFlagName, "1ns", flags.DelayTimeFlagName, "0ns"},
940940
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
941941
workload := &cartov1alpha1.Workload{
942942
ObjectMeta: metav1.ObjectMeta{
@@ -1011,7 +1011,7 @@ Error waiting for ready condition: timeout after 1ns waiting for "my-workload" t
10111011
},
10121012
{
10131013
Name: "create - successful wait for ready cond",
1014-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName},
1014+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
10151015
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
10161016
workload := &cartov1alpha1.Workload{
10171017
ObjectMeta: metav1.ObjectMeta{
@@ -1203,7 +1203,7 @@ Workload "my-workload" is ready
12031203
},
12041204
{
12051205
Name: "create - watcher error",
1206-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName},
1206+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
12071207
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
12081208
fakewatch := watchfakes.NewFakeWithWatch(true, config.Client, []watch.Event{})
12091209
ctx = watchhelper.WithWatcher(ctx, fakewatch)
@@ -1990,7 +1990,7 @@ Error: conflict updating workload, the object was modified by another user; plea
19901990
{
19911991
Name: "update - wait for ready condition - error with timeout",
19921992
Skip: runtm.GOOS == "windows",
1993-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns"},
1993+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns", flags.DelayTimeFlagName, "0ns"},
19941994
GivenObjects: []client.Object{
19951995
parent.
19961996
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -2102,7 +2102,7 @@ Error waiting for ready condition: timeout after 1ns waiting for "my-workload" t
21022102
{
21032103
Name: "update - wait timeout when there is no transition time",
21042104
Skip: runtm.GOOS == "windows",
2105-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns"},
2105+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns", flags.DelayTimeFlagName, "0ns"},
21062106
GivenObjects: []client.Object{
21072107
parent.
21082108
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -2209,7 +2209,7 @@ Error waiting for status change: timeout after 1ns waiting for "my-workload" to
22092209
{
22102210
Name: "update - wait timeout when there is no ready cond",
22112211
Skip: runtm.GOOS == "windows",
2212-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns"},
2212+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns", flags.DelayTimeFlagName, "0ns"},
22132213
GivenObjects: []client.Object{
22142214
parent.
22152215
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -2302,7 +2302,7 @@ Error waiting for status change: timeout after 1ns waiting for "my-workload" to
23022302
{
23032303
Name: "update - wait for timestamp change error with timeout",
23042304
Skip: runtm.GOOS == "windows",
2305-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns"},
2305+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.WaitTimeoutFlagName, "1ns", flags.DelayTimeFlagName, "0ns"},
23062306
GivenObjects: []client.Object{
23072307
parent.
23082308
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -2412,7 +2412,7 @@ Error waiting for status change: timeout after 1ns waiting for "my-workload" to
24122412
},
24132413
{
24142414
Name: "update - wait error for false condition",
2415-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName},
2415+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.DelayTimeFlagName, "0ns"},
24162416
GivenObjects: []client.Object{
24172417
parent.
24182418
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -2523,7 +2523,7 @@ Error waiting for ready condition: Failed to become ready: a hopefully informati
25232523
},
25242524
{
25252525
Name: "update - successful wait for ready condition",
2526-
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName},
2526+
Args: []string{workloadName, flags.ServiceRefFlagName, "database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db", flags.WaitFlagName, flags.YesFlagName, flags.DelayTimeFlagName, "0ns"},
25272527
GivenObjects: []client.Object{
25282528
parent.
25292529
SpecDie(func(d *diecartov1alpha1.WorkloadSpecDie) {
@@ -5356,7 +5356,7 @@ status:
53565356
Name: "output workload after update in yaml format with wait error",
53575357
Args: []string{workloadName, flags.ServiceRefFlagName,
53585358
"database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db",
5359-
flags.OutputFlagName, printer.OutputFormatYml, flags.WaitFlagName},
5359+
flags.OutputFlagName, printer.OutputFormatYml, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
53605360
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
53615361
workload := &cartov1alpha1.Workload{
53625362
ObjectMeta: metav1.ObjectMeta{
@@ -5499,7 +5499,7 @@ status:
54995499
Name: "console interaction - output workload after update in yaml format with wait",
55005500
Args: []string{workloadName, flags.ServiceRefFlagName,
55015501
"database=services.tanzu.vmware.com/v1alpha1:PostgreSQL:my-prod-db",
5502-
flags.OutputFlagName, printer.OutputFormatYml, flags.WaitFlagName},
5502+
flags.OutputFlagName, printer.OutputFormatYml, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
55035503
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
55045504
workload := &cartov1alpha1.Workload{
55055505
ObjectMeta: metav1.ObjectMeta{

pkg/commands/workload_create.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import (
2121
"errors"
2222
"fmt"
2323
"strings"
24+
"time"
2425

2526
"github.com/spf13/cobra"
2627
apierrs "k8s.io/apimachinery/pkg/api/errors"
@@ -144,7 +145,7 @@ func (opts *WorkloadCreateOptions) Exec(ctx context.Context, c *cli.Config) erro
144145
if opts.Wait || anyTail {
145146
cli.PrintPrompt(shouldPrint, c.Infof, "Waiting for workload %q to become ready...\n", opts.Name)
146147

147-
workers = append(workers, getReadyConditionWorker(c, workload))
148+
workers = append(workers, getReadyConditionWorker(c, workload, 0*time.Second))
148149

149150
if anyTail {
150151
workers = append(workers, getTailWorker(c, workload, opts.TailTimestamps))

pkg/commands/workload_create_test.go

+5-5
Original file line numberDiff line numberDiff line change
@@ -185,7 +185,7 @@ status:
185185
{
186186
Name: "create - output yaml with wait",
187187
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch,
188-
flags.OutputFlagName, printer.OutputFormatYaml, flags.WaitFlagName, flags.YesFlagName},
188+
flags.OutputFlagName, printer.OutputFormatYaml, flags.WaitFlagName, flags.YesFlagName, flags.DelayTimeFlagName, "0ns"},
189189
GivenObjects: givenNamespaceDefault,
190190
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
191191
workload := &cartov1alpha1.Workload{
@@ -310,7 +310,7 @@ status:
310310
},
311311
{
312312
Name: "wait error for false condition",
313-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName},
313+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
314314
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
315315
workload := &cartov1alpha1.Workload{
316316
ObjectMeta: metav1.ObjectMeta{
@@ -463,7 +463,7 @@ Error waiting for ready condition: timeout after 1ns waiting for "my-workload" t
463463
},
464464
{
465465
Name: "successful wait for ready cond",
466-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName},
466+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
467467
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
468468
workload := &cartov1alpha1.Workload{
469469
ObjectMeta: metav1.ObjectMeta{
@@ -756,7 +756,7 @@ Error: workload "default/my-workload" already exists
756756
},
757757
{
758758
Name: "watcher error",
759-
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName},
759+
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch, flags.YesFlagName, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
760760
Prepare: func(t *testing.T, ctx context.Context, config *cli.Config, tc *clitesting.CommandTestCase) (context.Context, error) {
761761
fakewatch := watchfakes.NewFakeWithWatch(true, config.Client, []watch.Event{})
762762
ctx = watchhelper.WithWatcher(ctx, fakewatch)
@@ -1901,7 +1901,7 @@ Error waiting for ready condition: failed to create watcher
19011901
Name: "output workload after create in json format with wait error",
19021902
GivenObjects: givenNamespaceDefault,
19031903
Args: []string{workloadName, flags.GitRepoFlagName, gitRepo, flags.GitBranchFlagName, gitBranch,
1904-
flags.TypeFlagName, "web", flags.OutputFlagName, printer.OutputFormatJson, flags.WaitFlagName},
1904+
flags.TypeFlagName, "web", flags.OutputFlagName, printer.OutputFormatJson, flags.WaitFlagName, flags.DelayTimeFlagName, "0ns"},
19051905
WithConsoleInteractions: func(t *testing.T, c *expect.Console) {
19061906
c.ExpectString(clitesting.ToInteractTerminal("Do you want to create this workload? [yN]: "))
19071907
c.Send(clitesting.InteractInputLine("y"))

pkg/flags/flags.go

+1
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ const (
3030
ConfigFlagName = "--config"
3131
ContextFlagName = cli.ContextFlagName
3232
DebugFlagName = "--debug"
33+
DelayTimeFlagName = "--delay"
3334
DryRunFlagName = "--dry-run"
3435
EnvFlagName = "--env"
3536
ExportFlagName = "--export"

0 commit comments

Comments
 (0)