Suport filtering registries by type in listing registry API

Suport filtering registries by type in listing registry API

Signed-off-by: Wenkai Yin <yinw@vmware.com>
This commit is contained in:
Wenkai Yin 2020-07-02 17:13:44 +08:00
parent 264bd02892
commit 02690d1d04
13 changed files with 104 additions and 121 deletions

View File

@ -1458,13 +1458,14 @@ paths:
get: get:
summary: List registries. summary: List registries.
description: | description: |
This endpoint let user list filtered registries by name, if name is nil, list returns all registries. List registries according to the query.
parameters: parameters:
- name: name - name: name
in: query in: query
type: string type: string
required: false required: false
description: Registry's name. description: Deprecated, use `q` instead.
- $ref: '#/parameters/query'
tags: tags:
- Products - Products
responses: responses:
@ -5668,3 +5669,10 @@ definitions:
type: string type: string
description: Webhook supportted notify type. description: Webhook supportted notify type.
example: 'http' example: 'http'
parameters:
query:
name: q
description: Query string to query resources. Supported query patterns are "exact match(k=v)", "fuzzy match(k=~v)", "range(k=[min~max])", "list with union releationship(k={v1 v2 v3})" and "list with intersetion relationship(k=(v1 v2 v3))". The value of range and list can be string(enclosed by " or '), integer or time(in format "2020-04-09 02:36:00"). All of these query patterns should be put in the query string "q=xxx" and splitted by ",". e.g. q=k1=v1,k2=~v2,k3=[min~max]
in: query
type: string
required: false

View File

@ -47,6 +47,7 @@ WITH_CHARTMUSEUM={{with_chartmuseum}}
REGISTRY_CREDENTIAL_USERNAME={{registry_username}} REGISTRY_CREDENTIAL_USERNAME={{registry_username}}
REGISTRY_CREDENTIAL_PASSWORD={{registry_password}} REGISTRY_CREDENTIAL_PASSWORD={{registry_password}}
CSRF_KEY={{csrf_key}} CSRF_KEY={{csrf_key}}
PERMITTED_REGISTRY_TYPES_FOR_PROXY_CACHE=docker-hub,harbor
HTTP_PROXY={{core_http_proxy}} HTTP_PROXY={{core_http_proxy}}
HTTPS_PROXY={{core_https_proxy}} HTTPS_PROXY={{core_https_proxy}}

View File

@ -5,6 +5,7 @@ import (
"github.com/goharbor/harbor/src/controller/event" "github.com/goharbor/harbor/src/controller/event"
"github.com/goharbor/harbor/src/core/config" "github.com/goharbor/harbor/src/core/config"
"github.com/goharbor/harbor/src/core/promgr/metamgr" "github.com/goharbor/harbor/src/core/promgr/metamgr"
"github.com/goharbor/harbor/src/lib/q"
"github.com/goharbor/harbor/src/pkg/notification" "github.com/goharbor/harbor/src/pkg/notification"
"github.com/goharbor/harbor/src/replication" "github.com/goharbor/harbor/src/replication"
daoModels "github.com/goharbor/harbor/src/replication/dao/models" daoModels "github.com/goharbor/harbor/src/replication/dao/models"
@ -158,7 +159,7 @@ func (f *fakedReplicationRegistryMgr) Add(*model.Registry) (int64, error) {
} }
// List registries, returns total count, registry list and error // List registries, returns total count, registry list and error
func (f *fakedReplicationRegistryMgr) List(...*model.RegistryQuery) (int64, []*model.Registry, error) { func (f *fakedReplicationRegistryMgr) List(query *q.Query) (int64, []*model.Registry, error) {
return 0, nil, nil return 0, nil, nil
} }

View File

@ -143,6 +143,17 @@ func (p *ProjectAPI) Post() {
p.SendNotFoundError(fmt.Errorf("registry %d not found", pro.RegistryID)) p.SendNotFoundError(fmt.Errorf("registry %d not found", pro.RegistryID))
return return
} }
permitted := false
for _, t := range config.GetPermittedRegistryTypesForProxyCache() {
if string(registry.Type) == t {
permitted = true
break
}
}
if !permitted {
p.SendBadRequestError(fmt.Errorf("unsupported registry type %s", string(registry.Type)))
return
}
} }
var hardLimits types.ResourceList var hardLimits types.ResourceList

View File

@ -11,6 +11,7 @@ import (
"github.com/goharbor/harbor/src/common/utils" "github.com/goharbor/harbor/src/common/utils"
"github.com/goharbor/harbor/src/core/api/models" "github.com/goharbor/harbor/src/core/api/models"
"github.com/goharbor/harbor/src/lib/log" "github.com/goharbor/harbor/src/lib/log"
"github.com/goharbor/harbor/src/lib/q"
"github.com/goharbor/harbor/src/replication" "github.com/goharbor/harbor/src/replication"
"github.com/goharbor/harbor/src/replication/adapter" "github.com/goharbor/harbor/src/replication/adapter"
"github.com/goharbor/harbor/src/replication/event" "github.com/goharbor/harbor/src/replication/event"
@ -165,15 +166,24 @@ func hideAccessSecret(credential *model.Credential) {
credential.AccessSecret = "*****" credential.AccessSecret = "*****"
} }
// List lists all registries that match a given registry name. // List lists all registries
func (t *RegistryAPI) List() { func (t *RegistryAPI) List() {
queryStr := t.GetString("q")
// keep backward compatibility for the "name" query
if len(queryStr) == 0 {
name := t.GetString("name") name := t.GetString("name")
if len(name) > 0 {
_, registries, err := t.manager.List(&model.RegistryQuery{ queryStr = fmt.Sprintf("name=~%s", name)
Name: name, }
}) }
query, err := q.Build(queryStr, 0, 0)
if err != nil {
t.SendError(err)
return
}
_, registries, err := t.manager.List(query)
if err != nil { if err != nil {
log.Errorf("failed to list registries %s: %v", name, err)
t.SendInternalServerError(err) t.SendInternalServerError(err)
return return
} }

View File

@ -15,6 +15,7 @@
package api package api
import ( import (
"github.com/goharbor/harbor/src/lib/q"
"net/http" "net/http"
"testing" "testing"
@ -29,7 +30,7 @@ type fakedRegistryManager struct{}
func (f *fakedRegistryManager) Add(*model.Registry) (int64, error) { func (f *fakedRegistryManager) Add(*model.Registry) (int64, error) {
return 0, nil return 0, nil
} }
func (f *fakedRegistryManager) List(...*model.RegistryQuery) (int64, []*model.Registry, error) { func (f *fakedRegistryManager) List(query *q.Query) (int64, []*model.Registry, error) {
return 0, nil, nil return 0, nil, nil
} }
func (f *fakedRegistryManager) Get(id int64) (*model.Registry, error) { func (f *fakedRegistryManager) Get(id int64) (*model.Registry, error) {

View File

@ -480,3 +480,12 @@ func QuotaSetting() (*models.QuotaSetting, error) {
StoragePerProject: cfgMgr.Get(common.StoragePerProject).GetInt64(), StoragePerProject: cfgMgr.Get(common.StoragePerProject).GetInt64(),
}, nil }, nil
} }
// GetPermittedRegistryTypesForProxyCache returns the permitted registry types for proxy cache
func GetPermittedRegistryTypesForProxyCache() []string {
types := os.Getenv("PERMITTED_REGISTRY_TYPES_FOR_PROXY_CACHE")
if len(types) == 0 {
return []string{}
}
return strings.Split(types, ",")
}

View File

@ -57,6 +57,7 @@ require (
github.com/opencontainers/go-digest v1.0.0-rc1 github.com/opencontainers/go-digest v1.0.0-rc1
github.com/opencontainers/image-spec v1.0.1 github.com/opencontainers/image-spec v1.0.1
github.com/opentracing/opentracing-go v1.1.0 // indirect github.com/opentracing/opentracing-go v1.1.0 // indirect
github.com/pkg/errors v0.9.1
github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35 // indirect github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35 // indirect
github.com/robfig/cron v1.0.0 github.com/robfig/cron v1.0.0
github.com/shiena/ansicolor v0.0.0-20151119151921-a422bbe96644 // indirect github.com/shiena/ansicolor v0.0.0-20151119151921-a422bbe96644 // indirect

View File

@ -1,21 +1,15 @@
package dao package dao
import ( import (
"context"
"github.com/astaxie/beego/orm" "github.com/astaxie/beego/orm"
"github.com/goharbor/harbor/src/common/dao" "github.com/goharbor/harbor/src/common/dao"
liborm "github.com/goharbor/harbor/src/lib/orm"
"github.com/goharbor/harbor/src/lib/q"
"github.com/goharbor/harbor/src/replication/dao/models" "github.com/goharbor/harbor/src/replication/dao/models"
) )
// ListRegistryQuery defines the query conditions to list registry.
type ListRegistryQuery struct {
// Query is name query
Query string
// Offset specifies the offset in the registry list to return
Offset int64
// Limit specifies the maximum registries to return
Limit int64
}
// AddRegistry add a new registry // AddRegistry add a new registry
func AddRegistry(registry *models.Registry) (int64, error) { func AddRegistry(registry *models.Registry) (int64, error) {
o := dao.GetOrmer() o := dao.GetOrmer()
@ -55,34 +49,37 @@ func GetRegistryByURL(url string) (*models.Registry, error) {
return &r, err return &r, err
} }
// ListRegistries lists registries. Registries returned are sorted by creation time. // ListRegistries lists registries
// - query: query to the registry name, name query and pagination are defined. func ListRegistries(ctx context.Context, query *q.Query) (int64, []*models.Registry, error) {
func ListRegistries(query ...*ListRegistryQuery) (int64, []*models.Registry, error) { var countQuery *q.Query
o := dao.GetOrmer() if query != nil {
// ignore the page number and size
q := o.QueryTable(&models.Registry{}) countQuery = &q.Query{
if len(query) > 0 && len(query[0].Query) > 0 { Keywords: query.Keywords,
q = q.Filter("name__contains", query[0].Query)
} }
}
total, err := q.Count() countQs, err := liborm.QuerySetter(ctx, &models.Registry{}, countQuery)
if err != nil { if err != nil {
return -1, nil, err return 0, nil, err
}
count, err := countQs.Count()
if err != nil {
return 0, nil, err
} }
// limit being -1 means no pagination specified. qs, err := liborm.QuerySetter(ctx, &models.Registry{}, query)
if len(query) > 0 && query[0].Limit != -1 { if err != nil {
q = q.Offset(query[0].Offset).Limit(query[0].Limit) return 0, nil, err
} }
var registries []*models.Registry var registries []*models.Registry
_, err = q.All(&registries) _, err = qs.All(&registries)
if err != nil { if err != nil {
return total, nil, err return 0, nil, err
} }
if registries == nil { if registries == nil {
registries = []*models.Registry{} registries = []*models.Registry{}
} }
return total, registries, nil return count, registries, nil
} }
// UpdateRegistry updates one registry // UpdateRegistry updates one registry

View File

@ -3,6 +3,8 @@ package dao
import ( import (
"testing" "testing"
"github.com/goharbor/harbor/src/lib/orm"
"github.com/goharbor/harbor/src/lib/q"
"github.com/goharbor/harbor/src/replication/dao/models" "github.com/goharbor/harbor/src/replication/dao/models"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"github.com/stretchr/testify/suite" "github.com/stretchr/testify/suite"
@ -93,62 +95,25 @@ func (suite *RegistrySuite) TestGetRegistryByURL() {
} }
func (suite *RegistrySuite) TestListRegistries() { func (suite *RegistrySuite) TestListRegistries() {
assert := assert.New(suite.T()) ctx := orm.Context()
// Insert on more registry // nil query
id, err := AddRegistry(testRegistry1) count, registries, err := ListRegistries(ctx, nil)
assert.Nil(err) suite.Require().Nil(err)
assert.NotEqual(0, id) if count < 1 {
suite.T().Errorf("At least %d should be found in total, but got %d", 1, count)
// List all registries, should succeed
total, registries, err := ListRegistries()
assert.Nil(err)
if total < 2 {
suite.T().Errorf("At least %d should be found in total, but got %d", 2, total)
} }
// List default registry by normal query, should succeed // query by name
total, registries, err = ListRegistries(&ListRegistryQuery{ count, registries, err = ListRegistries(ctx, &q.Query{
Query: "Default", Keywords: map[string]interface{}{
Offset: 0, "Name": "daoTestDefault",
Limit: 10, },
}) })
assert.Nil(err) suite.Require().Nil(err)
assert.Equal(int64(1), total) suite.Require().Equal(int64(1), count)
assert.Equal(defaultRegistry.Name, registries[0].Name) suite.Assert().Equal("daoTestDefault", registries[0].Name)
// List registry and limit to 1, should return one
total, registries, err = ListRegistries(&ListRegistryQuery{
Query: "dao",
Offset: 0,
Limit: 1,
})
assert.Nil(err)
assert.Equal(int64(2), total)
assert.Equal(1, len(registries))
// List registry and limit set to -1, should return all
total, registries, err = ListRegistries(&ListRegistryQuery{
Limit: -1,
})
assert.Nil(err)
if total < 2 {
suite.T().Errorf("At least %d should be found in total, but got %d", 2, total)
}
if len(registries) < 2 {
suite.T().Errorf("At least %d should be returned, but got %d", 2, len(registries))
}
// List registry and large offset, should return empty
total, registries, err = ListRegistries(&ListRegistryQuery{
Offset: 10,
Limit: 1,
})
assert.Nil(err)
if total < 2 {
suite.T().Errorf("At least %d should be found in total, but got %d", 2, total)
}
assert.Equal(0, len(registries))
} }
func (suite *RegistrySuite) TestUpdate() { func (suite *RegistrySuite) TestUpdate() {

View File

@ -15,6 +15,7 @@
package event package event
import ( import (
"github.com/goharbor/harbor/src/lib/q"
"testing" "testing"
"github.com/goharbor/harbor/src/replication/config" "github.com/goharbor/harbor/src/replication/config"
@ -184,7 +185,7 @@ type fakedRegistryManager struct{}
func (f *fakedRegistryManager) Add(*model.Registry) (int64, error) { func (f *fakedRegistryManager) Add(*model.Registry) (int64, error) {
return 0, nil return 0, nil
} }
func (f *fakedRegistryManager) List(...*model.RegistryQuery) (int64, []*model.Registry, error) { func (f *fakedRegistryManager) List(query *q.Query) (int64, []*model.Registry, error) {
return 0, nil, nil return 0, nil, nil
} }
func (f *fakedRegistryManager) Get(id int64) (*model.Registry, error) { func (f *fakedRegistryManager) Get(id int64) (*model.Registry, error) {

View File

@ -16,8 +16,6 @@ package model
import ( import (
"time" "time"
"github.com/goharbor/harbor/src/common/models"
) )
// const definition // const definition
@ -101,14 +99,6 @@ type Registry struct {
UpdateTime time.Time `json:"update_time"` UpdateTime time.Time `json:"update_time"`
} }
// RegistryQuery defines the query conditions for listing registries
type RegistryQuery struct {
// Name is name of the registry to query
Name string
// Pagination specifies the pagination
Pagination *models.Pagination
}
// FilterStyle ... // FilterStyle ...
type FilterStyle struct { type FilterStyle struct {
Type FilterType `json:"type"` Type FilterType `json:"type"`

View File

@ -19,6 +19,8 @@ import (
"github.com/goharbor/harbor/src/common/utils" "github.com/goharbor/harbor/src/common/utils"
"github.com/goharbor/harbor/src/lib/log" "github.com/goharbor/harbor/src/lib/log"
"github.com/goharbor/harbor/src/lib/orm"
"github.com/goharbor/harbor/src/lib/q"
"github.com/goharbor/harbor/src/replication/adapter" "github.com/goharbor/harbor/src/replication/adapter"
"github.com/goharbor/harbor/src/replication/config" "github.com/goharbor/harbor/src/replication/config"
"github.com/goharbor/harbor/src/replication/dao" "github.com/goharbor/harbor/src/replication/dao"
@ -31,7 +33,7 @@ type Manager interface {
// Add new registry // Add new registry
Add(*model.Registry) (int64, error) Add(*model.Registry) (int64, error)
// List registries, returns total count, registry list and error // List registries, returns total count, registry list and error
List(...*model.RegistryQuery) (int64, []*model.Registry, error) List(*q.Query) (int64, []*model.Registry, error)
// Get the specified registry // Get the specified registry
Get(int64) (*model.Registry, error) Get(int64) (*model.Registry, error)
// GetByName gets registry by name // GetByName gets registry by name
@ -85,36 +87,22 @@ func (m *DefaultManager) GetByName(name string) (*model.Registry, error) {
} }
// List lists registries according to query provided. // List lists registries according to query provided.
func (m *DefaultManager) List(query ...*model.RegistryQuery) (int64, []*model.Registry, error) { func (m *DefaultManager) List(query *q.Query) (int64, []*model.Registry, error) {
var registryQueries []*dao.ListRegistryQuery count, registries, err := dao.ListRegistries(orm.Context(), query)
if len(query) > 0 {
// limit being -1 indicates no pagination specified, result in all registries matching name returned.
listQuery := &dao.ListRegistryQuery{
Query: query[0].Name,
Limit: -1,
}
if query[0].Pagination != nil {
listQuery.Offset = query[0].Pagination.Page * query[0].Pagination.Size
listQuery.Limit = query[0].Pagination.Size
}
registryQueries = append(registryQueries, listQuery)
}
total, registries, err := dao.ListRegistries(registryQueries...)
if err != nil { if err != nil {
return -1, nil, err return 0, nil, err
} }
var results []*model.Registry var results []*model.Registry
for _, r := range registries { for _, r := range registries {
registry, err := fromDaoModel(r) registry, err := fromDaoModel(r)
if err != nil { if err != nil {
return -1, nil, err return 0, nil, err
} }
results = append(results, registry) results = append(results, registry)
} }
return total, results, nil return count, results, nil
} }
// Add adds a new registry // Add adds a new registry
@ -158,7 +146,7 @@ func (m *DefaultManager) Remove(id int64) error {
// HealthCheck checks health status of every registries and update their status. It will check whether a registry // HealthCheck checks health status of every registries and update their status. It will check whether a registry
// is reachable and the credential is valid // is reachable and the credential is valid
func (m *DefaultManager) HealthCheck() error { func (m *DefaultManager) HealthCheck() error {
_, registries, err := m.List() _, registries, err := m.List(nil)
if err != nil { if err != nil {
return err return err
} }