2018-07-02 13:34:04 +02:00
|
|
|
package chartserver
|
|
|
|
|
|
|
|
import (
|
2018-07-05 12:07:00 +02:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2018-07-02 13:34:04 +02:00
|
|
|
"net/http"
|
2018-07-05 12:07:00 +02:00
|
|
|
"net/url"
|
2018-07-14 09:49:38 +02:00
|
|
|
"path"
|
2018-07-05 12:07:00 +02:00
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ghodss/yaml"
|
2018-09-12 08:38:29 +02:00
|
|
|
"github.com/goharbor/harbor/src/core/filter"
|
2018-09-05 14:13:27 +02:00
|
|
|
"k8s.io/helm/cmd/helm/search"
|
2018-07-05 12:07:00 +02:00
|
|
|
helm_repo "k8s.io/helm/pkg/repo"
|
2018-07-14 09:49:38 +02:00
|
|
|
|
2018-08-23 09:02:20 +02:00
|
|
|
hlog "github.com/goharbor/harbor/src/common/utils/log"
|
2018-07-05 12:07:00 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
maxWorkers = 10
|
2018-09-05 14:13:27 +02:00
|
|
|
|
|
|
|
//Keep consistent with 'helm search' command
|
|
|
|
searchMaxScore = 25
|
2018-07-02 13:34:04 +02:00
|
|
|
)
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// RepositoryHandler defines all the handlers to handle the requests related with chart repository
|
|
|
|
// e.g: index.yaml and downloading chart objects
|
2018-07-02 13:34:04 +02:00
|
|
|
type RepositoryHandler struct {
|
2018-09-05 10:16:31 +02:00
|
|
|
// Proxy used to to transfer the traffic of requests
|
|
|
|
// It's mainly used to talk to the backend chart server
|
2018-07-03 15:36:20 +02:00
|
|
|
trafficProxy *ProxyEngine
|
2018-07-05 12:07:00 +02:00
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// HTTP client used to call the realted APIs of the backend chart repositories
|
2018-07-06 14:53:13 +02:00
|
|
|
apiClient *ChartClient
|
2018-07-05 12:07:00 +02:00
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Point to the url of the backend server
|
2018-07-05 12:07:00 +02:00
|
|
|
backendServerAddress *url.URL
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Result returned by worker
|
2018-07-05 12:07:00 +02:00
|
|
|
type processedResult struct {
|
|
|
|
namespace string
|
|
|
|
indexFileOfRepo *helm_repo.IndexFile
|
2018-07-02 13:34:04 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// GetIndexFileWithNS will read the index.yaml data under the specified namespace
|
2018-07-02 13:34:04 +02:00
|
|
|
func (rh *RepositoryHandler) GetIndexFileWithNS(w http.ResponseWriter, req *http.Request) {
|
2018-07-03 15:36:20 +02:00
|
|
|
rh.trafficProxy.ServeHTTP(w, req)
|
2018-07-02 13:34:04 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// GetIndexFile will read the index.yaml under all namespaces and merge them as a single one
|
|
|
|
// Please be aware that, to support this function, the backend chart repository server should
|
|
|
|
// enable multi-tenancies
|
2018-07-03 15:36:20 +02:00
|
|
|
func (rh *RepositoryHandler) GetIndexFile(w http.ResponseWriter, req *http.Request) {
|
2018-09-05 10:16:31 +02:00
|
|
|
// Get project manager references
|
2018-07-05 12:07:00 +02:00
|
|
|
projectMgr, err := filter.GetProjectManager(req)
|
|
|
|
if err != nil {
|
2018-07-23 11:41:40 +02:00
|
|
|
WriteInternalError(w, err)
|
2018-07-05 12:07:00 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Get all the projects
|
2018-07-05 12:07:00 +02:00
|
|
|
results, err := projectMgr.List(nil)
|
|
|
|
if err != nil {
|
2018-07-23 11:41:40 +02:00
|
|
|
WriteInternalError(w, err)
|
2018-07-05 12:07:00 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// If no projects existing, return empty index.yaml content immediately
|
2018-07-05 12:07:00 +02:00
|
|
|
if results.Total == 0 {
|
|
|
|
w.Write(emptyIndexFile())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-09-05 14:13:27 +02:00
|
|
|
namespaces := []string{}
|
|
|
|
for _, p := range results.Projects {
|
|
|
|
namespaces = append(namespaces, p.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
mergedIndexFile, err := rh.getIndexYaml(namespaces)
|
|
|
|
if err != nil {
|
|
|
|
WriteInternalError(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes, err := yaml.Marshal(mergedIndexFile)
|
|
|
|
if err != nil {
|
|
|
|
WriteInternalError(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Write(bytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadChartObject will download the stored chart object to the client
|
|
|
|
// e.g: helm install
|
|
|
|
func (rh *RepositoryHandler) DownloadChartObject(w http.ResponseWriter, req *http.Request) {
|
|
|
|
rh.trafficProxy.ServeHTTP(w, req)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SearchChart search charts in the specified namespaces with the keyword q.
|
|
|
|
// RegExp mode is enabled as default.
|
|
|
|
// For each chart, only the latest version will shown in the result list if matched to avoid duplicated entries.
|
|
|
|
// Keep consistent with `helm search` command.
|
|
|
|
func (rh *RepositoryHandler) SearchChart(q string, namespaces []string) ([]*search.Result, error) {
|
|
|
|
if len(q) == 0 || len(namespaces) == 0 {
|
|
|
|
// Return empty list
|
|
|
|
return []*search.Result{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the merged index yaml file of the namespaces
|
|
|
|
ind, err := rh.getIndexYaml(namespaces)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the search index
|
|
|
|
index := search.NewIndex()
|
|
|
|
// As the repo name is already merged into the index yaml, we use empty repo name.
|
|
|
|
// Set 'All' to false to return only one version for each chart.
|
|
|
|
index.AddRepo("", ind, false)
|
|
|
|
|
|
|
|
// Search
|
|
|
|
// RegExp is enabled
|
|
|
|
results, err := index.Search(q, searchMaxScore, true)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort results.
|
|
|
|
search.SortScore(results)
|
|
|
|
|
|
|
|
return results, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// getIndexYaml will get the index yaml files for all the namespaces and merge them
|
|
|
|
// as one unified index yaml file.
|
|
|
|
func (rh *RepositoryHandler) getIndexYaml(namespaces []string) (*helm_repo.IndexFile, error) {
|
2018-09-05 10:16:31 +02:00
|
|
|
// The final merged index file
|
2018-07-05 12:07:00 +02:00
|
|
|
mergedIndexFile := &helm_repo.IndexFile{
|
|
|
|
APIVersion: "v1",
|
|
|
|
Entries: make(map[string]helm_repo.ChartVersions),
|
|
|
|
Generated: time.Now(),
|
|
|
|
PublicKeys: []string{},
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Sync the output results from the retriever
|
2018-07-05 12:07:00 +02:00
|
|
|
resultChan := make(chan *processedResult, 1)
|
2018-09-05 10:16:31 +02:00
|
|
|
// Receive error
|
2018-07-05 12:07:00 +02:00
|
|
|
errorChan := make(chan error, 1)
|
2018-09-05 10:16:31 +02:00
|
|
|
// Signal chan for merging work
|
2018-07-05 12:07:00 +02:00
|
|
|
mergeDone := make(chan struct{}, 1)
|
2018-09-05 10:16:31 +02:00
|
|
|
// Total projects/namespaces
|
2018-09-05 14:13:27 +02:00
|
|
|
total := len(namespaces)
|
2018-09-05 10:16:31 +02:00
|
|
|
// Initialize
|
2018-07-05 12:07:00 +02:00
|
|
|
initialItemCount := maxWorkers
|
|
|
|
if total < maxWorkers {
|
2018-09-05 14:13:27 +02:00
|
|
|
initialItemCount = total
|
2018-07-05 12:07:00 +02:00
|
|
|
}
|
2018-09-05 14:13:27 +02:00
|
|
|
// Retrieve index.yaml for repositories
|
|
|
|
workerPool := make(chan struct{}, initialItemCount)
|
|
|
|
|
|
|
|
// Add initial tokens to the pool
|
2018-07-05 12:07:00 +02:00
|
|
|
for i := 0; i < initialItemCount; i++ {
|
2018-09-05 14:13:27 +02:00
|
|
|
workerPool <- struct{}{}
|
2018-07-05 12:07:00 +02:00
|
|
|
}
|
2018-09-05 14:13:27 +02:00
|
|
|
// Track all the background threads
|
|
|
|
waitGroup := new(sync.WaitGroup)
|
2018-07-05 12:07:00 +02:00
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Start the index files merging thread
|
2018-07-05 12:07:00 +02:00
|
|
|
go func() {
|
|
|
|
defer func() {
|
|
|
|
mergeDone <- struct{}{}
|
|
|
|
}()
|
|
|
|
|
|
|
|
for res := range resultChan {
|
|
|
|
rh.mergeIndexFile(res.namespace, mergedIndexFile, res.indexFileOfRepo)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Retrieve the index files for the repositories
|
|
|
|
// and blocking here
|
2018-09-05 14:13:27 +02:00
|
|
|
var err error
|
2018-07-05 12:07:00 +02:00
|
|
|
LOOP:
|
2018-09-05 14:13:27 +02:00
|
|
|
for _, ns := range namespaces {
|
|
|
|
// Check if error has occurred in some goroutines
|
2018-07-05 12:07:00 +02:00
|
|
|
select {
|
2018-07-11 11:31:34 +02:00
|
|
|
case err = <-errorChan:
|
|
|
|
break LOOP
|
2018-09-05 14:13:27 +02:00
|
|
|
default:
|
|
|
|
// do nothing
|
2018-07-05 12:07:00 +02:00
|
|
|
}
|
2018-09-05 14:13:27 +02:00
|
|
|
|
|
|
|
// Apply one token before processing
|
|
|
|
<-workerPool
|
|
|
|
|
|
|
|
waitGroup.Add(1)
|
|
|
|
go func(ns string) {
|
|
|
|
defer func() {
|
|
|
|
waitGroup.Done() //done
|
|
|
|
//Return the worker back to the pool
|
|
|
|
workerPool <- struct{}{}
|
|
|
|
}()
|
|
|
|
|
|
|
|
indexFile, err := rh.getIndexYamlWithNS(ns)
|
|
|
|
if err != nil {
|
|
|
|
if len(errorChan) == 0 {
|
|
|
|
//Only need one error as failure signal
|
|
|
|
errorChan <- err
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Output
|
|
|
|
resultChan <- &processedResult{
|
|
|
|
namespace: ns,
|
|
|
|
indexFileOfRepo: indexFile,
|
|
|
|
}
|
|
|
|
}(ns)
|
2018-07-05 12:07:00 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Hold util all the retrieving work are done
|
2018-07-05 12:07:00 +02:00
|
|
|
waitGroup.Wait()
|
|
|
|
|
2018-09-05 14:13:27 +02:00
|
|
|
// close merge channel
|
2018-07-05 12:07:00 +02:00
|
|
|
close(resultChan)
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Wait until merging thread quit
|
2018-07-05 12:07:00 +02:00
|
|
|
<-mergeDone
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// All the threads are done
|
2018-09-05 14:13:27 +02:00
|
|
|
// Make sure error in the chan is read
|
|
|
|
if err == nil && len(errorChan) > 0 {
|
|
|
|
err = <-errorChan
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Met an error
|
2018-07-11 11:31:34 +02:00
|
|
|
if err != nil {
|
2018-09-05 14:13:27 +02:00
|
|
|
return nil, err
|
2018-07-11 11:31:34 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Remove duplicated keys in public key list
|
2018-07-05 12:07:00 +02:00
|
|
|
hash := make(map[string]string)
|
|
|
|
for _, key := range mergedIndexFile.PublicKeys {
|
|
|
|
hash[key] = key
|
|
|
|
}
|
|
|
|
mergedIndexFile.PublicKeys = []string{}
|
|
|
|
for k := range hash {
|
|
|
|
mergedIndexFile.PublicKeys = append(mergedIndexFile.PublicKeys, k)
|
|
|
|
}
|
|
|
|
|
2018-09-05 14:13:27 +02:00
|
|
|
return mergedIndexFile, nil
|
2018-07-03 15:36:20 +02:00
|
|
|
}
|
2018-07-05 12:07:00 +02:00
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Get the index yaml file under the specified namespace from the backend server
|
2018-07-05 12:07:00 +02:00
|
|
|
func (rh *RepositoryHandler) getIndexYamlWithNS(namespace string) (*helm_repo.IndexFile, error) {
|
2018-09-05 10:16:31 +02:00
|
|
|
// Join url path
|
2018-07-14 09:49:38 +02:00
|
|
|
url := path.Join(namespace, "index.yaml")
|
|
|
|
url = fmt.Sprintf("%s/%s", rh.backendServerAddress.String(), url)
|
|
|
|
hlog.Debugf("Getting index.yaml from '%s'", url)
|
2018-07-05 12:07:00 +02:00
|
|
|
|
2018-07-06 14:53:13 +02:00
|
|
|
content, err := rh.apiClient.GetContent(url)
|
2018-07-05 12:07:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Traverse to index file object for merging
|
2018-07-05 12:07:00 +02:00
|
|
|
indexFile := helm_repo.NewIndexFile()
|
|
|
|
if err := yaml.Unmarshal(content, indexFile); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return indexFile, nil
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Merge the content of mergingIndexFile to the baseIndex
|
|
|
|
// The chart url should be without --chart-url prefix
|
2018-07-05 12:07:00 +02:00
|
|
|
func (rh *RepositoryHandler) mergeIndexFile(namespace string,
|
|
|
|
baseIndex *helm_repo.IndexFile,
|
|
|
|
mergingIndexFile *helm_repo.IndexFile) {
|
2018-09-05 10:16:31 +02:00
|
|
|
// Append entries
|
2018-07-05 12:07:00 +02:00
|
|
|
for chartName, chartVersions := range mergingIndexFile.Entries {
|
|
|
|
nameWithNS := fmt.Sprintf("%s/%s", namespace, chartName)
|
|
|
|
for _, version := range chartVersions {
|
|
|
|
version.Name = nameWithNS
|
2018-09-05 10:16:31 +02:00
|
|
|
// Currently there is only one url
|
2018-07-05 12:07:00 +02:00
|
|
|
for index, url := range version.URLs {
|
2018-07-14 09:49:38 +02:00
|
|
|
version.URLs[index] = path.Join(namespace, url)
|
2018-07-05 12:07:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Appended
|
2018-07-05 12:07:00 +02:00
|
|
|
baseIndex.Entries[nameWithNS] = chartVersions
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Update generated time
|
2018-07-05 12:07:00 +02:00
|
|
|
if mergingIndexFile.Generated.After(baseIndex.Generated) {
|
|
|
|
baseIndex.Generated = mergingIndexFile.Generated
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Merge public keys
|
2018-07-05 12:07:00 +02:00
|
|
|
baseIndex.PublicKeys = append(baseIndex.PublicKeys, mergingIndexFile.PublicKeys...)
|
|
|
|
}
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Generate empty index file
|
2018-07-05 12:07:00 +02:00
|
|
|
func emptyIndexFile() []byte {
|
|
|
|
emptyIndexFile := &helm_repo.IndexFile{}
|
|
|
|
emptyIndexFile.Generated = time.Now()
|
|
|
|
|
2018-09-05 10:16:31 +02:00
|
|
|
// Ignore the error
|
2018-07-05 12:07:00 +02:00
|
|
|
rawData, _ := json.Marshal(emptyIndexFile)
|
|
|
|
|
|
|
|
return rawData
|
|
|
|
}
|