2023-01-24 18:03:10 -05:00
|
|
|
package cveinfo
|
|
|
|
|
|
|
|
import (
|
2023-03-14 05:02:19 -05:00
|
|
|
"fmt"
|
2023-01-24 18:03:10 -05:00
|
|
|
"sort"
|
|
|
|
|
|
|
|
zerr "zotregistry.io/zot/errors"
|
2023-05-25 13:27:49 -05:00
|
|
|
"zotregistry.io/zot/pkg/common"
|
2023-01-24 18:03:10 -05:00
|
|
|
cvemodel "zotregistry.io/zot/pkg/extensions/search/cve/model"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2023-07-06 03:36:26 -05:00
|
|
|
AlphabeticAsc = cvemodel.SortCriteria("ALPHABETIC_ASC")
|
|
|
|
AlphabeticDsc = cvemodel.SortCriteria("ALPHABETIC_DSC")
|
|
|
|
SeverityDsc = cvemodel.SortCriteria("SEVERITY")
|
2023-01-24 18:03:10 -05:00
|
|
|
)
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
func SortFunctions() map[cvemodel.SortCriteria]func(pageBuffer []cvemodel.CVE) func(i, j int) bool {
|
|
|
|
return map[cvemodel.SortCriteria]func(pageBuffer []cvemodel.CVE) func(i, j int) bool{
|
2023-01-24 18:03:10 -05:00
|
|
|
AlphabeticAsc: SortByAlphabeticAsc,
|
|
|
|
AlphabeticDsc: SortByAlphabeticDsc,
|
|
|
|
SeverityDsc: SortBySeverity,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
func SortByAlphabeticAsc(pageBuffer []cvemodel.CVE) func(i, j int) bool {
|
2023-01-24 18:03:10 -05:00
|
|
|
return func(i, j int) bool {
|
|
|
|
return pageBuffer[i].ID < pageBuffer[j].ID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
func SortByAlphabeticDsc(pageBuffer []cvemodel.CVE) func(i, j int) bool {
|
2023-01-24 18:03:10 -05:00
|
|
|
return func(i, j int) bool {
|
|
|
|
return pageBuffer[i].ID > pageBuffer[j].ID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
func SortBySeverity(pageBuffer []cvemodel.CVE) func(i, j int) bool {
|
2023-01-24 18:03:10 -05:00
|
|
|
return func(i, j int) bool {
|
2023-09-17 17:12:20 -05:00
|
|
|
if cvemodel.CompareSeverities(pageBuffer[i].Severity, pageBuffer[j].Severity) == 0 {
|
2023-09-14 12:51:17 -05:00
|
|
|
return pageBuffer[i].ID < pageBuffer[j].ID
|
|
|
|
}
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
return cvemodel.CompareSeverities(pageBuffer[i].Severity, pageBuffer[j].Severity) < 0
|
2023-01-24 18:03:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// PageFinder permits keeping a pool of objects using Add
|
|
|
|
// and returning a specific page.
|
|
|
|
type PageFinder interface {
|
|
|
|
Add(cve cvemodel.CVE)
|
2023-05-25 13:27:49 -05:00
|
|
|
Page() ([]cvemodel.CVE, common.PageInfo)
|
2023-01-24 18:03:10 -05:00
|
|
|
Reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
// CvePageFinder implements PageFinder. It manages Cve objects and calculates the page
|
|
|
|
// using the given limit, offset and sortBy option.
|
|
|
|
type CvePageFinder struct {
|
|
|
|
limit int
|
|
|
|
offset int
|
2023-07-06 03:36:26 -05:00
|
|
|
sortBy cvemodel.SortCriteria
|
2023-01-24 18:03:10 -05:00
|
|
|
pageBuffer []cvemodel.CVE
|
|
|
|
}
|
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
func NewCvePageFinder(limit, offset int, sortBy cvemodel.SortCriteria) (*CvePageFinder, error) {
|
2023-01-24 18:03:10 -05:00
|
|
|
if sortBy == "" {
|
|
|
|
sortBy = SeverityDsc
|
|
|
|
}
|
|
|
|
|
|
|
|
if limit < 0 {
|
|
|
|
return nil, zerr.ErrLimitIsNegative
|
|
|
|
}
|
|
|
|
|
|
|
|
if offset < 0 {
|
|
|
|
return nil, zerr.ErrOffsetIsNegative
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, found := SortFunctions()[sortBy]; !found {
|
2023-03-14 05:02:19 -05:00
|
|
|
return nil, fmt.Errorf("sorting CVEs by '%s' is not supported %w", sortBy, zerr.ErrSortCriteriaNotSupported)
|
2023-01-24 18:03:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return &CvePageFinder{
|
|
|
|
limit: limit,
|
|
|
|
offset: offset,
|
|
|
|
sortBy: sortBy,
|
|
|
|
pageBuffer: make([]cvemodel.CVE, 0, limit),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpt *CvePageFinder) Reset() {
|
|
|
|
bpt.pageBuffer = []cvemodel.CVE{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpt *CvePageFinder) Add(cve cvemodel.CVE) {
|
|
|
|
bpt.pageBuffer = append(bpt.pageBuffer, cve)
|
|
|
|
}
|
|
|
|
|
2023-05-25 13:27:49 -05:00
|
|
|
func (bpt *CvePageFinder) Page() ([]cvemodel.CVE, common.PageInfo) {
|
2023-01-24 18:03:10 -05:00
|
|
|
if len(bpt.pageBuffer) == 0 {
|
2023-05-25 13:27:49 -05:00
|
|
|
return []cvemodel.CVE{}, common.PageInfo{}
|
2023-01-24 18:03:10 -05:00
|
|
|
}
|
|
|
|
|
2023-05-25 13:27:49 -05:00
|
|
|
pageInfo := &common.PageInfo{}
|
2023-01-24 18:03:10 -05:00
|
|
|
|
2023-09-17 17:12:20 -05:00
|
|
|
sort.Slice(bpt.pageBuffer, SortFunctions()[bpt.sortBy](bpt.pageBuffer))
|
2023-01-24 18:03:10 -05:00
|
|
|
|
|
|
|
// the offset and limit are calculated in terms of CVEs counted
|
|
|
|
start := bpt.offset
|
|
|
|
end := bpt.offset + bpt.limit
|
|
|
|
|
|
|
|
// we'll return an empty array when the offset is greater than the number of elements
|
|
|
|
if start >= len(bpt.pageBuffer) {
|
|
|
|
start = len(bpt.pageBuffer)
|
|
|
|
end = start
|
|
|
|
}
|
|
|
|
|
|
|
|
if end >= len(bpt.pageBuffer) {
|
|
|
|
end = len(bpt.pageBuffer)
|
|
|
|
}
|
|
|
|
|
|
|
|
cves := bpt.pageBuffer[start:end]
|
|
|
|
|
|
|
|
pageInfo.ItemCount = len(cves)
|
|
|
|
|
|
|
|
if start == 0 && end == 0 {
|
|
|
|
cves = bpt.pageBuffer
|
|
|
|
pageInfo.ItemCount = len(cves)
|
|
|
|
}
|
|
|
|
|
|
|
|
pageInfo.TotalCount = len(bpt.pageBuffer)
|
|
|
|
|
|
|
|
return cves, *pageInfo
|
|
|
|
}
|