2022-10-10 15:05:55 +03:00
|
|
|
//go:build search
|
|
|
|
// +build search
|
2020-10-14 14:47:20 -07:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
|
|
|
|
"github.com/briandowns/spinner"
|
|
|
|
"github.com/spf13/cobra"
|
2022-10-20 19:39:20 +03:00
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
zotErrors "zotregistry.io/zot/errors"
|
2020-07-06 18:44:32 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
func NewCveCommand(searchService SearchService) *cobra.Command {
|
|
|
|
searchCveParams := make(map[string]*string)
|
|
|
|
|
|
|
|
var servURL, user, outputFormat string
|
|
|
|
|
2022-09-23 19:24:01 +03:00
|
|
|
var isSpinner, verifyTLS, fixedFlag, verbose, debug bool
|
2020-07-06 18:44:32 -04:00
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
cveCmd := &cobra.Command{
|
2020-07-06 18:44:32 -04:00
|
|
|
Use: "cve [config-name]",
|
2022-01-11 01:15:35 +00:00
|
|
|
Short: "Lookup CVEs in images hosted on the zot registry",
|
|
|
|
Long: `List CVEs (Common Vulnerabilities and Exposures) of images hosted on the zot registry`,
|
2020-07-06 18:44:32 -04:00
|
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
|
|
home, err := os.UserHomeDir()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
configPath := path.Join(home + "/.zot")
|
|
|
|
if servURL == "" {
|
|
|
|
if len(args) > 0 {
|
|
|
|
urlFromConfig, err := getConfigValue(configPath, args[0], "url")
|
|
|
|
if err != nil {
|
|
|
|
cmd.SilenceUsage = true
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
return err
|
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
if urlFromConfig == "" {
|
|
|
|
return zotErrors.ErrNoURLProvided
|
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
servURL = urlFromConfig
|
|
|
|
} else {
|
|
|
|
return zotErrors.ErrNoURLProvided
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
var err error
|
|
|
|
isSpinner, err = parseBooleanConfig(configPath, args[0], showspinnerConfig)
|
|
|
|
if err != nil {
|
|
|
|
cmd.SilenceUsage = true
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
return err
|
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
verifyTLS, err = parseBooleanConfig(configPath, args[0], verifyTLSConfig)
|
|
|
|
if err != nil {
|
|
|
|
cmd.SilenceUsage = true
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
spin := spinner.New(spinner.CharSets[39], spinnerDuration, spinner.WithWriter(cmd.ErrOrStderr()))
|
2022-01-19 17:57:10 +02:00
|
|
|
spin.Prefix = fmt.Sprintf("Fetching from %s..", servURL)
|
|
|
|
spin.Suffix = "\n\b"
|
2020-07-06 18:44:32 -04:00
|
|
|
|
2021-05-28 19:27:17 +03:00
|
|
|
verbose = false
|
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
searchConfig := searchConfig{
|
|
|
|
params: searchCveParams,
|
|
|
|
searchService: searchService,
|
|
|
|
servURL: &servURL,
|
|
|
|
user: &user,
|
|
|
|
outputFormat: &outputFormat,
|
|
|
|
fixedFlag: &fixedFlag,
|
|
|
|
verifyTLS: &verifyTLS,
|
2021-05-28 19:27:17 +03:00
|
|
|
verbose: &verbose,
|
2022-09-23 19:24:01 +03:00
|
|
|
debug: &debug,
|
2020-07-06 18:44:32 -04:00
|
|
|
resultWriter: cmd.OutOrStdout(),
|
|
|
|
spinner: spinnerState{spin, isSpinner},
|
|
|
|
}
|
|
|
|
|
|
|
|
err = searchCve(searchConfig)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
cmd.SilenceUsage = true
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2020-07-06 18:44:32 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := cveFlagVariables{
|
|
|
|
searchCveParams: searchCveParams,
|
|
|
|
servURL: &servURL,
|
|
|
|
user: &user,
|
|
|
|
outputFormat: &outputFormat,
|
|
|
|
fixedFlag: &fixedFlag,
|
2022-09-23 19:24:01 +03:00
|
|
|
debug: &debug,
|
2020-07-06 18:44:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
setupCveFlags(cveCmd, vars)
|
|
|
|
|
|
|
|
return cveCmd
|
|
|
|
}
|
|
|
|
|
|
|
|
func setupCveFlags(cveCmd *cobra.Command, variables cveFlagVariables) {
|
|
|
|
variables.searchCveParams["imageName"] = cveCmd.Flags().StringP("image", "I", "", "List CVEs by IMAGENAME[:TAG]")
|
2022-01-19 17:57:10 +02:00
|
|
|
variables.searchCveParams["cveID"] = cveCmd.Flags().StringP("cve-id", "i", "", "List images affected by a CVE")
|
2020-07-06 18:44:32 -04:00
|
|
|
|
|
|
|
cveCmd.Flags().StringVar(variables.servURL, "url", "", "Specify zot server URL if config-name is not mentioned")
|
|
|
|
cveCmd.Flags().StringVarP(variables.user, "user", "u", "", `User Credentials of `+
|
|
|
|
`zot server in USERNAME:PASSWORD format`)
|
|
|
|
cveCmd.Flags().StringVarP(variables.outputFormat, "output", "o", "", "Specify output format [text/json/yaml]."+
|
|
|
|
" JSON and YAML format return all info for CVEs")
|
|
|
|
|
|
|
|
cveCmd.Flags().BoolVar(variables.fixedFlag, "fixed", false, "List tags which have fixed a CVE")
|
2022-09-23 19:24:01 +03:00
|
|
|
cveCmd.Flags().BoolVar(variables.debug, "debug", false, "Show debug output")
|
2020-07-06 18:44:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
type cveFlagVariables struct {
|
|
|
|
searchCveParams map[string]*string
|
|
|
|
servURL *string
|
|
|
|
user *string
|
|
|
|
outputFormat *string
|
|
|
|
fixedFlag *bool
|
2022-09-23 19:24:01 +03:00
|
|
|
debug *bool
|
2020-07-06 18:44:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func searchCve(searchConfig searchConfig) error {
|
2022-01-19 17:57:10 +02:00
|
|
|
var searchers []searcher
|
|
|
|
|
2022-10-23 09:44:20 +03:00
|
|
|
if checkExtEndPoint(searchConfig) {
|
2022-01-19 17:57:10 +02:00
|
|
|
searchers = getCveSearchersGQL()
|
|
|
|
} else {
|
|
|
|
searchers = getCveSearchers()
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, searcher := range searchers {
|
2020-07-06 18:44:32 -04:00
|
|
|
found, err := searcher.search(searchConfig)
|
|
|
|
if found {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return zotErrors.ErrInvalidFlagsCombination
|
|
|
|
}
|