mirror of
https://github.com/project-zot/zot.git
synced 2026-06-15 11:37:56 +08:00
2402296e9a
* fix: migrate to Go module v2 for proper semantic versioning This change updates the module path from 'zotregistry.dev/zot' to 'zotregistry.dev/zot/v2' to comply with Go's semantic versioning rules. According to Go's module versioning requirements, major version v2+ must include the major version in the module path. The current module path 'zotregistry.dev/zot' only supports v0.x.x and v1.x.x versions, making existing v2.x.x tags (like v2.1.8) unusable. Changes: - Updated go.mod module path to zotregistry.dev/zot/v2 - Updated all internal import paths across 280+ Go source files - Updated configuration files (golangcilint.yaml, gqlgen.yml) - Updated README.md Go reference badge This fix enables proper use of existing v2.x.x Git tags and allows external packages to import zot v2+ versions without compatibility errors. Resolves: Go module import compatibility for v2+ versions Fixes: #3071 Signed-off-by: Luca Muscariello <muscariello@ieee.org> * fix: regenerate GraphQL files with updated v2 import paths The gqlgen tool needs to regenerate the GraphQL schema files after the module path change to use the new v2 imports. Signed-off-by: Luca Muscariello <muscariello@ieee.org> --------- Signed-off-by: Luca Muscariello <muscariello@ieee.org>
279 lines
7.0 KiB
Go
279 lines
7.0 KiB
Go
//go:build search
|
|
// +build search
|
|
|
|
package client
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
zerr "zotregistry.dev/zot/v2/errors"
|
|
zcommon "zotregistry.dev/zot/v2/pkg/common"
|
|
)
|
|
|
|
const (
|
|
maxRetries = 20
|
|
)
|
|
|
|
func NewCveForImageCommand(searchService SearchService) *cobra.Command {
|
|
var (
|
|
searchedCVEID string
|
|
cveListSortFlag = CVEListSortFlag(SortBySeverity)
|
|
)
|
|
|
|
cveForImageCmd := &cobra.Command{
|
|
Use: "list [repo:tag]|[repo@digest]",
|
|
Short: "List CVEs by REPO:TAG or REPO@DIGEST",
|
|
Long: `List CVEs by REPO:TAG or REPO@DIGEST`,
|
|
Args: OneImageWithRefArg,
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
searchConfig, err := GetSearchConfigFromFlags(cmd, searchService)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = CheckExtEndPointQuery(searchConfig, CVEListForImageQuery())
|
|
if err != nil {
|
|
return fmt.Errorf("%w: '%s'", err, CVEListForImageQuery().Name)
|
|
}
|
|
|
|
image := args[0]
|
|
|
|
return SearchCVEForImageGQL(searchConfig, image, searchedCVEID)
|
|
},
|
|
}
|
|
|
|
cveForImageCmd.Flags().StringVar(&searchedCVEID, SearchedCVEID, "", "Search for a specific CVE by name/id")
|
|
cveForImageCmd.Flags().Var(&cveListSortFlag, SortByFlag,
|
|
fmt.Sprintf("Options for sorting the output: [%s]", CVEListSortOptionsStr()))
|
|
|
|
return cveForImageCmd
|
|
}
|
|
|
|
func NewImagesByCVEIDCommand(searchService SearchService) *cobra.Command {
|
|
var (
|
|
repo string
|
|
imageListSortFlag = ImageListSortFlag(SortByAlphabeticAsc)
|
|
)
|
|
|
|
imagesByCVEIDCmd := &cobra.Command{
|
|
Use: "affected [cveId]",
|
|
Short: "List images affected by a CVE",
|
|
Long: `List images affected by a CVE`,
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
if err := cobra.ExactArgs(1)(cmd, args); err != nil {
|
|
return err
|
|
}
|
|
|
|
if !strings.HasPrefix(args[0], "CVE") {
|
|
return fmt.Errorf("%w: expected a cveid 'CVE-...' got '%s'", zerr.ErrInvalidCLIParameter, args[0])
|
|
}
|
|
|
|
return nil
|
|
},
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
searchConfig, err := GetSearchConfigFromFlags(cmd, searchService)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = CheckExtEndPointQuery(searchConfig, ImageListForCVEQuery())
|
|
if err != nil {
|
|
return fmt.Errorf("%w: '%s'", err, ImageListForCVEQuery().Name)
|
|
}
|
|
|
|
searchedCVEID := args[0]
|
|
|
|
return SearchImagesByCVEIDGQL(searchConfig, repo, searchedCVEID)
|
|
},
|
|
}
|
|
|
|
imagesByCVEIDCmd.Flags().StringVar(&repo, "repo", "", "Search for a specific CVE by name/id")
|
|
imagesByCVEIDCmd.Flags().Var(&imageListSortFlag, SortByFlag,
|
|
fmt.Sprintf("Options for sorting the output: [%s]", ImageListSortOptionsStr()))
|
|
|
|
return imagesByCVEIDCmd
|
|
}
|
|
|
|
func NewFixedTagsCommand(searchService SearchService) *cobra.Command {
|
|
imageListSortFlag := ImageListSortFlag(SortByAlphabeticAsc)
|
|
|
|
fixedTagsCmd := &cobra.Command{
|
|
Use: "fixed [repo] [cveId]",
|
|
Short: "List tags where a CVE is fixed",
|
|
Long: `List tags where a CVE is fixed`,
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
const argCount = 2
|
|
|
|
if err := cobra.ExactArgs(argCount)(cmd, args); err != nil {
|
|
return err
|
|
}
|
|
|
|
if !zcommon.CheckIsCorrectRepoNameFormat(args[0]) {
|
|
return fmt.Errorf("%w: expected a valid repo name for first argument '%s'", zerr.ErrInvalidCLIParameter, args[0])
|
|
}
|
|
|
|
return nil
|
|
},
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
searchConfig, err := GetSearchConfigFromFlags(cmd, searchService)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = CheckExtEndPointQuery(searchConfig, ImageListWithCVEFixedQuery())
|
|
if err != nil {
|
|
return fmt.Errorf("%w: '%s'", err, ImageListWithCVEFixedQuery().Name)
|
|
}
|
|
|
|
repo := args[0]
|
|
searchedCVEID := args[1]
|
|
|
|
return SearchFixedTagsGQL(searchConfig, repo, searchedCVEID)
|
|
},
|
|
}
|
|
|
|
fixedTagsCmd.Flags().Var(&imageListSortFlag, SortByFlag,
|
|
fmt.Sprintf("Options for sorting the output: [%s]", ImageListSortOptionsStr()))
|
|
|
|
return fixedTagsCmd
|
|
}
|
|
|
|
func NewCVEDiffCommand(searchService SearchService) *cobra.Command {
|
|
var (
|
|
minuendStr, minuendArch string
|
|
subtrahendStr, subtrahendArch string
|
|
)
|
|
imagesByCVEIDCmd := &cobra.Command{
|
|
Use: "diff [minuend] ([minuend-platform]) [subtrahend] ([subtrahend-platform])",
|
|
Short: "List the CVE's present in minuend that are not present in subtrahend",
|
|
Long: `List the CVE's present in minuend that are not present in subtrahend`,
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
const (
|
|
twoArgs = 2
|
|
threeArgs = 3
|
|
fourArgs = 4
|
|
)
|
|
|
|
if err := cobra.RangeArgs(twoArgs, fourArgs)(cmd, args); err != nil {
|
|
return err
|
|
}
|
|
|
|
if !isRepoTag(args[0]) {
|
|
return fmt.Errorf("%w: first parameter should be a repo:tag", zerr.ErrInvalidArgs)
|
|
}
|
|
|
|
minuendStr = args[0]
|
|
|
|
if isRepoTag(args[1]) {
|
|
subtrahendStr = args[1]
|
|
} else {
|
|
minuendArch = args[1]
|
|
|
|
if len(args) == twoArgs {
|
|
return fmt.Errorf("%w: not enough arguments, specified only 1 image with arch", zerr.ErrInvalidArgs)
|
|
}
|
|
}
|
|
|
|
if len(args) == twoArgs {
|
|
return nil
|
|
}
|
|
|
|
if isRepoTag(args[2]) {
|
|
if subtrahendStr == "" {
|
|
subtrahendStr = args[2]
|
|
} else {
|
|
return fmt.Errorf("%w: too many repo:tag inputs", zerr.ErrInvalidArgs)
|
|
}
|
|
} else {
|
|
if subtrahendStr == "" {
|
|
return fmt.Errorf("%w: 3rd argument should be a repo:tag", zerr.ErrInvalidArgs)
|
|
} else {
|
|
subtrahendArch = args[2]
|
|
}
|
|
}
|
|
|
|
if len(args) == threeArgs {
|
|
return nil
|
|
}
|
|
|
|
if isRepoTag(args[3]) {
|
|
return fmt.Errorf("%w: 4th argument should not be a repo:tag but an arch", zerr.ErrInvalidArgs)
|
|
} else {
|
|
subtrahendArch = args[3]
|
|
}
|
|
|
|
return nil
|
|
},
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
searchConfig, err := GetSearchConfigFromFlags(cmd, searchService)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = CheckExtEndPointQuery(searchConfig, CVEDiffListForImagesQuery())
|
|
if err != nil {
|
|
return fmt.Errorf("%w: '%s'", err, CVEDiffListForImagesQuery().Name)
|
|
}
|
|
|
|
// parse the args and determine the input
|
|
minuend := getImageIdentifier(minuendStr, minuendArch)
|
|
subtrahend := getImageIdentifier(subtrahendStr, subtrahendArch)
|
|
|
|
return SearchCVEDiffList(searchConfig, minuend, subtrahend)
|
|
},
|
|
}
|
|
|
|
return imagesByCVEIDCmd
|
|
}
|
|
|
|
func isRepoTag(arg string) bool {
|
|
_, _, _, err := zcommon.GetRepoReference(arg) //nolint:dogsled
|
|
|
|
return err == nil
|
|
}
|
|
|
|
type osArch struct {
|
|
Os string
|
|
Arch string
|
|
}
|
|
|
|
type ImageIdentifier struct {
|
|
Repo string `json:"repo"`
|
|
Tag string `json:"tag"`
|
|
Digest string `json:"digest"`
|
|
Platform *osArch `json:"platform"`
|
|
}
|
|
|
|
func getImageIdentifier(repoTagStr, platformStr string) ImageIdentifier {
|
|
var tag, digest string
|
|
|
|
repo, ref, isTag, err := zcommon.GetRepoReference(repoTagStr)
|
|
if err != nil {
|
|
return ImageIdentifier{}
|
|
}
|
|
|
|
if isTag {
|
|
tag = ref
|
|
} else {
|
|
digest = ref
|
|
}
|
|
|
|
// check if the following input is a repo:tag or repo@digest, if not then it's a platform
|
|
var platform *osArch
|
|
|
|
if platformStr != "" {
|
|
os, arch, _ := strings.Cut(platformStr, "/")
|
|
platform = &osArch{Os: os, Arch: arch}
|
|
}
|
|
|
|
return ImageIdentifier{
|
|
Repo: repo,
|
|
Tag: tag,
|
|
Digest: digest,
|
|
Platform: platform,
|
|
}
|
|
}
|