diff --git a/tools/src/cts/expectations/update.go b/tools/src/cts/expectations/update.go index 155e142090..222914273c 100644 --- a/tools/src/cts/expectations/update.go +++ b/tools/src/cts/expectations/update.go @@ -73,6 +73,10 @@ func (c *Content) Update(results result.List, testlist []query.Query) (Diagnosti tagSets: tagSets, } + if err := u.preserveRetryOnFailures(); err != nil { + return nil, err + } + // Update those expectations! if err := u.build(); err != nil { return nil, fmt.Errorf("while updating expectations: %w", err) @@ -263,6 +267,35 @@ func (qt *queryTree) markAsConsumed(q query.Query, t result.Tags, line int) { } } +// preserveRetryOnFailures changes any results matching expectations with a +// RetryOnFailure expectation to RetryOnFailure. +func (u *updater) preserveRetryOnFailures() error { + // For each expectation... + for _, c := range u.in.Chunks { + for _, ex := range c.Expectations { + // Does the expectation contain a RetryOnFailure status? + if !container.NewSet(ex.Status...).Contains(string(result.RetryOnFailure)) { + continue // Nope. + } + + q := query.Parse(ex.Query) + + glob, err := u.qt.tree.Glob(q) + if err != nil { + return err + } + for _, indices := range glob { + for _, idx := range indices.Data { + if u.qt.results[idx].Tags.ContainsAll(ex.Tags) { + u.qt.results[idx].Status = result.RetryOnFailure + } + } + } + } + } + return nil +} + // build is the updater top-level function. // build first appends to u.out all chunks from 'u.in' with expectations updated // using the new results, and then appends any new expectations to u.out.