diff --git a/pkg/sarif.go b/pkg/sarif.go index d7b84ab58748..a4e959f7d04c 100644 --- a/pkg/sarif.go +++ b/pkg/sarif.go @@ -369,6 +369,17 @@ func createSARIFRun(uri, toolName, version, commit string, t time.Time, } } +func getOrCreateSARIFRun(runs map[string]*run, runName string, + uri, toolName, version, commit string, t time.Time, + category string) *run { + if prun, exists := runs[runName]; exists { + return prun + } + run := createSARIFRun(uri, toolName, version, commit, t, category, runName) + runs[runName] = &run + return &run +} + func generateRemediationMarkdown(remediation []string) string { r := "" for _, s := range remediation { @@ -427,7 +438,7 @@ func getCheckPolicyInfo(policy *spol.ScorecardPolicy, name string) (minScore int policies := policy.GetPolicies() if _, exists := policies[name]; !exists { return 0, false, - sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("Missing policy for check: %s", name)) + sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("missing policy for check: %s", name)) } cp := policies[name] if cp.GetMode() == spol.CheckPolicy_DISABLED { @@ -436,6 +447,39 @@ func getCheckPolicyInfo(policy *spol.ScorecardPolicy, name string) (minScore int return int(cp.GetScore()), true, nil } +func contains(l []string, elt string) bool { + for _, v := range l { + if v == elt { + return true + } + } + return false +} + +func computeCategory(repos []string) (string, error) { + // local < Git-local < GitHub + switch { + default: + return "", sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("repo types not supported: %v", repos)) + case contains(repos, "local"): + return "local", nil + // Note: Git-local is not supported by any checks yet. + case contains(repos, "Git-local"): + return "local-scm", nil + case contains(repos, "GitHub"), + contains(repos, "GitLab"): + return "online-scm", nil + } +} + +func createSARIFRuns(runs map[string]*run) []run { + res := []run{} + for _, v := range runs { + res = append(res, *v) + } + return res +} + // AsSARIF outputs ScorecardResult in SARIF 2.1.0 format. func (r *ScorecardResult) AsSARIF(showDetails bool, logLevel zapcore.Level, writer io.Writer, checkDocs docs.Doc, policy *spol.ScorecardPolicy, @@ -446,26 +490,25 @@ func (r *ScorecardResult) AsSARIF(showDetails bool, logLevel zapcore.Level, // see https://docs.github.com/en/code-security/secure-coding/integrating-with-code-scanning/sarif-support-for-code-scanning#supported-sarif-output-file-properties, // https://github.com/microsoft/sarif-tutorials. sarif := createSARIFHeader() - runs := []run{} + runs := make(map[string]*run) // nolint for _, check := range r.Checks { - results := []result{} - rules := []rule{} - - doc, e := checkDocs.GetCheck(check.Name) - if e != nil { - return sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("GetCheck: %v: %s", e, check.Name)) + doc, err := checkDocs.GetCheck(check.Name) + if err != nil { + return sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("GetCheck: %v: %s", err, check.Name)) } // We need to create a run entry even if the check is disabled or the policy is satisfied. // The reason is the following: if a check has findings and is later fixed by a user, // the absence of run for the check will indicate that the check was *not* run, // so GitHub would keep the findings in the dahsboard. We don't want that. - run := createSARIFRun("https://github.com/ossf/scorecard", "scorecard", - r.Scorecard.Version, r.Scorecard.CommitSHA, r.Date, - "supply-chain", check.Name) - runs = append(runs, run) + category, err := computeCategory(doc.GetSupportedRepoTypes()) + if err != nil { + return sce.WithMessage(sce.ErrScorecardInternal, fmt.Sprintf("computeCategory: %v: %s", err, check.Name)) + } + run := getOrCreateSARIFRun(runs, category, "https://github.com/ossf/scorecard", "scorecard", + r.Scorecard.Version, r.Scorecard.CommitSHA, r.Date, "supply-chain") // Check the policy configuration. minScore, enabled, err := getCheckPolicyInfo(policy, check.Name) @@ -481,18 +524,15 @@ func (r *ScorecardResult) AsSARIF(showDetails bool, logLevel zapcore.Level, continue } - // We only set rules if we have findings - // to avoid clobbering the results. It would be fine to add - // rules regardless of findings. + // We only set rules if we have findings to avoid clobbering the results. + // It would be fine to add rules regardless of findings. // See https://github.com/github/codeql-action/issues/810#issuecomment-962006930. - // Note: we use a single rule, i.e., one check per run. checkID := check.Name rule := createSARIFRule(check.Name, checkID, doc.GetDocumentationURL(r.Scorecard.CommitSHA), doc.GetDescription(), doc.GetShort(), doc.GetRisk(), doc.GetRemediation(), doc.GetTags()) - rules = append(rules, rule) - runs[len(runs)-1].Tool.Driver.Rules = rules + run.Tool.Driver.Rules = append(run.Tool.Driver.Rules, rule) // Unclear what to use for PartialFingerprints. // GitHub only uses `primaryLocationLineHash`, which is not properly defined @@ -513,22 +553,19 @@ func (r *ScorecardResult) AsSARIF(showDetails bool, logLevel zapcore.Level, // Use the `reason` as message. // Since we have a single rule per run, the indexId is 0. cr := createSARIFCheckResult(0, checkID, check.Reason, &locs[0]) - results = append(results, cr) + run.Results = append(run.Results, cr) } else { for _, loc := range locs { // Use the location's message (check's detail's message) as message. // Since we have a single rule per run, the indexId is 0. cr := createSARIFCheckResult(0, checkID, loc.Message.Text, &loc) - results = append(results, cr) + run.Results = append(run.Results, cr) } } - - // Set the results for the run. - runs[len(runs)-1].Results = results } // Set the sarif's runs. - sarif.Runs = runs + sarif.Runs = createSARIFRuns(runs) encoder := json.NewEncoder(writer) encoder.SetIndent("", " ") diff --git a/pkg/sarif_test.go b/pkg/sarif_test.go index 70ef894e7d91..29a268e3ee7f 100644 --- a/pkg/sarif_test.go +++ b/pkg/sarif_test.go @@ -36,6 +36,7 @@ func sarifMockDocRead() *mockDoc { description: "long description\n other line", url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name", tags: []string{"tag1", "tag2"}, + repos: []string{"GitHub", "local"}, remediation: []string{"not-used1", "not-used2"}, }, "Check-Name2": { @@ -45,6 +46,7 @@ func sarifMockDocRead() *mockDoc { description: "long description\n other line 2", url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name2", tags: []string{" tag1 ", " tag2 ", "tag3"}, + repos: []string{"GitHub", "local"}, remediation: []string{"not-used1", "not-used2"}, }, "Check-Name3": { @@ -54,6 +56,37 @@ func sarifMockDocRead() *mockDoc { description: "long description\n other line 3", url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name3", tags: []string{" tag1", " tag2", "tag3", "tag 4 "}, + repos: []string{"GitHub", "local"}, + remediation: []string{"not-used1", "not-used2"}, + }, + "Check-Name4": { + name: "Check-Name4", + risk: "Low", + short: "short description 4", + description: "long description\n other line 4", + url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name4", + tags: []string{" tag1", " tag2", "tag3", "tag 4 "}, + repos: []string{"GitHub"}, + remediation: []string{"not-used1", "not-used2"}, + }, + "Check-Name5": { + name: "Check-Name5", + risk: "Low", + short: "short description 5", + description: "long description\n other line 5", + url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name5", + tags: []string{" tag1", " tag2", "tag3", "tag 4 "}, + repos: []string{"local"}, + remediation: []string{"not-used1", "not-used2"}, + }, + "Check-Name6": { + name: "Check-Name6", + risk: "Low", + short: "short description 6", + description: "long description\n other line 6", + url: "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name6", + tags: []string{" tag1", " tag2", "tag3", "tag 4 "}, + repos: []string{"Git-local"}, remediation: []string{"not-used1", "not-used2"}, }, } @@ -605,6 +638,94 @@ func TestSARIFOutput(t *testing.T) { Metadata: []string{}, }, }, + { + name: "check-8", + showDetails: true, + expected: "./testdata/check8.sarif", + logLevel: zapcore.DebugLevel, + policy: spol.ScorecardPolicy{ + Version: 1, + Policies: map[string]*spol.CheckPolicy{ + "Check-Name4": &spol.CheckPolicy{ + Score: checker.MaxResultScore, + Mode: spol.CheckPolicy_ENFORCED, + }, + "Check-Name5": &spol.CheckPolicy{ + Score: checker.MaxResultScore, + Mode: spol.CheckPolicy_ENFORCED, + }, + "Check-Name6": &spol.CheckPolicy{ + Score: checker.MaxResultScore, + Mode: spol.CheckPolicy_ENFORCED, + }, + }, + }, + result: ScorecardResult{ + Repo: RepoInfo{ + Name: repoName, + CommitSHA: repoCommit, + }, + Scorecard: ScorecardInfo{ + Version: scorecardVersion, + CommitSHA: scorecardCommit, + }, + Date: date, + Checks: []checker.CheckResult{ + { + Details2: []checker.CheckDetail{ + { + Type: checker.DetailWarn, + Msg: checker.LogMessage{ + Text: "warn message", + Path: "src/file1.cpp", + Type: checker.FileTypeSource, + Offset: 5, + Snippet: "if (bad) {BUG();}", + }, + }, + }, + Score: 5, + Reason: "half score reason", + Name: "Check-Name4", + }, + { + Details2: []checker.CheckDetail{ + { + Type: checker.DetailWarn, + Msg: checker.LogMessage{ + Text: "warn message", + Path: "src/file1.cpp", + Type: checker.FileTypeSource, + Offset: 5, + Snippet: "if (bad) {BUG();}", + }, + }, + }, + Score: 8, + Reason: "half score reason", + Name: "Check-Name5", + }, + { + Details2: []checker.CheckDetail{ + { + Type: checker.DetailWarn, + Msg: checker.LogMessage{ + Text: "warn message", + Path: "src/file1.cpp", + Type: checker.FileTypeSource, + Offset: 5, + Snippet: "if (bad) {BUG();}", + }, + }, + }, + Score: 9, + Reason: "half score reason", + Name: "Check-Name6", + }, + }, + Metadata: []string{}, + }, + }, } for _, tt := range tests { tt := tt // Re-initializing variable so it is not changed while executing the closure below diff --git a/pkg/testdata/check1.sarif b/pkg/testdata/check1.sarif index 8d2e5adfc773..84fe124aae5f 100644 --- a/pkg/testdata/check1.sarif +++ b/pkg/testdata/check1.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { diff --git a/pkg/testdata/check2.sarif b/pkg/testdata/check2.sarif index 46186288ae72..aa46f33d22bf 100644 --- a/pkg/testdata/check2.sarif +++ b/pkg/testdata/check2.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { diff --git a/pkg/testdata/check3.sarif b/pkg/testdata/check3.sarif index b0fb4cddc7b8..d893d3827289 100644 --- a/pkg/testdata/check3.sarif +++ b/pkg/testdata/check3.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { @@ -38,46 +38,7 @@ "tag2" ] } - } - ] - } - }, - "results": [ - { - "ruleId": "Check-Name", - "ruleIndex": 0, - "message": { - "text": "warn message" - }, - "locations": [ - { - "physicalLocation": { - "region": { - "byteOffset": 0 - }, - "artifactLocation": { - "uri": "bin/binary.elf", - "uriBaseId": "%SRCROOT%" - } - }, - "message": { - "text": "warn message" - } - } - ] - } - ] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name2/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3", - "rules": [ + }, { "id": "Check-Name2", "name": "Check-Name2", @@ -105,49 +66,7 @@ "tag3" ] } - } - ] - } - }, - "results": [ - { - "ruleId": "Check-Name2", - "ruleIndex": 0, - "message": { - "text": "warn message" - }, - "locations": [ - { - "physicalLocation": { - "region": { - "charOffset": 3, - "snippet": { - "text": "some text" - } - }, - "artifactLocation": { - "uri": "src/doc.txt", - "uriBaseId": "%SRCROOT%" - } - }, - "message": { - "text": "warn message" - } - } - ] - } - ] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name3/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3", - "rules": [ + }, { "id": "Check-Name3", "name": "Check-Name3", @@ -181,6 +100,55 @@ } }, "results": [ + { + "ruleId": "Check-Name", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "byteOffset": 0 + }, + "artifactLocation": { + "uri": "bin/binary.elf", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + }, + { + "ruleId": "Check-Name2", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "charOffset": 3, + "snippet": { + "text": "some text" + } + }, + "artifactLocation": { + "uri": "src/doc.txt", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + }, { "ruleId": "Check-Name3", "ruleIndex": 0, diff --git a/pkg/testdata/check4.sarif b/pkg/testdata/check4.sarif index b0fb4cddc7b8..d893d3827289 100644 --- a/pkg/testdata/check4.sarif +++ b/pkg/testdata/check4.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { @@ -38,46 +38,7 @@ "tag2" ] } - } - ] - } - }, - "results": [ - { - "ruleId": "Check-Name", - "ruleIndex": 0, - "message": { - "text": "warn message" - }, - "locations": [ - { - "physicalLocation": { - "region": { - "byteOffset": 0 - }, - "artifactLocation": { - "uri": "bin/binary.elf", - "uriBaseId": "%SRCROOT%" - } - }, - "message": { - "text": "warn message" - } - } - ] - } - ] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name2/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3", - "rules": [ + }, { "id": "Check-Name2", "name": "Check-Name2", @@ -105,49 +66,7 @@ "tag3" ] } - } - ] - } - }, - "results": [ - { - "ruleId": "Check-Name2", - "ruleIndex": 0, - "message": { - "text": "warn message" - }, - "locations": [ - { - "physicalLocation": { - "region": { - "charOffset": 3, - "snippet": { - "text": "some text" - } - }, - "artifactLocation": { - "uri": "src/doc.txt", - "uriBaseId": "%SRCROOT%" - } - }, - "message": { - "text": "warn message" - } - } - ] - } - ] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name3/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3", - "rules": [ + }, { "id": "Check-Name3", "name": "Check-Name3", @@ -181,6 +100,55 @@ } }, "results": [ + { + "ruleId": "Check-Name", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "byteOffset": 0 + }, + "artifactLocation": { + "uri": "bin/binary.elf", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + }, + { + "ruleId": "Check-Name2", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "charOffset": 3, + "snippet": { + "text": "some text" + } + }, + "artifactLocation": { + "uri": "src/doc.txt", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + }, { "ruleId": "Check-Name3", "ruleIndex": 0, diff --git a/pkg/testdata/check5.sarif b/pkg/testdata/check5.sarif index 95dd85b1c74c..9f6efe41753b 100644 --- a/pkg/testdata/check5.sarif +++ b/pkg/testdata/check5.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { diff --git a/pkg/testdata/check6.sarif b/pkg/testdata/check6.sarif index f46fb1a09362..4d3e52578a2b 100644 --- a/pkg/testdata/check6.sarif +++ b/pkg/testdata/check6.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { diff --git a/pkg/testdata/check7.sarif b/pkg/testdata/check7.sarif index aa05535ff8b2..aa46f33d22bf 100644 --- a/pkg/testdata/check7.sarif +++ b/pkg/testdata/check7.sarif @@ -4,7 +4,7 @@ "runs": [ { "automationDetails": { - "id": "supply-chain/Check-Name/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" }, "tool": { "driver": { @@ -67,32 +67,6 @@ ] } ] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name2/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3" - } - }, - "results": [] - }, - { - "automationDetails": { - "id": "supply-chain/Check-Name3/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" - }, - "tool": { - "driver": { - "name": "Scorecard", - "informationUri": "https://github.com/ossf/scorecard", - "semanticVersion": "1.2.3" - } - }, - "results": [] } ] } diff --git a/pkg/testdata/check8.sarif b/pkg/testdata/check8.sarif new file mode 100644 index 000000000000..ca8902cf85dc --- /dev/null +++ b/pkg/testdata/check8.sarif @@ -0,0 +1,219 @@ +{ + "$schema": "https://raw.githubusercontent.com/oasis-tcs/sarif-spec/master/Schemata/sarif-schema-2.1.0.json", + "version": "2.1.0", + "runs": [ + { + "automationDetails": { + "id": "supply-chain/online-scm/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + }, + "tool": { + "driver": { + "name": "Scorecard", + "informationUri": "https://github.com/ossf/scorecard", + "semanticVersion": "1.2.3", + "rules": [ + { + "id": "Check-Name4", + "name": "Check-Name4", + "helpUri": "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name4", + "shortDescription": { + "text": "Check-Name4" + }, + "fullDescription": { + "text": "short description 4" + }, + "help": { + "text": "short description 4", + "markdown": "**Remediation**:\n\n- not-used1\n\n- not-used2\n\n\n\n**Severity**: Low\n\n\n\n**Details**:\n\nlong description\n\n other line 4" + }, + "defaultConfiguration": { + "level": "error" + }, + "properties": { + "precision": "high", + "problem.severity": "recommendation", + "security-severity": "1.0", + "tags": [ + "tag1", + "tag2", + "tag3", + "tag 4" + ] + } + } + ] + } + }, + "results": [ + { + "ruleId": "Check-Name4", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "startLine": 5, + "snippet": { + "text": "if (bad) {BUG();}" + } + }, + "artifactLocation": { + "uri": "src/file1.cpp", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + } + ] + }, + { + "automationDetails": { + "id": "supply-chain/local/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + }, + "tool": { + "driver": { + "name": "Scorecard", + "informationUri": "https://github.com/ossf/scorecard", + "semanticVersion": "1.2.3", + "rules": [ + { + "id": "Check-Name5", + "name": "Check-Name5", + "helpUri": "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name5", + "shortDescription": { + "text": "Check-Name5" + }, + "fullDescription": { + "text": "short description 5" + }, + "help": { + "text": "short description 5", + "markdown": "**Remediation**:\n\n- not-used1\n\n- not-used2\n\n\n\n**Severity**: Low\n\n\n\n**Details**:\n\nlong description\n\n other line 5" + }, + "defaultConfiguration": { + "level": "error" + }, + "properties": { + "precision": "high", + "problem.severity": "recommendation", + "security-severity": "1.0", + "tags": [ + "tag1", + "tag2", + "tag3", + "tag 4" + ] + } + } + ] + } + }, + "results": [ + { + "ruleId": "Check-Name5", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "startLine": 5, + "snippet": { + "text": "if (bad) {BUG();}" + } + }, + "artifactLocation": { + "uri": "src/file1.cpp", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + } + ] + }, + { + "automationDetails": { + "id": "supply-chain/local-scm/ccbc59901773ab4c051dfcea0cc4201a1567abdd-17 Aug 21 18:57 +0000" + }, + "tool": { + "driver": { + "name": "Scorecard", + "informationUri": "https://github.com/ossf/scorecard", + "semanticVersion": "1.2.3", + "rules": [ + { + "id": "Check-Name6", + "name": "Check-Name6", + "helpUri": "https://github.com/ossf/scorecard/blob/main/docs/checks.md#check-name6", + "shortDescription": { + "text": "Check-Name6" + }, + "fullDescription": { + "text": "short description 6" + }, + "help": { + "text": "short description 6", + "markdown": "**Remediation**:\n\n- not-used1\n\n- not-used2\n\n\n\n**Severity**: Low\n\n\n\n**Details**:\n\nlong description\n\n other line 6" + }, + "defaultConfiguration": { + "level": "error" + }, + "properties": { + "precision": "high", + "problem.severity": "recommendation", + "security-severity": "1.0", + "tags": [ + "tag1", + "tag2", + "tag3", + "tag 4" + ] + } + } + ] + } + }, + "results": [ + { + "ruleId": "Check-Name6", + "ruleIndex": 0, + "message": { + "text": "warn message" + }, + "locations": [ + { + "physicalLocation": { + "region": { + "startLine": 5, + "snippet": { + "text": "if (bad) {BUG();}" + } + }, + "artifactLocation": { + "uri": "src/file1.cpp", + "uriBaseId": "%SRCROOT%" + } + }, + "message": { + "text": "warn message" + } + } + ] + } + ] + } + ] +}