package cmd import ( "fmt" "os" "github.com/anchore/syft/internal" "github.com/anchore/syft/internal/bus" "github.com/anchore/syft/internal/log" "github.com/anchore/syft/internal/ui" "github.com/anchore/syft/internal/version" "github.com/anchore/syft/syft" "github.com/anchore/syft/syft/event" "github.com/anchore/syft/syft/presenter" "github.com/spf13/cobra" "github.com/wagoodman/go-partybus" ) var rootCmd = &cobra.Command{ Use: fmt.Sprintf("%s [SOURCE]", internal.ApplicationName), Short: "A tool for generating a Software Bill Of Materials (SBOM) from container images and filesystems", Long: internal.Tprintf(` Supports the following image sources: {{.appName}} yourrepo/yourimage:tag defaults to using images from a docker daemon {{.appName}} docker://yourrepo/yourimage:tag explicitly use the docker daemon {{.appName}} tar://path/to/yourimage.tar use a tarball from disk {{.appName}} dir://path/to/yourproject read directly from a path in disk `, map[string]interface{}{ "appName": internal.ApplicationName, }), Args: cobra.MaximumNArgs(1), Run: func(cmd *cobra.Command, args []string) { if len(args) == 0 { err := cmd.Help() if err != nil { log.Errorf(err.Error()) os.Exit(1) } os.Exit(1) } err := doRunCmd(cmd, args) if err != nil { log.Errorf(err.Error()) os.Exit(1) } }, } func startWorker(userInput string) <-chan error { errs := make(chan error) go func() { defer close(errs) if appConfig.CheckForAppUpdate { isAvailable, newVersion, err := version.IsUpdateAvailable() if err != nil { log.Errorf(err.Error()) } if isAvailable { log.Infof("new version of %s is available: %s", internal.ApplicationName, newVersion) bus.Publish(partybus.Event{ Type: event.AppUpdateAvailable, Value: newVersion, }) } else { log.Debugf("no new %s update available", internal.ApplicationName) } } config := syft.Config{ PluginDirectory: appConfig.Plugins.Directory, } client, err := syft.NewClient(config) if err != nil { errs <- fmt.Errorf("could not initialize syft: %+v", err) return } catalog, scope, distro, err := client.Catalog(userInput, appConfig.ScopeOpt) if err != nil { errs <- fmt.Errorf("failed to catalog input: %+v", err) return } bus.Publish(partybus.Event{ Type: event.CatalogerFinished, Value: presenter.GetPresenter(appConfig.PresenterOpt, *scope, catalog, distro), }) }() return errs } func doRunCmd(_ *cobra.Command, args []string) error { userInput := args[0] errs := startWorker(userInput) ux := ui.Select(appConfig.CliOptions.Verbosity > 0, appConfig.Quiet) return ux(errs, eventSubscription) }