2018-07-02 13:34:04 +02:00
|
|
|
package chartserver
|
|
|
|
|
2018-07-06 14:53:13 +02:00
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2018-08-29 09:26:21 +02:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2018-07-06 14:53:13 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/Masterminds/semver"
|
|
|
|
|
2018-08-23 09:02:20 +02:00
|
|
|
hlog "github.com/goharbor/harbor/src/common/utils/log"
|
2018-07-06 14:53:13 +02:00
|
|
|
"k8s.io/helm/pkg/chartutil"
|
|
|
|
helm_repo "k8s.io/helm/pkg/repo"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
readmeFileName = "README.md"
|
2018-07-23 11:41:40 +02:00
|
|
|
valuesFileName = "values.yaml"
|
2018-07-06 14:53:13 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
//ChartVersionDetails keeps the detailed data info of the chart version
|
|
|
|
type ChartVersionDetails struct {
|
|
|
|
Metadata *helm_repo.ChartVersion `json:"metadata"`
|
|
|
|
Dependencies []*chartutil.Dependency `json:"dependencies"`
|
|
|
|
Values map[string]interface{} `json:"values"`
|
|
|
|
Files map[string]string `json:"files"`
|
2018-07-27 09:07:22 +02:00
|
|
|
Security *SecurityReport `json:"security"`
|
|
|
|
}
|
|
|
|
|
|
|
|
//SecurityReport keeps the info related with security
|
|
|
|
//e.g.: digital signature, vulnerability scanning etc.
|
|
|
|
type SecurityReport struct {
|
|
|
|
Signature *DigitalSignature `json:"signature"`
|
|
|
|
}
|
|
|
|
|
|
|
|
//DigitalSignature used to indicate if the chart has been signed
|
|
|
|
type DigitalSignature struct {
|
|
|
|
Signed bool `json:"signed"`
|
|
|
|
Provenance string `json:"prov_file"`
|
2018-07-06 14:53:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//ChartInfo keeps the information of the chart
|
|
|
|
type ChartInfo struct {
|
|
|
|
Name string
|
|
|
|
TotalVersions uint32 `json:"total_versions"`
|
2018-08-30 08:40:41 +02:00
|
|
|
LatestVersion string `json:"latest_version"`
|
2018-07-06 14:53:13 +02:00
|
|
|
Created time.Time
|
2018-08-29 09:26:21 +02:00
|
|
|
Updated time.Time
|
2018-07-06 14:53:13 +02:00
|
|
|
Icon string
|
|
|
|
Home string
|
2018-08-06 10:31:41 +02:00
|
|
|
Deprecated bool
|
2018-07-06 14:53:13 +02:00
|
|
|
}
|
2018-07-02 13:34:04 +02:00
|
|
|
|
|
|
|
//ChartOperator is designed to process the contents of
|
|
|
|
//the specified chart version to get more details
|
|
|
|
type ChartOperator struct{}
|
|
|
|
|
|
|
|
//GetChartDetails parse the details from the provided content bytes
|
2018-07-06 14:53:13 +02:00
|
|
|
func (cho *ChartOperator) GetChartDetails(content []byte) (*ChartVersionDetails, error) {
|
|
|
|
if content == nil || len(content) == 0 {
|
|
|
|
return nil, errors.New("zero content")
|
|
|
|
}
|
|
|
|
|
|
|
|
//Load chart from in-memory content
|
|
|
|
reader := bytes.NewReader(content)
|
|
|
|
chartData, err := chartutil.LoadArchive(reader)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Parse the requirements of chart
|
|
|
|
requirements, err := chartutil.LoadRequirements(chartData)
|
|
|
|
if err != nil {
|
2018-07-23 08:53:40 +02:00
|
|
|
//If no requirements.yaml, return empty dependency list
|
|
|
|
if _, ok := err.(chartutil.ErrNoRequirementsFile); ok {
|
|
|
|
requirements = &chartutil.Requirements{
|
|
|
|
Dependencies: make([]*chartutil.Dependency, 0),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-07-06 14:53:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var values map[string]interface{}
|
|
|
|
files := make(map[string]string)
|
|
|
|
//Parse values
|
|
|
|
if chartData.Values != nil {
|
|
|
|
values = parseRawValues([]byte(chartData.Values.GetRaw()))
|
2018-07-23 08:53:40 +02:00
|
|
|
if len(values) > 0 {
|
|
|
|
//Append values.yaml file
|
2018-07-23 11:41:40 +02:00
|
|
|
files[valuesFileName] = chartData.Values.Raw
|
2018-07-23 08:53:40 +02:00
|
|
|
}
|
2018-07-06 14:53:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//Append other files like 'README.md'
|
|
|
|
for _, v := range chartData.GetFiles() {
|
|
|
|
if v.TypeUrl == readmeFileName {
|
|
|
|
files[readmeFileName] = string(v.GetValue())
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
theChart := &ChartVersionDetails{
|
|
|
|
Dependencies: requirements.Dependencies,
|
|
|
|
Values: values,
|
|
|
|
Files: files,
|
|
|
|
}
|
|
|
|
|
|
|
|
return theChart, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
//GetChartList returns a reorganized chart list
|
|
|
|
func (cho *ChartOperator) GetChartList(content []byte) ([]*ChartInfo, error) {
|
|
|
|
if content == nil || len(content) == 0 {
|
|
|
|
return nil, errors.New("zero content")
|
|
|
|
}
|
|
|
|
|
|
|
|
allCharts := make(map[string]helm_repo.ChartVersions)
|
|
|
|
if err := json.Unmarshal(content, &allCharts); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
chartList := make([]*ChartInfo, 0)
|
|
|
|
for key, chartVersions := range allCharts {
|
|
|
|
lVersion, oVersion := getTheTwoCharts(chartVersions)
|
|
|
|
if lVersion != nil && oVersion != nil {
|
|
|
|
chartInfo := &ChartInfo{
|
|
|
|
Name: key,
|
|
|
|
TotalVersions: uint32(len(chartVersions)),
|
|
|
|
}
|
|
|
|
chartInfo.Created = oVersion.Created
|
|
|
|
chartInfo.Home = lVersion.Home
|
|
|
|
chartInfo.Icon = lVersion.Icon
|
2018-08-06 10:31:41 +02:00
|
|
|
chartInfo.Deprecated = lVersion.Deprecated
|
2018-08-30 08:40:41 +02:00
|
|
|
chartInfo.LatestVersion = lVersion.GetVersion()
|
2018-07-06 14:53:13 +02:00
|
|
|
chartList = append(chartList, chartInfo)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-29 09:26:21 +02:00
|
|
|
//Sort the chart list by the updated time which is the create time
|
|
|
|
//of the latest version of the chart.
|
|
|
|
sort.Slice(chartList, func(i, j int) bool {
|
|
|
|
if chartList[i].Updated.Equal(chartList[j].Updated) {
|
|
|
|
return strings.Compare(chartList[i].Name, chartList[j].Name) < 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return chartList[i].Updated.After(chartList[j].Updated)
|
|
|
|
})
|
|
|
|
|
2018-07-06 14:53:13 +02:00
|
|
|
return chartList, nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 05:26:46 +02:00
|
|
|
//GetChartVersions returns the chart versions
|
|
|
|
func (cho *ChartOperator) GetChartVersions(content []byte) (helm_repo.ChartVersions, error) {
|
|
|
|
if content == nil || len(content) == 0 {
|
|
|
|
return nil, errors.New("zero content")
|
|
|
|
}
|
|
|
|
|
|
|
|
chartVersions := make(helm_repo.ChartVersions, 0)
|
|
|
|
if err := json.Unmarshal(content, &chartVersions); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return chartVersions, nil
|
|
|
|
}
|
|
|
|
|
2018-07-06 14:53:13 +02:00
|
|
|
//Get the latest and oldest chart versions
|
|
|
|
func getTheTwoCharts(chartVersions helm_repo.ChartVersions) (latestChart *helm_repo.ChartVersion, oldestChart *helm_repo.ChartVersion) {
|
|
|
|
if len(chartVersions) == 1 {
|
|
|
|
return chartVersions[0], chartVersions[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, chartVersion := range chartVersions {
|
|
|
|
currentV, err := semver.NewVersion(chartVersion.Version)
|
|
|
|
if err != nil {
|
2018-07-08 05:17:55 +02:00
|
|
|
//ignore it, just logged
|
|
|
|
hlog.Warningf("Malformed semversion %s for the chart %s", chartVersion.Version, chartVersion.Name)
|
2018-07-06 14:53:13 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
//Find latest chart
|
|
|
|
if latestChart == nil {
|
|
|
|
latestChart = chartVersion
|
|
|
|
} else {
|
2018-08-03 13:25:39 +02:00
|
|
|
lVersion, err := semver.NewVersion(latestChart.Version)
|
|
|
|
if err != nil {
|
|
|
|
//ignore it, just logged
|
|
|
|
hlog.Warningf("Malformed semversion %s for the chart %s", latestChart.Version, chartVersion.Name)
|
|
|
|
continue
|
|
|
|
}
|
2018-07-06 14:53:13 +02:00
|
|
|
if lVersion.LessThan(currentV) {
|
|
|
|
latestChart = chartVersion
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if oldestChart == nil {
|
|
|
|
oldestChart = chartVersion
|
|
|
|
} else {
|
|
|
|
if oldestChart.Created.After(chartVersion.Created) {
|
|
|
|
oldestChart = chartVersion
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return latestChart, oldestChart
|
|
|
|
}
|
|
|
|
|
|
|
|
//Parse the raw values to value map
|
|
|
|
func parseRawValues(rawValue []byte) map[string]interface{} {
|
|
|
|
valueMap := make(map[string]interface{})
|
|
|
|
|
|
|
|
if len(rawValue) == 0 {
|
|
|
|
return valueMap
|
|
|
|
}
|
|
|
|
|
|
|
|
values, err := chartutil.ReadValues(rawValue)
|
|
|
|
if err != nil || len(values) == 0 {
|
|
|
|
return valueMap
|
|
|
|
}
|
|
|
|
|
|
|
|
readValue(values, "", valueMap)
|
|
|
|
|
|
|
|
return valueMap
|
|
|
|
}
|
|
|
|
|
|
|
|
//Recursively read value
|
|
|
|
func readValue(values map[string]interface{}, keyPrefix string, valueMap map[string]interface{}) {
|
|
|
|
for key, value := range values {
|
|
|
|
longKey := key
|
|
|
|
if keyPrefix != "" {
|
|
|
|
longKey = fmt.Sprintf("%s.%s", keyPrefix, key)
|
|
|
|
}
|
|
|
|
|
|
|
|
if subValues, ok := value.(map[string]interface{}); ok {
|
|
|
|
readValue(subValues, longKey, valueMap)
|
|
|
|
} else {
|
|
|
|
valueMap[longKey] = value
|
|
|
|
}
|
|
|
|
}
|
2018-07-02 13:34:04 +02:00
|
|
|
}
|