2016-02-01 12:59:10 +01:00
|
|
|
/*
|
|
|
|
Copyright (c) 2016 VMware, Inc. All Rights Reserved.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
2016-02-26 11:54:14 +01:00
|
|
|
|
2016-05-25 09:24:44 +02:00
|
|
|
package cache
|
2016-02-01 12:59:10 +01:00
|
|
|
|
|
|
|
import (
|
2016-04-15 07:17:32 +02:00
|
|
|
"os"
|
2016-02-01 12:59:10 +01:00
|
|
|
"time"
|
|
|
|
|
2016-03-25 02:31:50 +01:00
|
|
|
"github.com/vmware/harbor/utils/log"
|
2016-04-15 07:17:32 +02:00
|
|
|
"github.com/vmware/harbor/utils/registry"
|
2016-06-21 10:39:03 +02:00
|
|
|
"github.com/vmware/harbor/utils/registry/auth"
|
2016-02-01 12:59:10 +01:00
|
|
|
|
|
|
|
"github.com/astaxie/beego/cache"
|
|
|
|
)
|
|
|
|
|
2016-04-27 11:59:43 +02:00
|
|
|
var (
|
|
|
|
// Cache is the global cache in system.
|
2016-06-23 09:18:32 +02:00
|
|
|
Cache cache.Cache
|
|
|
|
endpoint string
|
|
|
|
username string
|
2016-04-27 11:59:43 +02:00
|
|
|
)
|
2016-04-15 07:17:32 +02:00
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
const catalogKey string = "catalog"
|
2016-02-01 12:59:10 +01:00
|
|
|
|
|
|
|
func init() {
|
|
|
|
var err error
|
|
|
|
Cache, err = cache.NewCache("memory", `{"interval":720}`)
|
|
|
|
if err != nil {
|
2016-03-26 17:18:11 +01:00
|
|
|
log.Errorf("Failed to initialize cache, error:%v", err)
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
2016-04-15 07:17:32 +02:00
|
|
|
|
2016-04-27 11:59:43 +02:00
|
|
|
endpoint = os.Getenv("REGISTRY_URL")
|
2016-06-23 09:18:32 +02:00
|
|
|
username = "admin"
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// RefreshCatalogCache calls registry's API to get repository list and write it to cache.
|
2016-02-01 12:59:10 +01:00
|
|
|
func RefreshCatalogCache() error {
|
2016-04-15 07:17:32 +02:00
|
|
|
log.Debug("refreshing catalog cache...")
|
2016-04-27 11:59:43 +02:00
|
|
|
|
2016-06-23 09:18:32 +02:00
|
|
|
registryClient, err := NewRegistryClient(endpoint, true, username,
|
|
|
|
"registry", "catalog", "*")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2016-04-27 11:59:43 +02:00
|
|
|
}
|
|
|
|
|
2016-04-17 16:39:10 +02:00
|
|
|
rs, err := registryClient.Catalog()
|
2016-02-01 12:59:10 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-04-15 07:17:32 +02:00
|
|
|
|
|
|
|
repos := []string{}
|
|
|
|
|
2016-04-17 16:39:10 +02:00
|
|
|
for _, repo := range rs {
|
2016-06-23 09:18:32 +02:00
|
|
|
rc, err := NewRepositoryClient(endpoint, true, username,
|
2016-06-21 10:39:03 +02:00
|
|
|
repo, "repository", repo, "pull", "push", "*")
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("error occurred while initializing repository client used by cache: %s %v", repo, err)
|
|
|
|
continue
|
2016-04-27 11:59:43 +02:00
|
|
|
}
|
2016-06-21 10:39:03 +02:00
|
|
|
|
2016-04-27 11:59:43 +02:00
|
|
|
tags, err := rc.ListTag()
|
2016-04-15 07:17:32 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Errorf("error occurred while list tag for %s: %v", repo, err)
|
2016-04-29 10:59:54 +02:00
|
|
|
continue
|
2016-04-15 07:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(tags) != 0 {
|
|
|
|
repos = append(repos, repo)
|
|
|
|
log.Debugf("add %s to catalog cache", repo)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Cache.Put(catalogKey, repos, 600*time.Second)
|
2016-02-01 12:59:10 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// GetRepoFromCache get repository list from cache, it refreshes the cache if it's empty.
|
2016-02-01 12:59:10 +01:00
|
|
|
func GetRepoFromCache() ([]string, error) {
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
result := Cache.Get(catalogKey)
|
2016-02-01 12:59:10 +01:00
|
|
|
if result == nil {
|
|
|
|
err := RefreshCatalogCache()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-02-26 11:35:55 +01:00
|
|
|
cached := Cache.Get(catalogKey)
|
2016-02-01 12:59:10 +01:00
|
|
|
if cached != nil {
|
|
|
|
return cached.([]string), nil
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return result.([]string), nil
|
|
|
|
}
|
2016-06-21 10:39:03 +02:00
|
|
|
|
|
|
|
// NewRegistryClient ...
|
|
|
|
func NewRegistryClient(endpoint string, insecure bool, username, scopeType, scopeName string,
|
|
|
|
scopeActions ...string) (*registry.Registry, error) {
|
|
|
|
authorizer := auth.NewUsernameTokenAuthorizer(username, scopeType, scopeName, scopeActions...)
|
|
|
|
|
2016-06-22 06:03:50 +02:00
|
|
|
store, err := auth.NewAuthorizerStore(endpoint, insecure, authorizer)
|
2016-06-21 10:39:03 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
client, err := registry.NewRegistryWithModifiers(endpoint, insecure, store)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return client, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewRepositoryClient ...
|
|
|
|
func NewRepositoryClient(endpoint string, insecure bool, username, repository, scopeType, scopeName string,
|
|
|
|
scopeActions ...string) (*registry.Repository, error) {
|
|
|
|
|
|
|
|
authorizer := auth.NewUsernameTokenAuthorizer(username, scopeType, scopeName, scopeActions...)
|
|
|
|
|
2016-06-22 06:03:50 +02:00
|
|
|
store, err := auth.NewAuthorizerStore(endpoint, insecure, authorizer)
|
2016-06-21 10:39:03 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
client, err := registry.NewRepositoryWithModifiers(repository, endpoint, insecure, store)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return client, nil
|
|
|
|
}
|