From c2cd2a0b02be09b8cf6d8c019bc7c25aa3d0690d Mon Sep 17 00:00:00 2001 From: Andrew Cholakian Date: Fri, 29 Sep 2023 11:16:05 -0500 Subject: [PATCH] [Heartbeat] Fix missing monitor status on 1 of 2 attempts (#36704) [Heartbeat] Fix missing monitor.status value in initial attempt where max_attempts > 2. Introduced in #36623 adding tests to the scenario runner as well. Original cause was this PR #36519 that did not produce the correct monitor.status: down when the monitor is retried with the second attempt. --- .../wrappers/summarizer/plugstatestat.go | 24 ++++++------ x-pack/heartbeat/scenarios/basics_test.go | 39 +++++++++++++------ .../heartbeat/scenarios/browserscenarios.go | 2 +- .../scenarios/framework/framework.go | 6 +++ x-pack/heartbeat/scenarios/twists.go | 3 ++ 5 files changed, 49 insertions(+), 25 deletions(-) diff --git a/heartbeat/monitors/wrappers/summarizer/plugstatestat.go b/heartbeat/monitors/wrappers/summarizer/plugstatestat.go index f38c22d32abd..62a1c8ee5b85 100644 --- a/heartbeat/monitors/wrappers/summarizer/plugstatestat.go +++ b/heartbeat/monitors/wrappers/summarizer/plugstatestat.go @@ -63,13 +63,15 @@ func (ssp *BrowserStateStatusPlugin) BeforeSummary(event *beat.Event) BeforeSumm } res := ssp.cssp.BeforeSummary(event) - + // Browsers don't set this prior, so we set this here, as opposed to lightweight monitors _, _ = event.PutValue("monitor.status", string(ssp.cssp.js.Status)) + + _, _ = event.PutValue("synthetics", mapstr.M{"type": "heartbeat/summary"}) return res } func (ssp *BrowserStateStatusPlugin) BeforeRetry() { - // noop + ssp.cssp.BeforeRetry() } // LightweightStateStatusPlugin encapsulates the writing of the primary fields used by the summary, @@ -108,7 +110,7 @@ func (ssp *LightweightStateStatusPlugin) BeforeSummary(event *beat.Event) Before } func (ssp *LightweightStateStatusPlugin) BeforeRetry() { - // noop + ssp.cssp.BeforeRetry() } type commonSSP struct { @@ -162,17 +164,10 @@ func (ssp *commonSSP) BeforeSummary(event *beat.Event) BeforeSummaryActions { "summary": &jsCopy, "state": ms, } - if ssp.sf.Type == "browser" { - fields["synthetics"] = mapstr.M{"type": "heartbeat/summary"} - } - eventext.MergeEventFields(event, fields) - if retry { - // mutate the js into the state for the next attempt - ssp.js.BumpAttempt() - } + eventext.MergeEventFields(event, fields) - logp.L().Debugf("attempt info: %v == %v && %d < %d", ssp.js.Status, lastStatus, ssp.js.Attempt, ssp.js.MaxAttempts) + logp.L().Debugf("attempt info: current(%v) == lastStatus(%v) && attempts(%d < %d)", ssp.js.Status, lastStatus, ssp.js.Attempt, ssp.js.MaxAttempts) if retry { return RetryBeforeSummary @@ -180,3 +175,8 @@ func (ssp *commonSSP) BeforeSummary(event *beat.Event) BeforeSummaryActions { return 0 } + +func (ssp *commonSSP) BeforeRetry() { + // mutate the js into the state for the next attempt + ssp.js.BumpAttempt() +} diff --git a/x-pack/heartbeat/scenarios/basics_test.go b/x-pack/heartbeat/scenarios/basics_test.go index a8b39dbfaf10..08794c12146b 100644 --- a/x-pack/heartbeat/scenarios/basics_test.go +++ b/x-pack/heartbeat/scenarios/basics_test.go @@ -22,6 +22,7 @@ import ( "github.com/elastic/beats/v7/heartbeat/monitors/wrappers/monitorstate" "github.com/elastic/beats/v7/heartbeat/monitors/wrappers/summarizer/jobsummary" "github.com/elastic/beats/v7/heartbeat/monitors/wrappers/summarizer/summarizertesthelper" + "github.com/elastic/beats/v7/libbeat/beat" "github.com/elastic/beats/v7/x-pack/heartbeat/scenarios/framework" ) @@ -99,25 +100,22 @@ func TestLightweightUrls(t *testing.T) { func TestLightweightSummaries(t *testing.T) { t.Parallel() - scenarioDB.RunTag(t, "lightweight", func(t *testing.T, mtr *framework.MonitorTestRun, err error) { + scenarioDB.RunTagWithSeparateTwists(t, "lightweight", StdAttemptTwists, func(t *testing.T, mtr *framework.MonitorTestRun, err error) { all := mtr.Events() - lastEvent, firstEvents := all[len(all)-1], all[:len(all)-1] + lastEvent := all[len(all)-1] testslike.Test(t, SummaryValidatorForStatus(mtr.Meta.Status), lastEvent.Fields) - for _, e := range firstEvents { - summary, _ := e.GetValue("summary") - require.Nil(t, summary) - } + requireOneSummaryPerAttempt(t, all) }) } func TestBrowserSummaries(t *testing.T) { t.Parallel() - scenarioDB.RunTag(t, "browser", func(t *testing.T, mtr *framework.MonitorTestRun, err error) { + scenarioDB.RunTagWithSeparateTwists(t, "browser", StdAttemptTwists, func(t *testing.T, mtr *framework.MonitorTestRun, err error) { all := mtr.Events() - lastEvent, firstEvents := all[len(all)-1], all[:len(all)-1] + lastEvent := all[len(all)-1] testslike.Test(t, lookslike.Compose( @@ -126,13 +124,30 @@ func TestBrowserSummaries(t *testing.T) { ), lastEvent.Fields) - for _, e := range firstEvents { - summary, _ := e.GetValue("summary") - require.Nil(t, summary) - } + monStatus, _ := lastEvent.GetValue("monitor.status") + summaryIface, _ := lastEvent.GetValue("summary") + summary := summaryIface.(*jobsummary.JobSummary) + require.Equal(t, string(summary.Status), monStatus, "expected summary status and mon status to be equal in event: %v", lastEvent.Fields) + + requireOneSummaryPerAttempt(t, all) + }) } +func requireOneSummaryPerAttempt(t *testing.T, events []*beat.Event) { + attemptCounter := uint16(1) + // ensure we only have one summary per attempt + for _, e := range events { + summaryIface, _ := e.GetValue("summary") + if summaryIface != nil { + summary := summaryIface.(*jobsummary.JobSummary) + require.Equal(t, attemptCounter, summary.Attempt) + require.LessOrEqual(t, summary.Attempt, summary.MaxAttempts) + attemptCounter++ + } + } +} + func TestRunFromOverride(t *testing.T) { t.Parallel() scenarioDB.RunAllWithATwist(t, TwistAddRunFrom, func(t *testing.T, mtr *framework.MonitorTestRun, err error) { diff --git a/x-pack/heartbeat/scenarios/browserscenarios.go b/x-pack/heartbeat/scenarios/browserscenarios.go index 1760ef587502..e15a150db5f0 100644 --- a/x-pack/heartbeat/scenarios/browserscenarios.go +++ b/x-pack/heartbeat/scenarios/browserscenarios.go @@ -51,7 +51,7 @@ func init() { framework.Scenario{ Name: "failing-browser", Type: "browser", - Tags: []string{"browser", "browser-inline", "down"}, + Tags: []string{"browser", "browser-inline", "down", "browser-down"}, Runner: func(t *testing.T) (config mapstr.M, meta framework.ScenarioRunMeta, close func(), err error) { err = os.Setenv("ELASTIC_SYNTHETICS_CAPABLE", "true") if err != nil { diff --git a/x-pack/heartbeat/scenarios/framework/framework.go b/x-pack/heartbeat/scenarios/framework/framework.go index c4e3e54b5bc7..c6dc0ae9ad6d 100644 --- a/x-pack/heartbeat/scenarios/framework/framework.go +++ b/x-pack/heartbeat/scenarios/framework/framework.go @@ -214,6 +214,12 @@ func (sdb *ScenarioDB) RunTagWithATwist(t *testing.T, tagName string, twist *Twi } } +func (sdb *ScenarioDB) RunTagWithSeparateTwists(t *testing.T, tagName string, twists []*Twist, callback func(*testing.T, *MonitorTestRun, error)) { + for _, twist := range twists { + sdb.RunTagWithATwist(t, tagName, twist, callback) + } +} + type MonitorTestRun struct { StdFields stdfields.StdMonitorFields Meta ScenarioRunMeta diff --git a/x-pack/heartbeat/scenarios/twists.go b/x-pack/heartbeat/scenarios/twists.go index 5f4d1093020a..ad24b8a42f92 100644 --- a/x-pack/heartbeat/scenarios/twists.go +++ b/x-pack/heartbeat/scenarios/twists.go @@ -36,6 +36,9 @@ func TwistMultiRun(times int) *framework.Twist { }) } +// StdAttemptTwists is a list of real world attempt numbers, that is to say both one and two twists. +var StdAttemptTwists = []*framework.Twist{TwistMaxAttempts(1), TwistMaxAttempts(2)} + func TwistMaxAttempts(maxAttempts int) *framework.Twist { return framework.MakeTwist(fmt.Sprintf("run with %d max_attempts", maxAttempts), func(s framework.Scenario) framework.Scenario { s.Tags = append(s.Tags, "retry")