Skip to content

Commit

Permalink
Merge pull request #7851 from pedjak/filter-charts-4.6
Browse files Browse the repository at this point in the history
Bug 1916860: Filter out helm charts based on kubernetes version
  • Loading branch information
openshift-merge-robot committed Feb 17, 2021
2 parents 6ef026d + 29f25f2 commit fc7642a
Show file tree
Hide file tree
Showing 17 changed files with 1,231 additions and 33 deletions.
4 changes: 4 additions & 0 deletions docs/helm/endpoints_api.md
Expand Up @@ -335,6 +335,10 @@ _Returns repository index file containing all entries from all configured reposi

`GET`

* **Supported URL Query Parameter:**
* `onlyCompatible` - `true`/`false` Setting true would return helm charts which are supported in the provided cluster.
Default value is set to true if not provided.

* **Success Response:**

* **Code:** 200 <br />
Expand Down
29 changes: 24 additions & 5 deletions pkg/helm/chartproxy/proxy.go
@@ -1,22 +1,26 @@
package chartproxy

import (
"helm.sh/helm/v3/pkg/chartutil"
"helm.sh/helm/v3/pkg/repo"
"k8s.io/client-go/dynamic"
"k8s.io/client-go/kubernetes"
v1 "k8s.io/client-go/kubernetes/typed/core/v1"
"k8s.io/client-go/rest"

"github.com/openshift/console/pkg/version"
)

type proxy struct {
config *rest.Config
dynamicClient dynamic.Interface
coreV1Client v1.CoreV1Interface
helmRepoGetter HelmRepoGetter
kubeVersion string
}

type Proxy interface {
IndexFile() (*repo.IndexFile, error)
IndexFile(onlyCompatible bool) (*repo.IndexFile, error)
}

type RestConfigProvider func() (*rest.Config, error)
Expand All @@ -43,13 +47,16 @@ func coreClientProvider(p *proxy) error {

var defaultOptions = []ProxyOption{dynamicKubeClientProvider, coreClientProvider}

func New(k8sConfig RestConfigProvider, opts ...ProxyOption) (Proxy, error) {
func New(k8sConfig RestConfigProvider, kubeVersionGetter version.KubeVersionGetter, opts ...ProxyOption) (Proxy, error) {
config, err := k8sConfig()

if err != nil {
return nil, err
}

p := &proxy{
config: config,
config: config,
kubeVersion: kubeVersionGetter.GetKubeVersion(),
}

if len(opts) == 0 {
Expand All @@ -63,7 +70,7 @@ func New(k8sConfig RestConfigProvider, opts ...ProxyOption) (Proxy, error) {
return p, nil
}

func (p *proxy) IndexFile() (*repo.IndexFile, error) {
func (p *proxy) IndexFile(onlyCompatible bool) (*repo.IndexFile, error) {
helmRepos, err := p.helmRepoGetter.List()
if err != nil {
return nil, err
Expand All @@ -76,8 +83,20 @@ func (p *proxy) IndexFile() (*repo.IndexFile, error) {
plog.Errorf("Error retrieving index file for %v: %v", helmRepo, err)
continue
}

for key, entry := range idxFile.Entries {
indexFile.Entries[key+"--"+helmRepo.Name] = entry
if onlyCompatible {
for i := len(entry) - 1; i >= 0; i-- {
if entry[i].Metadata.KubeVersion != "" && p.kubeVersion != "" {
if !chartutil.IsCompatibleRange(entry[i].Metadata.KubeVersion, p.kubeVersion) {
entry = append(entry[:i], entry[i+1:]...)
}
}
}
}
if len(entry) > 0 {
indexFile.Entries[key+"--"+helmRepo.Name] = entry
}
}
}
return indexFile, nil
Expand Down
80 changes: 70 additions & 10 deletions pkg/helm/chartproxy/proxy_test.go
@@ -1,31 +1,90 @@
package chartproxy

import (
"golang.org/x/net/context"
"helm.sh/helm/v3/pkg/repo"
"io/ioutil"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"reflect"
"testing"

"golang.org/x/net/context"
"helm.sh/helm/v3/pkg/repo"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/client-go/rest"

"github.com/openshift/console/pkg/helm/actions/fake"
)

type MockKubeVersion struct {
fakeVersion string
}

func (v MockKubeVersion) GetKubeVersion() string {
return v.fakeVersion
}

func TestProxy_IndexFile(t *testing.T) {
tests := []struct {
name string
indexFiles []string
mergedFile string
helmCRS []*unstructured.Unstructured
name string
indexFiles []string
mergedFile string
kubeVersion string
helmCRS []*unstructured.Unstructured
onlyCompatible bool
}{
{
name: "returned index file for configured helm repo",
indexFiles: []string{"testdata/sampleRepoIndex.yaml"},
mergedFile: "testdata/mergedSampleRepoIndex2.yaml",
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.16.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndex3FilteredKube1-16-0.yaml",
kubeVersion: "v1.16.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.15.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndexFilteredKube1-15-0.yaml",
kubeVersion: "v1.15.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.14.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndexFilteredKube1-14-0.yaml",
kubeVersion: "v1.14.0",
onlyCompatible: true,
},
{
name: "return empty index file if not charts are compatible with given cluster",
indexFiles: []string{"testdata/incompatibleRepoIndex.yaml"},
mergedFile: "",
kubeVersion: "v1.15.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-beta2",
indexFiles: []string{"testdata/RepoIndexPreRelease.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-beta2.yaml",
kubeVersion: "v1.20.0-beta.2",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-alpha2",
indexFiles: []string{"testdata/RepoIndexPreRelease.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-alpha2.yaml",
kubeVersion: "v1.20.0-alpha.2",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-beta.2",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-0.yaml",
kubeVersion: "v1.20.0-beta.2",
onlyCompatible: true,
},
{
name: "returned merged index file for configured helm repos",
indexFiles: []string{"testdata/azureRepoIndex.yaml", "testdata/sampleRepoIndex.yaml"},
Expand Down Expand Up @@ -82,18 +141,19 @@ func TestProxy_IndexFile(t *testing.T) {
t.Error(err)
}
}

fakeProxyOption := func(p *proxy) error {
p.dynamicClient = dynamicClient
return nil
}
p, err := New(func() (r *rest.Config, err error) {
return &rest.Config{}, nil
}, fakeProxyOption)
}, MockKubeVersion{fakeVersion: tt.kubeVersion}, fakeProxyOption)
if err != nil {
t.Error(err)
}

indexFile, err := p.IndexFile()
indexFile, err := p.IndexFile(tt.onlyCompatible)
if err != nil {
t.Error(err)
}
Expand Down
78 changes: 78 additions & 0 deletions pkg/helm/chartproxy/testdata/RepoIndexPreRelease.yaml
@@ -0,0 +1,78 @@
apiVersion: v1
entries:
ibm-cpq-prod:
- apiVersion: v2
appVersion: 10.0.0.6
created: "2020-07-15T16:32:34.607411698+05:30"
description: |-
IBM Sterling Configure, Price, Quote (CPQ) solution enables you to quickly configure, price, quote and order the right products and services.\n
To sell competitively in today’s multichannel environment, organizations need a way to easily manage product and service configuration and pricing rules that allow prospects, customers, sales staff, call center representatives and Business Partners to quickly find, configure and order the right products and services.\n
IBM® Sterling Configure, Price, Quote solution automates every step of the configure, price and quote process to help organizations easily create Web storefronts, offer dynamic catalog and pricing information, and direct customers and partners to find, configure and order the right products and services.\n
This enables you to transform how you sell complex products and services by removing the internal complexity of multi-tiered selling within your organization and with your partners.\n
Documentation\n
Additional information about installation can be found at https://ibm.biz/BdqqUS.
License\n
Sterling Configure Price Quote Software is licensed under https://ibm.biz/Bdqqgd which must be accepted during the install of the Product.
digest: 2d93c570ff47b27858901dacafc22fe64d868e9a8ecb4742867396f2912eded8
icon: https://raw.githubusercontent.com/IBM/charts/master/logo/ibm-oms-logo.png
keywords:
- cpq
- ifs
- sterling
- yantra
- order
- fulfillment
- om
- amd64
- framework
- Commercial
- RHOCP
- All items
- Languages
- Databases
- Middleware
- CI/CD
- Other
kubeVersion: '>=1.20.0-alpha.2'
maintainers:
- name: IBM
name: ibm-cpq-prod
response:
- https://redhat-developer.github.com/redhat-helm-charts/charts/ibm-cpq-prod-2.0.0.tgz
version: 2.0.0
ibm-operator-catalog-enablement:
- apiVersion: v2
appVersion: 1.0.0
created: "2020-07-15T16:32:34.697730479+05:30"
description: |
IBM Operator Catalog enablement deploys custom CatalogSources for operators
available to deploy and maintain IBM products. The operators are publically
available, but products they install may require purchase and entitlement keys.
Documentation
For additional details regarding installation see https://ibm.biz/operator-catalog-readme
License
By installing this catalog you accept the license terms https://www.apache.org/licenses/LICENSE-2.0
digest: 5a765d2f3f36f1b35c403a546df3590e3d6014cbf86bd36775f8febb54baa58c
home: http://ibm.biz/oprcatalog
icon: 
keywords:
- amd64
- deploy
- Catalog
- Commercial
- Limited
- Other
- RHOCP
kubeVersion: '>=1.20.0-beta.2'
maintainers:
- name: IBM
name: ibm-operator-catalog-enablement
type: application
response:
- https://redhat-developer.github.com/redhat-helm-charts/charts/ibm-operator-catalog-enablement-1.0.0.tgz
version: 1.0.0
generated: "2020-03-30T16:27:13.4275217-05:00"

0 comments on commit fc7642a

Please sign in to comment.