mirror of
https://github.com/anchore/syft.git
synced 2025-11-17 08:23:15 +01:00
* feat: add preRun func to version to restore stdout Signed-off-by: Christopher Phillips <christopher.phillips@anchore.com> * test: add test to capture version in output Signed-off-by: Christopher Phillips <christopher.phillips@anchore.com> * change stdout buffering to log to be opt-in per command Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com> * fix tests Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com> --------- Signed-off-by: Christopher Phillips <christopher.phillips@anchore.com> Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com> Co-authored-by: Alex Goodman <wagoodman@users.noreply.github.com>
104 lines
2.9 KiB
Go
104 lines
2.9 KiB
Go
package commands
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/anchore/clio"
|
|
"github.com/anchore/syft/cmd/syft/cli/options"
|
|
"github.com/anchore/syft/cmd/syft/internal/ui"
|
|
"github.com/anchore/syft/internal"
|
|
"github.com/anchore/syft/internal/log"
|
|
"github.com/anchore/syft/syft/format"
|
|
)
|
|
|
|
const (
|
|
convertExample = ` {{.appName}} {{.command}} img.syft.json -o spdx-json convert a syft SBOM to spdx-json, output goes to stdout
|
|
{{.appName}} {{.command}} img.syft.json -o cyclonedx-json=img.cdx.json convert a syft SBOM to CycloneDX, output is written to the file "img.cdx.json""
|
|
{{.appName}} {{.command}} - -o spdx-json convert an SBOM from STDIN to spdx-json
|
|
`
|
|
)
|
|
|
|
type ConvertOptions struct {
|
|
options.Config `yaml:",inline" mapstructure:",squash"`
|
|
options.Output `yaml:",inline" mapstructure:",squash"`
|
|
options.UpdateCheck `yaml:",inline" mapstructure:",squash"`
|
|
}
|
|
|
|
//nolint:dupl
|
|
func Convert(app clio.Application) *cobra.Command {
|
|
id := app.ID()
|
|
|
|
opts := &ConvertOptions{
|
|
UpdateCheck: options.DefaultUpdateCheck(),
|
|
Output: options.DefaultOutput(),
|
|
}
|
|
|
|
return app.SetupCommand(&cobra.Command{
|
|
Use: "convert [SOURCE-SBOM] -o [FORMAT]",
|
|
Short: "Convert between SBOM formats",
|
|
Long: "[Experimental] Convert SBOM files to, and from, SPDX, CycloneDX and Syft's format. For more info about data loss between formats see https://github.com/anchore/syft#format-conversion-experimental",
|
|
Example: internal.Tprintf(convertExample, map[string]interface{}{
|
|
"appName": id.Name,
|
|
"command": "convert",
|
|
}),
|
|
Args: validateConvertArgs,
|
|
PreRunE: applicationUpdateCheck(id, &opts.UpdateCheck),
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
restoreStdout := ui.CaptureStdoutToTraceLog()
|
|
defer restoreStdout()
|
|
|
|
return RunConvert(opts, args[0])
|
|
},
|
|
}, opts)
|
|
}
|
|
|
|
func validateConvertArgs(cmd *cobra.Command, args []string) error {
|
|
return validateArgs(cmd, args, "an SBOM argument is required")
|
|
}
|
|
|
|
func RunConvert(opts *ConvertOptions, userInput string) error {
|
|
log.Warn("convert is an experimental feature, run `syft convert -h` for help")
|
|
|
|
writer, err := opts.SBOMWriter()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var reader io.ReadSeekCloser
|
|
|
|
if userInput == "-" {
|
|
// though os.Stdin is an os.File, it does not support seeking
|
|
// you will get errors such as "seek /dev/stdin: illegal seek".
|
|
// We need to buffer what we read.
|
|
reader = internal.NewBufferedSeeker(os.Stdin)
|
|
} else {
|
|
f, err := os.Open(userInput)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open SBOM file: %w", err)
|
|
}
|
|
defer func() {
|
|
_ = f.Close()
|
|
}()
|
|
reader = f
|
|
}
|
|
|
|
s, _, _, err := format.Decode(reader)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to decode SBOM: %w", err)
|
|
}
|
|
|
|
if s == nil {
|
|
return fmt.Errorf("no SBOM produced")
|
|
}
|
|
|
|
if err := writer.Write(*s); err != nil {
|
|
return fmt.Errorf("failed to write SBOM: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|