{"uuid": "2b5f0e1a-554c-4383-ac01-938c566c713a", "vulnerability_lookup_origin": "1a89b78e-f703-45f3-bb86-59eb712668bd", "author": "9f56dd64-161d-43a6-b9c3-555944290a09", "vulnerability": "CVE-2019-8943", "type": "seen", "source": "https://gist.github.com/tu-trinh-scale/cb28617ddc3f6adf2b7f5da89706e0f7", "content": "diff --git a/contrib/wordpress/fixtures/dev/wp-staging.toml b/contrib/wordpress/fixtures/dev/wp-staging.toml\nindex 59d0b4b..9f7917a 100644\n--- a/contrib/wordpress/fixtures/dev/wp-staging.toml\n+++ b/contrib/wordpress/fixtures/dev/wp-staging.toml\n@@ -1,7 +1,7 @@\n [servers.wp-staging]\n host = \"10.0.1.50\"\n scanModules = [\"wordpress\"]\n-ignoreCves = []\n+ignoreCves = [\"CVE-2019-8943\"]\n \n [servers.wp-staging.wordpress]\n osUser = \"www-data\"\ndiff --git a/detector/detector.go b/detector/detector.go\nindex 0da7994..f7331dc 100644\n--- a/detector/detector.go\n+++ b/detector/detector.go\n@@ -134,8 +134,8 @@ func Detect(dbclient DBClient, rs []models.ScanResult, dir string) ([]models.Sca\n \t}\n \n \tfor i, r := range rs {\n-\t\tr = r.FilterByCvssOver(c.Conf.CvssScoreOver)\n-\t\tr = r.FilterUnfixed(c.Conf.IgnoreUnfixed)\n+\t\tr.ScannedCves = r.ScannedCves.FilterByCvssOver(c.Conf.CvssScoreOver)\n+\t\tr.ScannedCves = r.ScannedCves.FilterUnfixed(c.Conf.IgnoreUnfixed)\n \t\tr = r.FilterInactiveWordPressLibs(c.Conf.WpScan.DetectInactive)\n \n \t\t// IgnoreCves\n@@ -145,7 +145,7 @@ func Detect(dbclient DBClient, rs []models.ScanResult, dir string) ([]models.Sca\n \t\t} else if con, ok := c.Conf.Servers[r.ServerName].Containers[r.Container.Name]; ok {\n \t\t\tignoreCves = con.IgnoreCves\n \t\t}\n-\t\tr = r.FilterIgnoreCves(ignoreCves)\n+\t\tr.ScannedCves = r.ScannedCves.FilterIgnoreCves(ignoreCves)\n \n \t\t// ignorePkgs\n \t\tignorePkgsRegexps := []string{}\n@@ -154,7 +154,7 @@ func Detect(dbclient DBClient, rs []models.ScanResult, dir string) ([]models.Sca\n \t\t} else if s, ok := c.Conf.Servers[r.ServerName].Containers[r.Container.Name]; ok {\n \t\t\tignorePkgsRegexps = s.IgnorePkgsRegexp\n \t\t}\n-\t\tr = r.FilterIgnorePkgs(ignorePkgsRegexps)\n+\t\tr.ScannedCves = r.ScannedCves.FilterIgnorePkgs(ignorePkgsRegexps)\n \n \t\t// IgnoreUnscored\n \t\tif c.Conf.IgnoreUnscoredCves {\ndiff --git a/detector/wordpress.go b/detector/wordpress.go\nindex 0aabcdb..eff34f8 100644\n--- a/detector/wordpress.go\n+++ b/detector/wordpress.go\n@@ -61,7 +61,7 @@ func detectWordPressCves(r *models.ScanResult, cnf *c.WpScanConf) (int, error) {\n \t\t\tfmt.Sprintf(\"Failed to get WordPress core version.\"))\n \t}\n \turl := fmt.Sprintf(\"https://wpscan.com/api/v3/wordpresses/%s\", ver)\n-\twpVinfos, err := wpscan(url, ver, cnf.Token)\n+\twpVinfos, err := wpscan(url, models.WPCore, cnf.Token)\n \tif err != nil {\n \t\treturn 0, err\n \t}\n@@ -224,7 +224,7 @@ func httpRequest(url, token string) (string, error) {\n \tdefer cancel()\n \tif err != nil {\n \t\treturn \"\", errof.New(errof.ErrFailedToAccessWpScan,\n-\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. err: %s\", err))\n+\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. URL: %s, err: %s\", url, err))\n \t}\n \treq.Header.Set(\"Authorization\", fmt.Sprintf(\"Token token=%s\", token))\n \tclient, err := util.GetHTTPClient(c.Conf.HTTPProxy)\n@@ -234,12 +234,12 @@ func httpRequest(url, token string) (string, error) {\n \tresp, err := client.Do(req)\n \tif err != nil {\n \t\treturn \"\", errof.New(errof.ErrFailedToAccessWpScan,\n-\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. err: %s\", err))\n+\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. URL: %s, err: %s\", url, err))\n \t}\n \tbody, err := ioutil.ReadAll(resp.Body)\n \tif err != nil {\n \t\treturn \"\", errof.New(errof.ErrFailedToAccessWpScan,\n-\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. err: %s\", err))\n+\t\t\tfmt.Sprintf(\"Failed to access to wpscan.com. URL: %s, err: %s\", url, err))\n \t}\n \tdefer resp.Body.Close()\n \tif resp.StatusCode == 200 {\n@@ -249,9 +249,9 @@ func httpRequest(url, token string) (string, error) {\n \t\treturn \"\", nil\n \t} else if resp.StatusCode == 429 {\n \t\treturn \"\", errof.New(errof.ErrWpScanAPILimitExceeded,\n-\t\t\tfmt.Sprintf(\"wpscan.com API limit exceeded: %+v\", resp.Status))\n+\t\t\tfmt.Sprintf(\"wpscan.com API limit exceeded. URL: %s, status: %+v\", url, resp.Status))\n \t} else {\n-\t\tlogging.Log.Warnf(\"wpscan.com unknown status code: %+v\", resp.Status)\n+\t\tlogging.Log.Warnf(\"wpscan.com unknown status code. URL: %s, status: %+v\", url, resp.Status)\n \t\treturn \"\", nil\n \t}\n }\ndiff --git a/detector/wordpress_core_additional_test.go b/detector/wordpress_core_additional_test.go\nnew file mode 100644\nindex 0000000..fc0c033\n--- /dev/null\n+++ b/detector/wordpress_core_additional_test.go\n@@ -0,0 +1,48 @@\n+package detector\n+\n+import (\n+\t\"reflect\"\n+\t\"testing\"\n+\t\"time\"\n+\n+\t\"github.com/future-architect/vuls/models\"\n+)\n+\n+func TestConvertToVinfosAttributesWordPressCore(t *testing.T) {\n+\tbody := `{\"51\":{\"vulnerabilities\":[{\"id\":\"4474\",\"title\":\"WordPress &lt;= 5.1 - Core XSS\",\"created_at\":\"2019-03-13T00:00:00Z\",\"updated_at\":\"2019-03-14T00:00:00Z\",\"vuln_type\":\"XSS\",\"references\":{\"cve\":[\"2019-8943\"],\"url\":[\"https://example.com\"]},\"fixed_in\":\"5.1.1\"}]}}`\n+\n+\tcreatedAt, err := time.Parse(time.RFC3339, \"2019-03-13T00:00:00Z\")\n+\tif err != nil {\n+\t\tt.Fatal(err)\n+\t}\n+\tupdatedAt, err := time.Parse(time.RFC3339, \"2019-03-14T00:00:00Z\")\n+\tif err != nil {\n+\t\tt.Fatal(err)\n+\t}\n+\n+\tgot, err := convertToVinfos(models.WPCore, body)\n+\tif err != nil {\n+\t\tt.Fatal(err)\n+\t}\n+\twant := []models.VulnInfo{\n+\t\t{\n+\t\t\tCveID: \"CVE-2019-8943\",\n+\t\t\tCveContents: models.NewCveContents(models.CveContent{\n+\t\t\t\tType:         models.WpScan,\n+\t\t\t\tCveID:        \"CVE-2019-8943\",\n+\t\t\t\tTitle:        \"WordPress &lt;= 5.1 - Core XSS\",\n+\t\t\t\tReferences:   []models.Reference{{Link: \"https://example.com\"}},\n+\t\t\t\tPublished:    createdAt,\n+\t\t\t\tLastModified: updatedAt,\n+\t\t\t}),\n+\t\t\tVulnType:    \"XSS\",\n+\t\t\tConfidences: []models.Confidence{models.WpScanMatch},\n+\t\t\tWpPackageFixStats: []models.WpPackageFixStatus{\n+\t\t\t\t{Name: models.WPCore, FixedIn: \"5.1.1\"},\n+\t\t\t},\n+\t\t},\n+\t}\n+\tif !reflect.DeepEqual(got, want) {\n+\t\tt.Fatalf(\"convertToVinfos() = %#v, want %#v\", got, want)\n+\t}\n+}\ndiff --git a/models/scanresults.go b/models/scanresults.go\nindex f22c1bb..abeb284 100644\n--- a/models/scanresults.go\n+++ b/models/scanresults.go\n@@ -4,14 +4,12 @@ import (\n \t\"bytes\"\n \t\"fmt\"\n \t\"reflect\"\n-\t\"regexp\"\n \t\"strings\"\n \t\"time\"\n \n \t\"github.com/future-architect/vuls/config\"\n \t\"github.com/future-architect/vuls/constant\"\n \t\"github.com/future-architect/vuls/cwe\"\n-\t\"github.com/future-architect/vuls/logging\"\n )\n \n // ScanResults is a slide of ScanResult\n@@ -84,85 +82,25 @@ type Kernel struct {\n \n // FilterByCvssOver is filter function.\n func (r ScanResult) FilterByCvssOver(over float64) ScanResult {\n-\tfiltered := r.ScannedCves.Find(func(v VulnInfo) bool {\n-\t\tif over &lt;= v.MaxCvssScore().Value.Score {\n-\t\t\treturn true\n-\t\t}\n-\t\treturn false\n-\t})\n-\tr.ScannedCves = filtered\n+\tr.ScannedCves = r.ScannedCves.FilterByCvssOver(over)\n \treturn r\n }\n \n // FilterIgnoreCves is filter function.\n func (r ScanResult) FilterIgnoreCves(ignoreCves []string) ScanResult {\n-\tfiltered := r.ScannedCves.Find(func(v VulnInfo) bool {\n-\t\tfor _, c := range ignoreCves {\n-\t\t\tif v.CveID == c {\n-\t\t\t\treturn false\n-\t\t\t}\n-\t\t}\n-\t\treturn true\n-\t})\n-\tr.ScannedCves = filtered\n+\tr.ScannedCves = r.ScannedCves.FilterIgnoreCves(ignoreCves)\n \treturn r\n }\n \n // FilterUnfixed is filter function.\n func (r ScanResult) FilterUnfixed(ignoreUnfixed bool) ScanResult {\n-\tif !ignoreUnfixed {\n-\t\treturn r\n-\t}\n-\tfiltered := r.ScannedCves.Find(func(v VulnInfo) bool {\n-\t\t// Report cves detected by CPE because Vuls can't know 'fixed' or 'unfixed'\n-\t\tif len(v.CpeURIs) != 0 {\n-\t\t\treturn true\n-\t\t}\n-\t\tNotFixedAll := true\n-\t\tfor _, p := range v.AffectedPackages {\n-\t\t\tNotFixedAll = NotFixedAll &amp;&amp; p.NotFixedYet\n-\t\t}\n-\t\treturn !NotFixedAll\n-\t})\n-\tr.ScannedCves = filtered\n+\tr.ScannedCves = r.ScannedCves.FilterUnfixed(ignoreUnfixed)\n \treturn r\n }\n \n // FilterIgnorePkgs is filter function.\n func (r ScanResult) FilterIgnorePkgs(ignorePkgsRegexps []string) ScanResult {\n-\tregexps := []*regexp.Regexp{}\n-\tfor _, pkgRegexp := range ignorePkgsRegexps {\n-\t\tre, err := regexp.Compile(pkgRegexp)\n-\t\tif err != nil {\n-\t\t\tlogging.Log.Warnf(\"Failed to parse %s. err: %+v\", pkgRegexp, err)\n-\t\t\tcontinue\n-\t\t} else {\n-\t\t\tregexps = append(regexps, re)\n-\t\t}\n-\t}\n-\tif len(regexps) == 0 {\n-\t\treturn r\n-\t}\n-\n-\tfiltered := r.ScannedCves.Find(func(v VulnInfo) bool {\n-\t\tif len(v.AffectedPackages) == 0 {\n-\t\t\treturn true\n-\t\t}\n-\t\tfor _, p := range v.AffectedPackages {\n-\t\t\tmatch := false\n-\t\t\tfor _, re := range regexps {\n-\t\t\t\tif re.MatchString(p.Name) {\n-\t\t\t\t\tmatch = true\n-\t\t\t\t}\n-\t\t\t}\n-\t\t\tif !match {\n-\t\t\t\treturn true\n-\t\t\t}\n-\t\t}\n-\t\treturn false\n-\t})\n-\n-\tr.ScannedCves = filtered\n+\tr.ScannedCves = r.ScannedCves.FilterIgnorePkgs(ignorePkgsRegexps)\n \treturn r\n }\n \n@@ -178,6 +116,9 @@ func (r ScanResult) FilterInactiveWordPressLibs(detectInactive bool) ScanResult\n \t\t}\n \t\t// Ignore if all libs in this vulnInfo inactive\n \t\tfor _, wp := range v.WpPackageFixStats {\n+\t\t\tif wp.Name == WPCore {\n+\t\t\t\treturn true\n+\t\t\t}\n \t\t\tif p, ok := r.WordPressPackages.Find(wp.Name); ok {\n \t\t\t\tif p.Status != Inactive {\n \t\t\t\t\treturn true\ndiff --git a/models/scanresults_wordpress_additional_test.go b/models/scanresults_wordpress_additional_test.go\nnew file mode 100644\nindex 0000000..9cd142b\n--- /dev/null\n+++ b/models/scanresults_wordpress_additional_test.go\n@@ -0,0 +1,42 @@\n+package models\n+\n+import (\n+\t\"reflect\"\n+\t\"testing\"\n+)\n+\n+func TestScanResultFilterInactiveWordPressLibsKeepsCore(t *testing.T) {\n+\tr := ScanResult{\n+\t\tScannedCves: VulnInfos{\n+\t\t\t\"CVE-2019-0001\": {\n+\t\t\t\tCveID: \"CVE-2019-0001\",\n+\t\t\t\tWpPackageFixStats: WpPackageFixStats{\n+\t\t\t\t\t{Name: WPCore, FixedIn: \"5.1.1\"},\n+\t\t\t\t},\n+\t\t\t},\n+\t\t\t\"CVE-2019-0002\": {\n+\t\t\t\tCveID: \"CVE-2019-0002\",\n+\t\t\t\tWpPackageFixStats: WpPackageFixStats{\n+\t\t\t\t\t{Name: \"inactive-plugin\", FixedIn: \"1.0.1\"},\n+\t\t\t\t},\n+\t\t\t},\n+\t\t},\n+\t\tWordPressPackages: WordPressPackages{\n+\t\t\t{Name: WPCore, Type: WPCore, Version: \"5.1\"},\n+\t\t\t{Name: \"inactive-plugin\", Type: WPPlugin, Status: Inactive, Version: \"1.0.0\"},\n+\t\t},\n+\t}\n+\n+\tgot := r.FilterInactiveWordPressLibs(false).ScannedCves\n+\twant := VulnInfos{\n+\t\t\"CVE-2019-0001\": {\n+\t\t\tCveID: \"CVE-2019-0001\",\n+\t\t\tWpPackageFixStats: WpPackageFixStats{\n+\t\t\t\t{Name: WPCore, FixedIn: \"5.1.1\"},\n+\t\t\t},\n+\t\t},\n+\t}\n+\tif !reflect.DeepEqual(got, want) {\n+\t\tt.Fatalf(\"FilterInactiveWordPressLibs() = %#v, want %#v\", got, want)\n+\t}\n+}\ndiff --git a/models/vulninfos.go b/models/vulninfos.go\nindex 8ea0567..b55d2b8 100644\n--- a/models/vulninfos.go\n+++ b/models/vulninfos.go\n@@ -3,10 +3,12 @@ package models\n import (\n \t\"bytes\"\n \t\"fmt\"\n+\t\"regexp\"\n \t\"sort\"\n \t\"strings\"\n \t\"time\"\n \n+\t\"github.com/future-architect/vuls/logging\"\n \texploitmodels \"github.com/vulsio/go-exploitdb/models\"\n )\n \n@@ -36,6 +38,84 @@ func (v VulnInfos) FindScoredVulns() VulnInfos {\n \t})\n }\n \n+// FilterByCvssOver returns vulnerabilities whose maximum CVSS score or severity meets the threshold.\n+func (v VulnInfos) FilterByCvssOver(over float64) VulnInfos {\n+\treturn v.Find(func(vv VulnInfo) bool {\n+\t\treturn over &lt;= vv.MaxCvssScore().Value.Score\n+\t})\n+}\n+\n+// FilterIgnoreCves returns vulnerabilities excluding ignored CVE IDs.\n+func (v VulnInfos) FilterIgnoreCves(ignoreCveIDs []string) VulnInfos {\n+\tignore := map[string]struct{}{}\n+\tfor _, cveID := range ignoreCveIDs {\n+\t\tignore[cveID] = struct{}{}\n+\t}\n+\n+\treturn v.Find(func(vv VulnInfo) bool {\n+\t\t_, found := ignore[vv.CveID]\n+\t\treturn !found\n+\t})\n+}\n+\n+// FilterUnfixed returns vulnerabilities excluding CVEs whose affected packages are all not fixed yet.\n+func (v VulnInfos) FilterUnfixed(ignoreUnfixed bool) VulnInfos {\n+\tif !ignoreUnfixed {\n+\t\treturn v.Find(func(vv VulnInfo) bool {\n+\t\t\treturn true\n+\t\t})\n+\t}\n+\n+\treturn v.Find(func(vv VulnInfo) bool {\n+\t\t// Report CVEs detected by CPE because Vuls can't know fixed or unfixed.\n+\t\tif len(vv.CpeURIs) != 0 {\n+\t\t\treturn true\n+\t\t}\n+\n+\t\tnotFixedAll := true\n+\t\tfor _, p := range vv.AffectedPackages {\n+\t\t\tnotFixedAll = notFixedAll &amp;&amp; p.NotFixedYet\n+\t\t}\n+\t\treturn !notFixedAll\n+\t})\n+}\n+\n+// FilterIgnorePkgs returns vulnerabilities excluding CVEs whose affected packages all match ignored package regexps.\n+func (v VulnInfos) FilterIgnorePkgs(ignorePkgsRegexps []string) VulnInfos {\n+\tregexps := []*regexp.Regexp{}\n+\tfor _, pkgRegexp := range ignorePkgsRegexps {\n+\t\tre, err := regexp.Compile(pkgRegexp)\n+\t\tif err != nil {\n+\t\t\tlogging.Log.Warnf(\"Failed to parse %s. err: %+v\", pkgRegexp, err)\n+\t\t\tcontinue\n+\t\t}\n+\t\tregexps = append(regexps, re)\n+\t}\n+\tif len(regexps) == 0 {\n+\t\treturn v.Find(func(vv VulnInfo) bool {\n+\t\t\treturn true\n+\t\t})\n+\t}\n+\n+\treturn v.Find(func(vv VulnInfo) bool {\n+\t\tif len(vv.AffectedPackages) == 0 {\n+\t\t\treturn true\n+\t\t}\n+\t\tfor _, p := range vv.AffectedPackages {\n+\t\t\tmatch := false\n+\t\t\tfor _, re := range regexps {\n+\t\t\t\tif re.MatchString(p.Name) {\n+\t\t\t\t\tmatch = true\n+\t\t\t\t}\n+\t\t\t}\n+\t\t\tif !match {\n+\t\t\t\treturn true\n+\t\t\t}\n+\t\t}\n+\t\treturn false\n+\t})\n+}\n+\n // ToSortedSlice returns slice of VulnInfos that is sorted by Score, CVE-ID\n func (v VulnInfos) ToSortedSlice() (sorted []VulnInfo) {\n \tfor k := range v {\ndiff --git a/models/vulninfos_filters_additional_test.go b/models/vulninfos_filters_additional_test.go\nnew file mode 100644\nindex 0000000..4cfc7fd\n--- /dev/null\n+++ b/models/vulninfos_filters_additional_test.go\n@@ -0,0 +1,87 @@\n+package models\n+\n+import (\n+\t\"reflect\"\n+\t\"testing\"\n+\t\"time\"\n+)\n+\n+func TestVulnInfosFilterUnfixedKeepsCpeOnly(t *testing.T) {\n+\tvinfos := VulnInfos{\n+\t\t\"CVE-2017-0001\": {\n+\t\t\tCveID:   \"CVE-2017-0001\",\n+\t\t\tCpeURIs: []string{\"cpe:/a:example:example:1.0\"},\n+\t\t},\n+\t\t\"CVE-2017-0002\": {\n+\t\t\tCveID: \"CVE-2017-0002\",\n+\t\t\tAffectedPackages: PackageFixStatuses{\n+\t\t\t\t{Name: \"example\", NotFixedYet: true},\n+\t\t\t},\n+\t\t},\n+\t}\n+\n+\twant := VulnInfos{\n+\t\t\"CVE-2017-0001\": {\n+\t\t\tCveID:   \"CVE-2017-0001\",\n+\t\t\tCpeURIs: []string{\"cpe:/a:example:example:1.0\"},\n+\t\t},\n+\t}\n+\tif got := vinfos.FilterUnfixed(true); !reflect.DeepEqual(got, want) {\n+\t\tt.Fatalf(\"FilterUnfixed() = %#v, want %#v\", got, want)\n+\t}\n+}\n+\n+func TestVulnInfosFiltersAreComposable(t *testing.T) {\n+\tvinfos := VulnInfos{\n+\t\t\"CVE-2017-0001\": {\n+\t\t\tCveID: \"CVE-2017-0001\",\n+\t\t\tCveContents: NewCveContents(CveContent{\n+\t\t\t\tType:         Nvd,\n+\t\t\t\tCveID:        \"CVE-2017-0001\",\n+\t\t\t\tCvss2Score:   8.0,\n+\t\t\t\tLastModified: time.Time{},\n+\t\t\t}),\n+\t\t\tAffectedPackages: PackageFixStatuses{{Name: \"kernel\"}},\n+\t\t},\n+\t\t\"CVE-2017-0002\": {\n+\t\t\tCveID: \"CVE-2017-0002\",\n+\t\t\tCveContents: NewCveContents(CveContent{\n+\t\t\t\tType:         Nvd,\n+\t\t\t\tCveID:        \"CVE-2017-0002\",\n+\t\t\t\tCvss2Score:   8.5,\n+\t\t\t\tLastModified: time.Time{},\n+\t\t\t}),\n+\t\t\tAffectedPackages: PackageFixStatuses{{Name: \"openssl\"}},\n+\t\t},\n+\t\t\"CVE-2017-0003\": {\n+\t\t\tCveID: \"CVE-2017-0003\",\n+\t\t\tCveContents: NewCveContents(CveContent{\n+\t\t\t\tType:         Nvd,\n+\t\t\t\tCveID:        \"CVE-2017-0003\",\n+\t\t\t\tCvss2Score:   4.0,\n+\t\t\t\tLastModified: time.Time{},\n+\t\t\t}),\n+\t\t\tAffectedPackages: PackageFixStatuses{{Name: \"vim\"}},\n+\t\t},\n+\t}\n+\n+\tgot := vinfos.\n+\t\tFilterByCvssOver(7.0).\n+\t\tFilterIgnorePkgs([]string{\"^kernel$\", \"[\"}).\n+\t\tFilterIgnoreCves([]string{\"CVE-2017-0003\"})\n+\twant := VulnInfos{\n+\t\t\"CVE-2017-0002\": {\n+\t\t\tCveID: \"CVE-2017-0002\",\n+\t\t\tCveContents: NewCveContents(CveContent{\n+\t\t\t\tType:         Nvd,\n+\t\t\t\tCveID:        \"CVE-2017-0002\",\n+\t\t\t\tCvss2Score:   8.5,\n+\t\t\t\tLastModified: time.Time{},\n+\t\t\t}),\n+\t\t\tAffectedPackages: PackageFixStatuses{{Name: \"openssl\"}},\n+\t\t},\n+\t}\n+\tif !reflect.DeepEqual(got, want) {\n+\t\tt.Fatalf(\"composed filters = %#v, want %#v\", got, want)\n+\t}\n+}\n", "creation_timestamp": "2026-06-29T07:54:37.424423Z"}