-
Notifications
You must be signed in to change notification settings - Fork 8
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add merge reports option in report command #10
Add merge reports option in report command #10
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks! I left some comments. Also it would be nice if we can have a simple test for the MergeReport
function
cmd/diki/app/app.go
Outdated
if len(args) != 1 { | ||
return errors.New("report requires a single filepath argument") | ||
if len(args) == 0 { | ||
return errors.New("report requires a minimum of one filepath arguments") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return errors.New("report requires a minimum of one filepath arguments") | |
return errors.New("report command requires a minimum of one filepath argument") |
cmd/diki/app/app.go
Outdated
} | ||
|
||
if len(args) > 1 && len(opts.distinctBy) == 0 { | ||
return errors.New("report requires a single filepath argument when flag distinct-by is not set ") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return errors.New("report requires a single filepath argument when flag distinct-by is not set ") | |
return errors.New("report command requires a single filepath argument when the distinct-by flag is not set ") |
README.md
Outdated
@@ -38,11 +38,18 @@ diki run --config=config.yaml --provider=gardener --ruleset-id=disa-kubernetes-s | |||
|
|||
#### Report | |||
|
|||
Generate an html report | |||
Diki report generates a human readable html report from the .json output of a `diki run` execution. Merged reports can be generated using the `distinct-by` flag, the value of this flag is a list of key:value pairs where the keys are the IDs of the providers we want to include in the merged report and the values are the unique metadata fields to be used for IDs of the different reports. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Diki report generates a human readable html report from the .json output of a `diki run` execution. Merged reports can be generated using the `distinct-by` flag, the value of this flag is a list of key:value pairs where the keys are the IDs of the providers we want to include in the merged report and the values are the unique metadata fields to be used for IDs of the different reports. | |
Diki can generate a human readable report from the output files of a `diki run` execution. Merged reports can be produced by setting the `distinct-by` flag. The value of this flag is a list of `key=value` pairs where the keys are the IDs of the providers we want to include in the merged report and the values are the unique metadata fields to be used as distinction values between different provider runs. |
cmd/diki/app/app.go
Outdated
for _, arg := range args { | ||
fileData, err := os.ReadFile(filepath.Clean(arg)) | ||
if err != nil { | ||
return fmt.Errorf("failed to read file %s:%w", arg, err) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return fmt.Errorf("failed to read file %s:%w", arg, err) | |
return fmt.Errorf("failed to read file %s: %w", arg, err) |
pkg/report/render.go
Outdated
return r.templates["report"].Execute(w, rep) | ||
return r.templates[tmplReportName].Execute(w, rep) | ||
case *MergedReport: | ||
return r.templates[tmplMergedReportName].Execute(w, rep) | ||
default: | ||
return fmt.Errorf("unsupported report type: %T", rep) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return fmt.Errorf("unsupported report type: %T", rep) | |
return fmt.Errorf("unsupported report type: %T", report) |
pkg/report/merged_report.go
Outdated
} | ||
|
||
if _, ok := mergedProvider.Metadata[uniqueAttr]; ok { | ||
return &MergedReport{}, fmt.Errorf("distinct attribute %s is not unique", mergedProvider.DistinctBy) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return &MergedReport{}, fmt.Errorf("distinct attribute %s is not unique", mergedProvider.DistinctBy) | |
return nil, fmt.Errorf("distinct attribute %s is not unique", mergedProvider.DistinctBy) |
pkg/report/merged_report.go
Outdated
} | ||
|
||
mergedProvider.Metadata[uniqueAttr] = report.Providers[idx].Metadata | ||
mergedProvider.Metadata[uniqueAttr]["time"] = report.Time.Format("01-02-2006") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we also add the time of the report?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That would be a good addition for a more detailed report. I have added HH-MM-SS now as well.
pkg/report/merged_report.go
Outdated
func numOfMergedRulesWithStatus(ruleset *MergedRuleset, status rule.Status) int { | ||
num := 0 | ||
for _, rule := range ruleset.Rules { | ||
for _, check := range rule.Checks { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we can use slices.ContainsFunc
here?
pkg/report/merged_report.go
Outdated
} | ||
|
||
func metadataTextForMergedProvider(mp MergedProvider) map[string]string { | ||
metadataTexts := map[string]string{} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
metadataTexts := map[string]string{} | |
metadataTexts := make(map[string]string, len(mp.Metadata)) |
pkg/report/merged_report.go
Outdated
func metadataTextForMergedProvider(mp MergedProvider) map[string]string { | ||
metadataTexts := map[string]string{} | ||
for id, metadata := range mp.Metadata { | ||
metadataText := "(" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is a good place to use strings.Builder
pkg/report/merged_report.go
Outdated
for key := range distinctByAttrs { | ||
distinctByAttrsProviders = append(distinctByAttrsProviders, key) | ||
} | ||
sort.Strings(distinctByAttrsProviders) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
please use slices.Sort
pkg/report/report_suite_test.go
Outdated
handler := slog.NewJSONHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelInfo}) | ||
logger := slog.New(handler) | ||
testLogger = logger |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not used
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks!
/lgtm
What this PR does / why we need it:
This PR add a new functionality to the
diki report
command which allows the merge of multiplejson
reports into a singlehtml
report. To use this feature thedistinct-by
flag should be set with unique attributes which would be used for each provider to distinct the different reports.TODO: write documentation
Which issue(s) this PR fixes:
Fixes #
Special notes for your reviewer:
Release note: