Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Split out Kube code (with Helm & Workloads) into it's own package #475

Merged
merged 104 commits into from
Oct 2, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
104 commits
Select commit Hold shift + click to select a range
cd44987
Ensure filename/no filename text in connect by file dialog is aligned…
richard-cox Jul 28, 2020
f627688
Add typed entity access and `custom-src` removal to change log (#4496)
richard-cox Aug 7, 2020
d7102e7
Fixes #4335: Create Stratos static web site with better documentation…
Aug 7, 2020
50617f7
Fix and update customization docs (#4478)
richard-cox Aug 7, 2020
b713c78
Update base README, point to website (#4488)
richard-cox Aug 7, 2020
e12f60f
Update developer docs (#4489)
richard-cox Aug 7, 2020
4209358
Fix docs (#4497)
richard-cox Aug 7, 2020
dc2d6d6
Ensure images dependent on techPreview flag are included in imagelist…
richard-cox Aug 18, 2020
4df1516
Improve clean-symlinks script (#4509)
nwmac Aug 18, 2020
3a8cd74
Change nginx ciphers and make configurable via helm chart values (#4507)
nwmac Aug 18, 2020
200726c
Update version to 4.0.1, add change log (#4514)
richard-cox Aug 18, 2020
0285873
Fix deploy from gitlab using a group's repo (#4479)
richard-cox Aug 20, 2020
ad64761
Add additional time ranges to base metrics range selector (#4480)
richard-cox Aug 20, 2020
d27a12d
Add some time saving comments to cf permissions checker (#4508)
richard-cox Aug 20, 2020
8da357a
Move tab-nav and xsrf module source files to the src folder (#4518)
nwmac Aug 21, 2020
286d508
Fix GitHub branch limit (#4510)
richard-cox Aug 21, 2020
9a70ea6
Remove use of nodejs util module (#4521)
nwmac Aug 21, 2020
77d5a20
Remove dependencies between store and core that have crept in (#4517)
nwmac Aug 21, 2020
027f9c3
Fix store testing package (#4520)
nwmac Aug 21, 2020
5214f9f
Fix the position of the header guide array (#4524)
richard-cox Aug 24, 2020
009c203
Metrics: Ensure trailing slashes are ignored when comparing URLs (#4527)
nwmac Aug 24, 2020
da6285c
Remove imports of the form 'frontend/....' (#4519)
nwmac Aug 24, 2020
307fc20
Move endpopints-health-check.ts (#4530)
nwmac Aug 24, 2020
0aefbb1
Add UMD Ids to external modules in store package (#4522)
nwmac Aug 24, 2020
55f2244
Improvements & more checks to autoscaler scheduled date tests (#4535)
richard-cox Aug 25, 2020
a0b338a
Remove api driven views (#4537)
nwmac Aug 25, 2020
0ea6632
Remove logger service and action history (#4538)
nwmac Aug 25, 2020
5eb7f0e
Add support for API keys (#4515)
ikapelyukhin Aug 25, 2020
68d5df7
Update moment imports to remove warning when building library (#4534)
nwmac Aug 25, 2020
3c157d1
Improve presentation and fix issue with development versions
nwmac Aug 25, 2020
46732d8
Merge remote-tracking branch 'origin/improve-monocular-ui' into helm-…
nwmac Aug 25, 2020
641ccf8
WIP
richard-cox Aug 26, 2020
d845669
Fix Helm upgrade bug (#4544)
nwmac Aug 26, 2020
303bbb7
Add support for Helm Upgrade and Helm history
nwmac Aug 27, 2020
62d1a77
Remove console logging
nwmac Aug 27, 2020
d2dbafd
Add comment
nwmac Aug 27, 2020
cc8e2ac
Minor tweaks following self-review
nwmac Aug 27, 2020
dc24e61
Merge remote-tracking branch 'origin/master' into helm-hub
richard-cox Aug 27, 2020
ac18f0b
WIP
richard-cox Aug 27, 2020
d665348
Merge remote-tracking branch 'origin/master' into helm-hub
richard-cox Aug 28, 2020
adcd11a
fix install button
richard-cox Aug 28, 2020
43b2c33
Fix actual helm install
richard-cox Aug 28, 2020
a0ab8b6
Remove helm repos view
richard-cox Aug 28, 2020
f6f2d9f
Store api_keys.last_used in UTC (#4541)
ikapelyukhin Sep 1, 2020
fd2ee0f
Show per endpoint type actions in endpoints table
richard-cox Aug 28, 2020
ffd0de0
Tidy up registration step
richard-cox Sep 1, 2020
8042df9
Tidy up provider interceptor pattern
richard-cox Sep 1, 2020
7916f8a
Tidy up services, fix charts list filter by repo/helm hub
richard-cox Sep 1, 2020
7247090
Add endpoint unRegisterable, make helm types sub types, tidy up
richard-cox Sep 1, 2020
412fbf2
Remove debug logging
nwmac Sep 2, 2020
c8250ee
Merge remote-tracking branch 'origin/master' into helm-features
nwmac Sep 2, 2020
cffb053
Fix whitespace
nwmac Sep 2, 2020
b560761
Add UI for Stratos API Keys (#4523)
richard-cox Sep 2, 2020
20d79e0
Minor tidy ups
nwmac Sep 2, 2020
d1915b8
Merge remote-tracking branch 'origin/master' into helm-features
nwmac Sep 2, 2020
545669a
Add basic e2e tests for API Keys (#4536)
richard-cox Sep 2, 2020
4c09409
Fix compile issue
nwmac Sep 2, 2020
9dcdcce
Fixed versions again, and bugs following subtype split
richard-cox Sep 2, 2020
043c5fd
Merge remote-tracking branch 'origin/master' into helm-hub
richard-cox Sep 2, 2020
509e018
Fix unit tests
richard-cox Sep 2, 2020
e815dc4
Add db migration script to update existing helm endpoints with repo s…
richard-cox Sep 2, 2020
3726821
Bump docusaurus version and add support for versioning (#4506)
richard-cox Sep 3, 2020
9e7e679
Fix front-end unit tests
nwmac Sep 3, 2020
b63e39a
API Keys: Make feature configurable for different user types (#4540)
ikapelyukhin Sep 4, 2020
c767913
Add docs for UAA SSO user permissions management (#4554)
richard-cox Sep 4, 2020
cfbfab8
Add basic developers guide for working with helm (#4511)
richard-cox Sep 4, 2020
0f2e499
Always show upgrade button
nwmac Sep 7, 2020
bb89eb5
Minor entity store type updates
richard-cox Sep 7, 2020
6af9d88
Enable/disable API keys UI given API keys config setting (#4559)
richard-cox Sep 7, 2020
6409469
Convert unRegisterable to registeredLimit
richard-cox Sep 7, 2020
559ecc4
Changes following review
richard-cox Sep 8, 2020
1744b9a
Merge remote-tracking branch 'origin/master' into helm-hub
richard-cox Sep 8, 2020
150da8a
Fix display of helm type in favourite cards
richard-cox Sep 8, 2020
857c54c
Multiple Fixes
richard-cox Sep 8, 2020
b844e8c
Align icons in workload summary page
richard-cox Sep 8, 2020
a987ee7
Enable linting for all packages (#4561)
richard-cox Sep 8, 2020
14125c2
Fix lint failures
richard-cox Sep 8, 2020
f5eb911
Merge remote-tracking branch 'origin/master' into helm-features
nwmac Sep 9, 2020
ed15df6
Address PR feedback
nwmac Sep 9, 2020
ac5625a
Revert
nwmac Sep 9, 2020
7114563
Remove description when checking for similar charts
nwmac Sep 10, 2020
e85f7c2
Only show button when the helm chart is available
nwmac Sep 10, 2020
051338a
Bump angular json schema form (#4564)
nwmac Sep 10, 2020
ef00d18
Docs: Update internal versions & Automate future updates (#4558)
richard-cox Sep 10, 2020
a357a83
Merge remote-tracking branch 'origin/helm-features' into helm-hub
richard-cox Sep 10, 2020
77869e1
Improve types, start work on helm hub upgrade
richard-cox Sep 10, 2020
19c6363
Fixes following merge
richard-cox Sep 10, 2020
d97a270
Merge remote-tracking branch 'origin/master' into helm-hub
richard-cox Sep 10, 2020
520aae9
Remove TODOs
richard-cox Sep 10, 2020
ab83546
Merge remote-tracking branch 'upstream/master' into merge-upstream
richard-cox Sep 10, 2020
b8b019a
Merge fixes, remove fdescribe already in master
richard-cox Sep 10, 2020
058fb38
Fix linting
richard-cox Sep 10, 2020
557778d
Merge remote-tracking branch 'origin/helm-hub' into create-kube-package
richard-cox Sep 10, 2020
2b0772a
Merge remote-tracking branch 'origin/merge-upstream' into create-kube…
richard-cox Sep 10, 2020
853b1be
Create kubernetes package
richard-cox Sep 10, 2020
6ef4327
Fix imports
richard-cox Sep 11, 2020
f9f78cf
Fix unit tests
richard-cox Sep 11, 2020
fdf6538
Merge remote-tracking branch 'origin/master' into create-kube-package
richard-cox Sep 17, 2020
bfccf06
Fixes following review
richard-cox Sep 17, 2020
6ef9d74
Merge remote-tracking branch 'origin/master' into create-kube-package
richard-cox Sep 21, 2020
8b0bf54
Fix unit tests
richard-cox Sep 21, 2020
3540db9
Merge remote-tracking branch 'origin/master' into create-kube-package
richard-cox Sep 25, 2020
2228579
Merge remote-tracking branch 'origin/master' into create-kube-package
nwmac Oct 2, 2020
d68fc84
Fix theming for suse login
nwmac Oct 2, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Fix actual helm install
  • Loading branch information
richard-cox committed Aug 28, 2020
commit 43b2c339066aeec7936006e8fcc2dde51a0cb70c
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';

import { createMonocularEndpointUrlPart } from '../monocular.interceptor';
import { getMonocularEndpoint } from '../monocular.interceptor';

@Component({
selector: 'app-monocular',
Expand Down Expand Up @@ -29,7 +29,7 @@ export class MonocularChartViewComponent implements OnInit {

if (!!parts.version) {
breadcrumbs.push(
{ value: this.title, routerLink: `/monocular/charts/${createMonocularEndpointUrlPart(this.route)}/${parts.repo}/${parts.chartName}` }
{ value: this.title, routerLink: `/monocular/charts/${getMonocularEndpoint(this.route)}/${parts.repo}/${parts.chartName}` }
);
this.title = parts.version;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import { kubeEntityCatalog } from '../../kubernetes/kubernetes-entity-catalog';
import { KUBERNETES_ENDPOINT_TYPE } from '../../kubernetes/kubernetes-entity-factory';
import { KubernetesNamespace } from '../../kubernetes/store/kube.types';
import { helmEntityCatalog } from '../helm-entity-catalog';
import { createMonocularEndpointUrlPart, createMonocularProviders } from '../monocular.interceptor';
import { createMonocularProviders, getMonocularEndpoint } from '../monocular.interceptor';
import { HelmInstallValues } from '../store/helm.types';

@Component({
Expand Down Expand Up @@ -65,7 +65,7 @@ export class CreateReleaseComponent implements OnInit, OnDestroy {
private confirmDialog: ConfirmationDialogService,
) {
const chart = this.route.snapshot.params;
this.cancelUrl = `/monocular/charts/${createMonocularEndpointUrlPart(this.route)}/${chart.repo}/${chart.chartName}/${chart.version}`;
this.cancelUrl = `/monocular/charts/${getMonocularEndpoint(this.route)}/${chart.repo}/${chart.chartName}/${chart.version}`;

this.setupDetailsStep();

Expand Down Expand Up @@ -253,7 +253,8 @@ export class CreateReleaseComponent implements OnInit, OnDestroy {
const values: HelmInstallValues = {
...this.details.value,
...this.overrides.value,
chart: this.route.snapshot.params
chart: this.route.snapshot.params,
monocularEndpoint: getMonocularEndpoint(this.route, null, null)
};

// Make the request
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
import {
HTTP_INTERCEPTORS,
HttpBackend,
HttpClient,
HttpEvent,
HttpHandler,
HttpInterceptor,
HttpRequest,
HTTP_INTERCEPTORS,
HttpBackend,
HttpClient,
HttpEvent,
HttpHandler,
HttpInterceptor,
HttpRequest,
} from '@angular/common/http';
import { Injectable } from '@angular/core';
import { ActivatedRoute } from '@angular/router';
Expand All @@ -21,11 +21,13 @@ import { ReposService } from './monocular/shared/services/repos.service';
@Injectable()
export class MonocularInterceptor implements HttpInterceptor {


// TODO: RC ... install process. (install button. stepper cancel. on success, etc)
// TODO: RC create pr... self review
// TODO: RC remove 'repos' view
// TODO: RC fix 'repo' filter
// TODO: RC review interceptor and providing
// TODO: RC visit https://localhost:4700/monocular/charts/JXB9uEhvVCUEQU3Eq_WCD8uzVNc/billimek/adguard-home/1.2.0, long github expands side bar
constructor(
private route: ActivatedRoute,
) {
Expand Down Expand Up @@ -110,8 +112,8 @@ export const createMonocularProviders = () => [
/**
* Add the monocular endpoint id to a url. This could be the helm hub endpoint guid or `default` for stratos monocular
*/
export const createMonocularEndpointUrlPart = (route?: ActivatedRoute, chart?: Chart) => {
export const getMonocularEndpoint = (route?: ActivatedRoute, chart?: Chart, ifEmpty = 'default') => {
const endpointFromRoute = route ? route.snapshot.params.endpoint : null;
const endpointFromChart = chart ? chart.stratosEndpointId : null;
return endpointFromRoute || endpointFromChart || 'default';
return endpointFromRoute || endpointFromChart || ifEmpty;
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { DomSanitizer } from '@angular/platform-browser';
import { ActivatedRoute } from '@angular/router';

import { EndpointsService } from '../../../../../../../core/src/core/endpoints.service';
import { createMonocularEndpointUrlPart } from '../../../monocular.interceptor';
import { getMonocularEndpoint } from '../../../monocular.interceptor';
import { Chart } from '../../shared/models/chart';

@Component({
Expand Down Expand Up @@ -39,7 +39,7 @@ export class ChartDetailsUsageComponent implements OnInit {
}

get installUrl(): string {
return `/monocular/install/${createMonocularEndpointUrlPart(this.route, this.chart)}/${this.chart.id}/${this.currentVersion}`;
return `/monocular/install/${getMonocularEndpoint(this.route, this.chart)}/${this.chart.id}/${this.currentVersion}`;
}

}
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { Component, Input } from '@angular/core';
import { ActivatedRoute } from '@angular/router';

import { createMonocularEndpointUrlPart } from '../../../monocular.interceptor';
import { getMonocularEndpoint } from '../../../monocular.interceptor';
import { ChartAttributes } from '../../shared/models/chart';
import { ChartVersion } from '../../shared/models/chart-version';

Expand All @@ -19,7 +19,7 @@ export class ChartDetailsVersionsComponent {

goToVersionUrl(version: ChartVersion): string {
const chart: ChartAttributes = version.relationships.chart.data;
return `/monocular/charts/${createMonocularEndpointUrlPart(this.route)}/${chart.repo.name}/${chart.name}/${version.attributes.version}`;
return `/monocular/charts/${getMonocularEndpoint(this.route)}/${chart.repo.name}/${chart.name}/${version.attributes.version}`;
}

isSelected(version: ChartVersion): boolean {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { Component, OnInit } from '@angular/core';
import { ActivatedRoute, Params } from '@angular/router';
import { first } from 'rxjs/operators';

import { createMonocularEndpointUrlPart, createMonocularProviders } from '../../monocular.interceptor';
import { createMonocularProviders, getMonocularEndpoint } from '../../monocular.interceptor';
import { Chart } from '../shared/models/chart';
import { ChartVersion } from '../shared/models/chart-version';
import { ChartsService } from '../shared/services/charts.service';
Expand Down Expand Up @@ -59,6 +59,6 @@ export class ChartDetailsComponent implements OnInit {
updateMetaTags(): void { }

goToRepoUrl(): string {
return `/charts/${createMonocularEndpointUrlPart(null, this.chart)}/${this.chart.attributes.repo.name}`;
return `/charts/${getMonocularEndpoint(null, this.chart)}/${this.chart.attributes.repo.name}`;
}
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Component, OnInit } from '@angular/core';

import { createMonocularEndpointUrlPart, createMonocularProviders } from '../../monocular.interceptor';
import { createMonocularProviders, getMonocularEndpoint } from '../../monocular.interceptor';
import { Chart } from '../shared/models/chart';
import { ChartsService } from '../shared/services/charts.service';

Expand Down Expand Up @@ -30,11 +30,11 @@ export class ChartItemComponent implements OnInit {
}

goToDetailUrl(): string {
return `/monocular/charts/${createMonocularEndpointUrlPart(null, this.chart)}/${this.chart.attributes.repo.name}/${this.chart.attributes
return `/monocular/charts/${getMonocularEndpoint(null, this.chart)}/${this.chart.attributes.repo.name}/${this.chart.attributes
.name}`;
}

goToRepoUrl(): string {
return `/monocular/charts//${createMonocularEndpointUrlPart(null, this.chart)}/${this.chart.attributes.repo.name}`;
return `/monocular/charts//${getMonocularEndpoint(null, this.chart)}/${this.chart.attributes.repo.name}`;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ import {
} from './helm.actions';
import { HelmVersion } from './helm.types';

// TODO: RC location of these?
type MonocularChartsResponse = {
data: Chart[];
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ export enum HelmStatus {

export interface HelmInstallValues {
endpoint: string;
monocularEndpoint: string;
releaseName: string;
releaseNamespace: string;
values: string;
Expand Down
37 changes: 11 additions & 26 deletions src/jetstream/plugins/kubernetes/install_release.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ import (
"errors"
"fmt"

"github.com/helm/monocular/chartsvc"
"github.com/labstack/echo"
log "github.com/sirupsen/logrus"
"sigs.k8s.io/yaml"
Expand All @@ -23,11 +22,12 @@ import (
const chartCollection = "charts"

type installRequest struct {
Endpoint string `json:"endpoint"`
Name string `json:"releaseName"`
Namespace string `json:"releaseNamespace"`
Values string `json:"values"`
Chart struct {
Endpoint string `json:"endpoint"`
MonocularEndpoint string `json:"monocularEndpoint"`
Name string `json:"releaseName"`
Namespace string `json:"releaseNamespace"`
Values string `json:"values"`
Chart struct {
Name string `json:"chartName"`
Repository string `json:"repo"`
Version string `json:"version"`
Expand All @@ -36,7 +36,7 @@ type installRequest struct {

// Monocular is a plugin for Monocular
type Monocular interface {
GetChartStore() *chartsvc.ChartSvcDatastore
GetChartDownloadUrl(monocularEndpoint, chartID, chartVersion string) (string, error)
}

// InstallRelease will install a Helm 3 release
Expand All @@ -55,9 +55,9 @@ func (c *KubernetesSpecification) InstallRelease(ec echo.Context) error {

log.Debugf("Helm: Installing release %s", chartID)

downloadURL, err := c.getChart(chartID, params.Chart.Version)
downloadURL, err := c.getChart(params.MonocularEndpoint, chartID, params.Chart.Version)
if err != nil {
return interfaces.NewJetstreamErrorf("Could not get the Download URL for the Helm Chart")
return interfaces.NewJetstreamErrorf("Could not get the Download URL for the Helm Chart", err)
}

log.Debugf("Chart Download URL: %s", downloadURL)
Expand Down Expand Up @@ -125,7 +125,7 @@ func (c *KubernetesSpecification) InstallRelease(ec echo.Context) error {
return ec.JSON(200, release)
}

func (c *KubernetesSpecification) getChart(chartID, version string) (string, error) {
func (c *KubernetesSpecification) getChart(monocularEndpoint, chartID, version string) (string, error) {
helm := c.portalProxy.GetPlugin("monocular")
if helm == nil {
return "", errors.New("Could not find monocular plugin")
Expand All @@ -136,22 +136,7 @@ func (c *KubernetesSpecification) getChart(chartID, version string) (string, err
return "", errors.New("Could not find monocular plugin interface")
}

store := monocular.GetChartStore()
chart, err := store.GetChart(chartID)
if err != nil {
return "", errors.New("Could not find Chart")
}

// Find the download URL for the version
for _, chartVersion := range chart.ChartVersions {
if chartVersion.Version == version {
if len(chartVersion.URLs) == 1 {
return chartVersion.URLs[0], nil
}
}
}

return "", errors.New("Could not find Chart Version")
return monocular.GetChartDownloadUrl(monocularEndpoint, chartID, version)
}

// DeleteRelease will delete a release
Expand Down
84 changes: 76 additions & 8 deletions src/jetstream/plugins/monocular/main.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package monocular

import (
"encoding/json"
"errors"
"fmt"
"io/ioutil"
Expand All @@ -18,6 +19,7 @@ import (

"github.com/helm/monocular/chartsvc"
"github.com/helm/monocular/chartsvc/foundationdb"
"github.com/helm/monocular/chartsvc/models"
)

const (
Expand All @@ -44,15 +46,27 @@ type Monocular struct {
devSyncPID int
}

type HelmHubChart struct {
ID string `json:"id"`
Type string `json:"type"`
Attributes *models.ChartVersion `json:"attributes"`
Links map[string]interface{} `json:"links"`
Relationships map[string]interface{} `json:"relationships"`
}

type HelmHubChartResponse struct {
Data HelmHubChart `json:"data"`
}

// Init creates a new Monocular
func Init(portalProxy interfaces.PortalProxy) (interfaces.StratosPlugin, error) {
return &Monocular{portalProxy: portalProxy}, nil
}

// GetChartStore gets the chart store
func (m *Monocular) GetChartStore() *chartsvc.ChartSvcDatastore {
return m.RepoQueryStore
}
// // GetChartStore gets the chart store
// func (m *Monocular) GetChartStore() *chartsvc.ChartSvcDatastore {
// return m.RepoQueryStore
// }

// Init performs plugin initialization
func (m *Monocular) Init() error {
Expand Down Expand Up @@ -344,7 +358,7 @@ func (m *Monocular) baseHandleMonocular(c echo.Context, monocularEndpoint *inter
}
log.Warnf("URL to monocular: %v", url.String())

req, err := http.NewRequest(http.MethodGet, url.String(), nil)
req, err := http.NewRequest(c.Request().Method, url.String(), nil)

// TODO: RC use the generic function elsewhere
for k, v := range c.Request().Header {
Expand All @@ -368,21 +382,75 @@ func (m *Monocular) baseHandleMonocular(c echo.Context, monocularEndpoint *inter
res, err := client.Do(req)

if err != nil {
log.Errorf("MON ERRORED: %+v", err)
// log.Errorf("MON ERRORED: %+v", err)
c.Response().Status = 500
c.Response().Write([]byte(err.Error()))
} else if res.Body != nil {
log.Errorf("MON BODY")
// log.Errorf("MON BODY")
c.Response().Status = res.StatusCode
c.Response().Header().Set("Content-Type", res.Header.Get("Content-Type"))
body, _ := ioutil.ReadAll(res.Body)
c.Response().Write(body)
defer res.Body.Close()
} else {
log.Errorf("MON NO BODY")
// log.Errorf("MON NO BODY")
c.Response().Status = 200
c.Response().Write([]byte("{ data: [] }")) // TODO: RC
}

return nil
}

// GetChartDownloadUrl ... Get the download url for the bits required to install the given chart
func (m *Monocular) GetChartDownloadUrl(monocularEndpoint, chartID, version string) (string, error) {
if len(monocularEndpoint) > 0 {
// Fetch the monocular endpoint for the url
endpoint, err := m.validateMonocularRequest(monocularEndpoint)
if err != nil {
return "", err
}
url := endpoint.APIEndpoint

// Fetch the chart, this will give us the url to download the bits
url.Path += "/chartsvc/v1/charts/" + chartID + "/versions/" + version
req, err := http.NewRequest(http.MethodGet, url.String(), nil)
req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON)
client := &http.Client{Timeout: 30 * time.Second}
res, err := client.Do(req)
if err != nil {
return "", err
} else if res.Body != nil {
body, _ := ioutil.ReadAll(res.Body)
defer res.Body.Close()

// Reach into the chart response for the download URL
chartVersionResponse := &HelmHubChartResponse{}
err := json.Unmarshal(body, chartVersionResponse)
if err != nil {
return "", err
}
if len(chartVersionResponse.Data.Attributes.URLs) < 1 {
return "", errors.New("Response contained no chart package urls")
}
return chartVersionResponse.Data.Attributes.URLs[0], err
} else {
return "", errors.New("No body in response to chart request")
}
} else {
store := m.RepoQueryStore
chart, err := store.GetChart(chartID)
if err != nil {
return "", errors.New("Could not find Chart")
}

// Find the download URL for the version
for _, chartVersion := range chart.ChartVersions {
if chartVersion.Version == version {
if len(chartVersion.URLs) == 1 {
return chartVersion.URLs[0], nil
}
}
}
return "", errors.New("Could not find Chart Version")
}
}