| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | Copyright The Helm Authors. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package action | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	"text/template" | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-04 01:42:59 +08:00
										 |  |  | 	"github.com/Masterminds/sprig/v3" | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	"github.com/pkg/errors" | 
					
						
							| 
									
										
										
										
											2020-02-21 03:56:03 +08:00
										 |  |  | 	v1 "k8s.io/api/core/v1" | 
					
						
							| 
									
										
										
										
											2019-08-17 06:26:09 +08:00
										 |  |  | 	apierrors "k8s.io/apimachinery/pkg/api/errors" | 
					
						
							| 
									
										
										
										
											2020-02-21 03:56:03 +08:00
										 |  |  | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 	"k8s.io/cli-runtime/pkg/resource" | 
					
						
							| 
									
										
										
										
											2020-02-21 03:56:03 +08:00
										 |  |  | 	"sigs.k8s.io/yaml" | 
					
						
							| 
									
										
										
										
											2019-01-09 05:37:55 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-04 02:27:05 +08:00
										 |  |  | 	"helm.sh/helm/v3/pkg/chart" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/chartutil" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/cli" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/downloader" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/getter" | 
					
						
							| 
									
										
										
										
											2020-02-21 08:42:47 +08:00
										 |  |  | 	"helm.sh/helm/v3/pkg/kube" | 
					
						
							| 
									
										
										
										
											2019-10-04 02:27:05 +08:00
										 |  |  | 	kubefake "helm.sh/helm/v3/pkg/kube/fake" | 
					
						
							| 
									
										
										
										
											2019-09-24 01:29:24 +08:00
										 |  |  | 	"helm.sh/helm/v3/pkg/postrender" | 
					
						
							| 
									
										
										
										
											2019-10-04 02:27:05 +08:00
										 |  |  | 	"helm.sh/helm/v3/pkg/release" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/releaseutil" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/repo" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/storage" | 
					
						
							|  |  |  | 	"helm.sh/helm/v3/pkg/storage/driver" | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // releaseNameMaxLen is the maximum length of a release name.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // As of Kubernetes 1.4, the max limit on a name is 63 chars. We reserve 10 for
 | 
					
						
							|  |  |  | // charts to add data. Effectively, that gives us 53 chars.
 | 
					
						
							| 
									
										
										
										
											2019-03-06 03:57:39 +08:00
										 |  |  | // See https://github.com/helm/helm/issues/1528
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | const releaseNameMaxLen = 53 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NOTESFILE_SUFFIX that we want to treat special. It goes through the templating engine
 | 
					
						
							|  |  |  | // but it's not a yaml file (resource) hence can't have hooks, etc. And the user actually
 | 
					
						
							|  |  |  | // wants to see this file after rendering in the status command. However, it must be a suffix
 | 
					
						
							|  |  |  | // since there can be filepath in front of it.
 | 
					
						
							|  |  |  | const notesFileSuffix = "NOTES.txt" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 16:46:42 +08:00
										 |  |  | const defaultDirectoryPermission = 0755 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | // Install performs an installation operation.
 | 
					
						
							|  |  |  | type Install struct { | 
					
						
							|  |  |  | 	cfg *Configuration | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	ChartPathOptions | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-30 02:17:54 +08:00
										 |  |  | 	ClientOnly               bool | 
					
						
							| 
									
										
										
										
											2020-02-21 03:56:03 +08:00
										 |  |  | 	CreateNamespace          bool | 
					
						
							| 
									
										
										
										
											2019-10-30 02:17:54 +08:00
										 |  |  | 	DryRun                   bool | 
					
						
							|  |  |  | 	DisableHooks             bool | 
					
						
							|  |  |  | 	Replace                  bool | 
					
						
							|  |  |  | 	Wait                     bool | 
					
						
							|  |  |  | 	Devel                    bool | 
					
						
							|  |  |  | 	DependencyUpdate         bool | 
					
						
							|  |  |  | 	Timeout                  time.Duration | 
					
						
							|  |  |  | 	Namespace                string | 
					
						
							|  |  |  | 	ReleaseName              string | 
					
						
							|  |  |  | 	GenerateName             bool | 
					
						
							|  |  |  | 	NameTemplate             string | 
					
						
							|  |  |  | 	Description              string | 
					
						
							|  |  |  | 	OutputDir                string | 
					
						
							|  |  |  | 	Atomic                   bool | 
					
						
							|  |  |  | 	SkipCRDs                 bool | 
					
						
							|  |  |  | 	SubNotes                 bool | 
					
						
							|  |  |  | 	DisableOpenAPIValidation bool | 
					
						
							| 
									
										
										
										
											2020-02-05 05:16:51 +08:00
										 |  |  | 	IncludeCRDs              bool | 
					
						
							| 
									
										
										
										
											2019-10-08 03:23:42 +08:00
										 |  |  | 	// APIVersions allows a manual set of supported API Versions to be passed
 | 
					
						
							|  |  |  | 	// (for things like templating). These are ignored if ClientOnly is false
 | 
					
						
							|  |  |  | 	APIVersions chartutil.VersionSet | 
					
						
							| 
									
										
										
										
											2019-12-18 20:04:08 +08:00
										 |  |  | 	// Used by helm template to render charts with .Release.IsUpgrade. Ignored if Dry-Run is false
 | 
					
						
							| 
									
										
										
										
											2019-11-22 23:27:08 +08:00
										 |  |  | 	IsUpgrade bool | 
					
						
							| 
									
										
										
										
											2020-02-05 00:27:38 +08:00
										 |  |  | 	// Used by helm template to add the release as part of OutputDir path
 | 
					
						
							|  |  |  | 	// OutputDir/<ReleaseName>
 | 
					
						
							|  |  |  | 	UseReleaseName bool | 
					
						
							| 
									
										
										
										
											2019-09-24 01:29:24 +08:00
										 |  |  | 	PostRenderer   postrender.PostRenderer | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 06:26:09 +08:00
										 |  |  | // ChartPathOptions captures common options used for controlling chart paths
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | type ChartPathOptions struct { | 
					
						
							|  |  |  | 	CaFile   string // --ca-file
 | 
					
						
							|  |  |  | 	CertFile string // --cert-file
 | 
					
						
							|  |  |  | 	KeyFile  string // --key-file
 | 
					
						
							|  |  |  | 	Keyring  string // --keyring
 | 
					
						
							|  |  |  | 	Password string // --password
 | 
					
						
							|  |  |  | 	RepoURL  string // --repo
 | 
					
						
							|  |  |  | 	Username string // --username
 | 
					
						
							|  |  |  | 	Verify   bool   // --verify
 | 
					
						
							|  |  |  | 	Version  string // --version
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewInstall creates a new Install object with the given configuration.
 | 
					
						
							|  |  |  | func NewInstall(cfg *Configuration) *Install { | 
					
						
							|  |  |  | 	return &Install{ | 
					
						
							|  |  |  | 		cfg: cfg, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 05:31:26 +08:00
										 |  |  | func (i *Install) installCRDs(crds []chart.CRD) error { | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 	// We do these one file at a time in the order they were read.
 | 
					
						
							|  |  |  | 	totalItems := []*resource.Info{} | 
					
						
							|  |  |  | 	for _, obj := range crds { | 
					
						
							|  |  |  | 		// Read in the resources
 | 
					
						
							| 
									
										
										
										
											2020-01-21 05:31:26 +08:00
										 |  |  | 		res, err := i.cfg.KubeClient.Build(bytes.NewBuffer(obj.File.Data), false) | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return errors.Wrapf(err, "failed to install CRD %s", obj.Name) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Send them to Kube
 | 
					
						
							|  |  |  | 		if _, err := i.cfg.KubeClient.Create(res); err != nil { | 
					
						
							|  |  |  | 			// If the error is CRD already exists, continue.
 | 
					
						
							|  |  |  | 			if apierrors.IsAlreadyExists(err) { | 
					
						
							|  |  |  | 				crdName := res[0].Name | 
					
						
							|  |  |  | 				i.cfg.Log("CRD %s is already present. Skipping.", crdName) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-11-13 10:41:01 +08:00
										 |  |  | 			return errors.Wrapf(err, "failed to install CRD %s", obj.Name) | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		totalItems = append(totalItems, res...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Invalidate the local cache, since it will not have the new CRDs
 | 
					
						
							|  |  |  | 	// present.
 | 
					
						
							|  |  |  | 	discoveryClient, err := i.cfg.RESTClientGetter.ToDiscoveryClient() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	i.cfg.Log("Clearing discovery cache") | 
					
						
							|  |  |  | 	discoveryClient.Invalidate() | 
					
						
							|  |  |  | 	// Give time for the CRD to be recognized.
 | 
					
						
							|  |  |  | 	if err := i.cfg.KubeClient.Wait(totalItems, 60*time.Second); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Make sure to force a rebuild of the cache.
 | 
					
						
							|  |  |  | 	discoveryClient.ServerGroups() | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | // Run executes the installation
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // If DryRun is set to true, this will prepare the release, but not install it
 | 
					
						
							| 
									
										
										
										
											2019-08-01 23:04:36 +08:00
										 |  |  | func (i *Install) Run(chrt *chart.Chart, vals map[string]interface{}) (*release.Release, error) { | 
					
						
							| 
									
										
										
										
											2019-09-13 20:14:25 +08:00
										 |  |  | 	// Check reachability of cluster unless in client-only mode (e.g. `helm template` without `--validate`)
 | 
					
						
							|  |  |  | 	if !i.ClientOnly { | 
					
						
							|  |  |  | 		if err := i.cfg.KubeClient.IsReachable(); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-08-26 05:46:12 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	if err := i.availableName(); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 	// Pre-install anything in the crd/ directory. We do this before Helm
 | 
					
						
							|  |  |  | 	// contacts the upstream server and builds the capabilities object.
 | 
					
						
							| 
									
										
										
										
											2020-01-28 06:06:06 +08:00
										 |  |  | 	if crds := chrt.CRDObjects(); !i.ClientOnly && !i.SkipCRDs && len(crds) > 0 { | 
					
						
							| 
									
										
										
										
											2019-09-04 22:32:24 +08:00
										 |  |  | 		// On dry run, bail here
 | 
					
						
							|  |  |  | 		if i.DryRun { | 
					
						
							|  |  |  | 			i.cfg.Log("WARNING: This chart or one of its subcharts contains CRDs. Rendering may fail or contain inaccuracies.") | 
					
						
							|  |  |  | 		} else if err := i.installCRDs(crds); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-20 03:27:14 +08:00
										 |  |  | 	if i.ClientOnly { | 
					
						
							|  |  |  | 		// Add mock objects in here so it doesn't use Kube API server
 | 
					
						
							|  |  |  | 		// NOTE(bacongobbler): used for `helm template`
 | 
					
						
							|  |  |  | 		i.cfg.Capabilities = chartutil.DefaultCapabilities | 
					
						
							| 
									
										
										
										
											2019-10-08 03:23:42 +08:00
										 |  |  | 		i.cfg.Capabilities.APIVersions = append(i.cfg.Capabilities.APIVersions, i.APIVersions...) | 
					
						
							| 
									
										
										
										
											2019-07-20 03:27:14 +08:00
										 |  |  | 		i.cfg.KubeClient = &kubefake.PrintingKubeClient{Out: ioutil.Discard} | 
					
						
							| 
									
										
										
										
											2020-01-30 11:56:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		mem := driver.NewMemory() | 
					
						
							|  |  |  | 		mem.SetNamespace(i.Namespace) | 
					
						
							|  |  |  | 		i.cfg.Releases = storage.Init(mem) | 
					
						
							| 
									
										
										
										
											2019-10-08 03:23:42 +08:00
										 |  |  | 	} else if !i.ClientOnly && len(i.APIVersions) > 0 { | 
					
						
							|  |  |  | 		i.cfg.Log("API Version list given outside of client only mode, this list will be ignored") | 
					
						
							| 
									
										
										
										
											2019-07-20 03:27:14 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-01 23:04:36 +08:00
										 |  |  | 	if err := chartutil.ProcessDependencies(chrt, vals); err != nil { | 
					
						
							| 
									
										
										
										
											2019-07-25 16:11:43 +08:00
										 |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 	// Make sure if Atomic is set, that wait is set as well. This makes it so
 | 
					
						
							|  |  |  | 	// the user doesn't have to specify both
 | 
					
						
							|  |  |  | 	i.Wait = i.Wait || i.Atomic | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-08 03:02:07 +08:00
										 |  |  | 	caps, err := i.cfg.getCapabilities() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-22 23:27:08 +08:00
										 |  |  | 	//special case for helm template --is-upgrade
 | 
					
						
							|  |  |  | 	isUpgrade := i.IsUpgrade && i.DryRun | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	options := chartutil.ReleaseOptions{ | 
					
						
							|  |  |  | 		Name:      i.ReleaseName, | 
					
						
							| 
									
										
										
										
											2019-05-16 05:08:51 +08:00
										 |  |  | 		Namespace: i.Namespace, | 
					
						
							| 
									
										
										
										
											2019-10-18 02:04:45 +08:00
										 |  |  | 		Revision:  1, | 
					
						
							| 
									
										
										
										
											2019-11-22 23:27:08 +08:00
										 |  |  | 		IsInstall: !isUpgrade, | 
					
						
							|  |  |  | 		IsUpgrade: isUpgrade, | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-08-01 23:04:36 +08:00
										 |  |  | 	valuesToRender, err := chartutil.ToRenderValues(chrt, vals, options, caps) | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-01 23:04:36 +08:00
										 |  |  | 	rel := i.createRelease(chrt, vals) | 
					
						
							| 
									
										
										
										
											2019-08-17 06:26:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	var manifestDoc *bytes.Buffer | 
					
						
							| 
									
										
										
										
											2020-04-23 00:09:34 +08:00
										 |  |  | 	rel.Hooks, manifestDoc, rel.Info.Notes, err = i.cfg.renderResources(chrt, valuesToRender, i.ReleaseName, i.OutputDir, i.SubNotes, i.UseReleaseName, i.IncludeCRDs, i.PostRenderer, i.DryRun) | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	// Even for errors, attach this if available
 | 
					
						
							|  |  |  | 	if manifestDoc != nil { | 
					
						
							|  |  |  | 		rel.Manifest = manifestDoc.String() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Check error from render
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		rel.SetStatus(release.StatusFailed, fmt.Sprintf("failed to render resource: %s", err.Error())) | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 		// Return a release with partial data so that the client can show debugging information.
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 		return rel, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Mark this release as in-progress
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	rel.SetStatus(release.StatusPendingInstall, "Initial install underway") | 
					
						
							| 
									
										
										
										
											2019-07-25 04:24:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 10:41:30 +08:00
										 |  |  | 	var toBeAdopted kube.ResourceList | 
					
						
							| 
									
										
										
										
											2019-10-30 02:17:54 +08:00
										 |  |  | 	resources, err := i.cfg.KubeClient.Build(bytes.NewBufferString(rel.Manifest), !i.DisableOpenAPIValidation) | 
					
						
							| 
									
										
										
										
											2019-07-25 04:24:32 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, errors.Wrap(err, "unable to build kubernetes objects from release manifest") | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 10:22:15 +08:00
										 |  |  | 	// It is safe to use "force" here because these are resources currently rendered by the chart.
 | 
					
						
							|  |  |  | 	err = resources.Visit(setMetadataVisitor(rel.Name, rel.Namespace, true)) | 
					
						
							| 
									
										
										
										
											2020-02-21 09:53:27 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-01 06:05:41 +08:00
										 |  |  | 	// Install requires an extra validation step of checking that resources
 | 
					
						
							|  |  |  | 	// don't already exist before we actually create resources. If we continue
 | 
					
						
							|  |  |  | 	// forward and create the release object with resources that already exist,
 | 
					
						
							|  |  |  | 	// we'll end up in a state where we will delete those resources upon
 | 
					
						
							|  |  |  | 	// deleting the release because the manifest will be pointing at that
 | 
					
						
							|  |  |  | 	// resource
 | 
					
						
							| 
									
										
										
										
											2019-11-22 23:27:08 +08:00
										 |  |  | 	if !i.ClientOnly && !isUpgrade { | 
					
						
							| 
									
										
										
										
											2020-03-06 10:41:30 +08:00
										 |  |  | 		toBeAdopted, err = existingResourceConflict(resources, rel.Name, rel.Namespace) | 
					
						
							| 
									
										
										
										
											2020-02-21 08:42:47 +08:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2019-10-02 03:38:57 +08:00
										 |  |  | 			return nil, errors.Wrap(err, "rendered manifests contain a resource that already exists. Unable to continue with install") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Bail out here if it is a dry run
 | 
					
						
							|  |  |  | 	if i.DryRun { | 
					
						
							|  |  |  | 		rel.Info.Description = "Dry run complete" | 
					
						
							|  |  |  | 		return rel, nil | 
					
						
							| 
									
										
										
										
											2019-10-01 06:05:41 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-21 03:56:03 +08:00
										 |  |  | 	if i.CreateNamespace { | 
					
						
							|  |  |  | 		ns := &v1.Namespace{ | 
					
						
							|  |  |  | 			TypeMeta: metav1.TypeMeta{ | 
					
						
							|  |  |  | 				APIVersion: "v1", | 
					
						
							|  |  |  | 				Kind:       "Namespace", | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			ObjectMeta: metav1.ObjectMeta{ | 
					
						
							|  |  |  | 				Name: i.Namespace, | 
					
						
							|  |  |  | 				Labels: map[string]string{ | 
					
						
							|  |  |  | 					"name": i.Namespace, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		buf, err := yaml.Marshal(ns) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		resourceList, err := i.cfg.KubeClient.Build(bytes.NewBuffer(buf), true) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if _, err := i.cfg.KubeClient.Create(resourceList); err != nil && !apierrors.IsAlreadyExists(err) { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-26 02:45:03 +08:00
										 |  |  | 	// If Replace is true, we need to supercede the last release.
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	if i.Replace { | 
					
						
							|  |  |  | 		if err := i.replaceRelease(rel); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Store the release in history before continuing (new in Helm 3). We always know
 | 
					
						
							|  |  |  | 	// that this is a create operation.
 | 
					
						
							|  |  |  | 	if err := i.cfg.Releases.Create(rel); err != nil { | 
					
						
							|  |  |  | 		// We could try to recover gracefully here, but since nothing has been installed
 | 
					
						
							|  |  |  | 		// yet, this is probably safer than trying to continue when we know storage is
 | 
					
						
							|  |  |  | 		// not working.
 | 
					
						
							|  |  |  | 		return rel, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// pre-install hooks
 | 
					
						
							|  |  |  | 	if !i.DisableHooks { | 
					
						
							| 
									
										
										
										
											2019-07-26 02:45:03 +08:00
										 |  |  | 		if err := i.cfg.execHook(rel, release.HookPreInstall, i.Timeout); err != nil { | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 			return i.failRelease(rel, fmt.Errorf("failed pre-install: %s", err)) | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// At this point, we can do the install. Note that before we were detecting whether to
 | 
					
						
							|  |  |  | 	// do an update, but it's not clear whether we WANT to do an update if the re-use is set
 | 
					
						
							|  |  |  | 	// to true, since that is basically an upgrade operation.
 | 
					
						
							| 
									
										
										
										
											2020-03-06 10:41:30 +08:00
										 |  |  | 	if len(toBeAdopted) == 0 { | 
					
						
							|  |  |  | 		if _, err := i.cfg.KubeClient.Create(resources); err != nil { | 
					
						
							|  |  |  | 			return i.failRelease(rel, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if _, err := i.cfg.KubeClient.Update(toBeAdopted, resources, false); err != nil { | 
					
						
							|  |  |  | 			return i.failRelease(rel, err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-25 08:18:42 +08:00
										 |  |  | 	if i.Wait { | 
					
						
							| 
									
										
										
										
											2019-07-25 04:24:32 +08:00
										 |  |  | 		if err := i.cfg.KubeClient.Wait(resources, i.Timeout); err != nil { | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 			return i.failRelease(rel, err) | 
					
						
							| 
									
										
										
										
											2019-04-25 08:18:42 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	if !i.DisableHooks { | 
					
						
							| 
									
										
										
										
											2019-07-26 02:45:03 +08:00
										 |  |  | 		if err := i.cfg.execHook(rel, release.HookPostInstall, i.Timeout); err != nil { | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 			return i.failRelease(rel, fmt.Errorf("failed post-install: %s", err)) | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 00:48:31 +08:00
										 |  |  | 	if len(i.Description) > 0 { | 
					
						
							|  |  |  | 		rel.SetStatus(release.StatusDeployed, i.Description) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		rel.SetStatus(release.StatusDeployed, "Install complete") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// This is a tricky case. The release has been created, but the result
 | 
					
						
							|  |  |  | 	// cannot be recorded. The truest thing to tell the user is that the
 | 
					
						
							|  |  |  | 	// release was created. However, the user will not be able to do anything
 | 
					
						
							|  |  |  | 	// further with this release.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// One possible strategy would be to do a timed retry to see if we can get
 | 
					
						
							|  |  |  | 	// this stored in the future.
 | 
					
						
							| 
									
										
										
										
											2019-11-13 05:12:45 +08:00
										 |  |  | 	if err := i.recordRelease(rel); err != nil { | 
					
						
							|  |  |  | 		i.cfg.Log("failed to record the release: %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return rel, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | func (i *Install) failRelease(rel *release.Release, err error) (*release.Release, error) { | 
					
						
							|  |  |  | 	rel.SetStatus(release.StatusFailed, fmt.Sprintf("Release %q failed: %s", i.ReleaseName, err.Error())) | 
					
						
							|  |  |  | 	if i.Atomic { | 
					
						
							| 
									
										
										
										
											2019-07-12 02:28:22 +08:00
										 |  |  | 		i.cfg.Log("Install failed and atomic is set, uninstalling release") | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 		uninstall := NewUninstall(i.cfg) | 
					
						
							|  |  |  | 		uninstall.DisableHooks = i.DisableHooks | 
					
						
							|  |  |  | 		uninstall.KeepHistory = false | 
					
						
							|  |  |  | 		uninstall.Timeout = i.Timeout | 
					
						
							|  |  |  | 		if _, uninstallErr := uninstall.Run(i.ReleaseName); uninstallErr != nil { | 
					
						
							| 
									
										
										
										
											2019-07-12 02:28:22 +08:00
										 |  |  | 			return rel, errors.Wrapf(uninstallErr, "an error occurred while uninstalling the release. original install error: %s", err) | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-12 02:28:22 +08:00
										 |  |  | 		return rel, errors.Wrapf(err, "release %s failed, and has been uninstalled due to atomic being set", i.ReleaseName) | 
					
						
							| 
									
										
										
										
											2019-07-09 03:36:34 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	i.recordRelease(rel) // Ignore the error, since we have another error to deal with.
 | 
					
						
							|  |  |  | 	return rel, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | // availableName tests whether a name is available
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Roughly, this will return an error if name is
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //	- empty
 | 
					
						
							|  |  |  | //	- too long
 | 
					
						
							| 
									
										
										
										
											2019-07-31 03:54:23 +08:00
										 |  |  | //	- already in use, and not deleted
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | //	- used by a deleted release, and i.Replace is false
 | 
					
						
							|  |  |  | func (i *Install) availableName() error { | 
					
						
							|  |  |  | 	start := i.ReleaseName | 
					
						
							|  |  |  | 	if start == "" { | 
					
						
							|  |  |  | 		return errors.New("name is required") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(start) > releaseNameMaxLen { | 
					
						
							|  |  |  | 		return errors.Errorf("release name %q exceeds max length of %d", start, releaseNameMaxLen) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-31 03:54:23 +08:00
										 |  |  | 	if i.DryRun { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-08 08:45:14 +08:00
										 |  |  | 	h, err := i.cfg.Releases.History(start) | 
					
						
							|  |  |  | 	if err != nil || len(h) < 1 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	releaseutil.Reverse(h, releaseutil.SortByRevision) | 
					
						
							|  |  |  | 	rel := h[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if st := rel.Info.Status; i.Replace && (st == release.StatusUninstalled || st == release.StatusFailed) { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return errors.New("cannot re-use a name that is still in use") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // createRelease creates a new release object
 | 
					
						
							|  |  |  | func (i *Install) createRelease(chrt *chart.Chart, rawVals map[string]interface{}) *release.Release { | 
					
						
							|  |  |  | 	ts := i.cfg.Now() | 
					
						
							|  |  |  | 	return &release.Release{ | 
					
						
							|  |  |  | 		Name:      i.ReleaseName, | 
					
						
							|  |  |  | 		Namespace: i.Namespace, | 
					
						
							|  |  |  | 		Chart:     chrt, | 
					
						
							|  |  |  | 		Config:    rawVals, | 
					
						
							|  |  |  | 		Info: &release.Info{ | 
					
						
							|  |  |  | 			FirstDeployed: ts, | 
					
						
							|  |  |  | 			LastDeployed:  ts, | 
					
						
							|  |  |  | 			Status:        release.StatusUnknown, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Version: 1, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // recordRelease with an update operation in case reuse has been set.
 | 
					
						
							|  |  |  | func (i *Install) recordRelease(r *release.Release) error { | 
					
						
							|  |  |  | 	// This is a legacy function which has been reduced to a oneliner. Could probably
 | 
					
						
							|  |  |  | 	// refactor it out.
 | 
					
						
							|  |  |  | 	return i.cfg.Releases.Update(r) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // replaceRelease replaces an older release with this one
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This allows us to re-use names by superseding an existing release with a new one
 | 
					
						
							|  |  |  | func (i *Install) replaceRelease(rel *release.Release) error { | 
					
						
							|  |  |  | 	hist, err := i.cfg.Releases.History(rel.Name) | 
					
						
							|  |  |  | 	if err != nil || len(hist) == 0 { | 
					
						
							|  |  |  | 		// No releases exist for this name, so we can return early
 | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	releaseutil.Reverse(hist, releaseutil.SortByRevision) | 
					
						
							|  |  |  | 	last := hist[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Update version to the next available
 | 
					
						
							|  |  |  | 	rel.Version = last.Version + 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Do not change the status of a failed release.
 | 
					
						
							|  |  |  | 	if last.Info.Status == release.StatusFailed { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// For any other status, mark it as superseded and store the old record
 | 
					
						
							|  |  |  | 	last.SetStatus(release.StatusSuperseded, "superseded by new release") | 
					
						
							|  |  |  | 	return i.recordRelease(last) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-12 21:31:34 +08:00
										 |  |  | // write the <data> to <output-dir>/<name>. <append> controls if the file is created or content will be appended
 | 
					
						
							|  |  |  | func writeToFile(outputDir string, name string, data string, append bool) error { | 
					
						
							| 
									
										
										
										
											2019-05-20 16:46:42 +08:00
										 |  |  | 	outfileName := strings.Join([]string{outputDir, name}, string(filepath.Separator)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := ensureDirectoryForFile(outfileName) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-12 21:31:34 +08:00
										 |  |  | 	f, err := createOrOpenFile(outfileName, append) | 
					
						
							| 
									
										
										
										
											2019-05-20 16:46:42 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer f.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-12 21:31:34 +08:00
										 |  |  | 	_, err = f.WriteString(fmt.Sprintf("---\n# Source: %s\n%s\n", name, data)) | 
					
						
							| 
									
										
										
										
											2019-05-20 16:46:42 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fmt.Printf("wrote %s\n", outfileName) | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-12 21:31:34 +08:00
										 |  |  | func createOrOpenFile(filename string, append bool) (*os.File, error) { | 
					
						
							|  |  |  | 	if append { | 
					
						
							|  |  |  | 		return os.OpenFile(filename, os.O_APPEND|os.O_WRONLY, 0600) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return os.Create(filename) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 16:46:42 +08:00
										 |  |  | // check if the directory exists to create file. creates if don't exists
 | 
					
						
							|  |  |  | func ensureDirectoryForFile(file string) error { | 
					
						
							|  |  |  | 	baseDir := path.Dir(file) | 
					
						
							|  |  |  | 	_, err := os.Stat(baseDir) | 
					
						
							|  |  |  | 	if err != nil && !os.IsNotExist(err) { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return os.MkdirAll(baseDir, defaultDirectoryPermission) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | // NameAndChart returns the name and chart that should be used.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This will read the flags and handle name generation if necessary.
 | 
					
						
							|  |  |  | func (i *Install) NameAndChart(args []string) (string, string, error) { | 
					
						
							|  |  |  | 	flagsNotSet := func() error { | 
					
						
							|  |  |  | 		if i.GenerateName { | 
					
						
							|  |  |  | 			return errors.New("cannot set --generate-name and also specify a name") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i.NameTemplate != "" { | 
					
						
							|  |  |  | 			return errors.New("cannot set --name-template and also specify a name") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-17 01:14:06 +08:00
										 |  |  | 	if len(args) > 2 { | 
					
						
							|  |  |  | 		return args[0], args[1], errors.Errorf("expected at most two arguments, unexpected arguments: %v", strings.Join(args[2:], ", ")) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	if len(args) == 2 { | 
					
						
							|  |  |  | 		return args[0], args[1], flagsNotSet() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if i.NameTemplate != "" { | 
					
						
							|  |  |  | 		name, err := TemplateName(i.NameTemplate) | 
					
						
							|  |  |  | 		return name, args[0], err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if i.ReleaseName != "" { | 
					
						
							|  |  |  | 		return i.ReleaseName, args[0], nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !i.GenerateName { | 
					
						
							|  |  |  | 		return "", args[0], errors.New("must either provide a name or specify --generate-name") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	base := filepath.Base(args[0]) | 
					
						
							|  |  |  | 	if base == "." || base == "" { | 
					
						
							|  |  |  | 		base = "chart" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-10-31 04:09:43 +08:00
										 |  |  | 	// if present, strip out the file extension from the name
 | 
					
						
							|  |  |  | 	if idx := strings.Index(base, "."); idx != -1 { | 
					
						
							|  |  |  | 		base = base[0:idx] | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return fmt.Sprintf("%s-%d", base, time.Now().Unix()), args[0], nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 06:26:09 +08:00
										 |  |  | // TemplateName renders a name template, returning the name or an error.
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | func TemplateName(nameTemplate string) (string, error) { | 
					
						
							|  |  |  | 	if nameTemplate == "" { | 
					
						
							|  |  |  | 		return "", nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t, err := template.New("name-template").Funcs(sprig.TxtFuncMap()).Parse(nameTemplate) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return "", err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var b bytes.Buffer | 
					
						
							|  |  |  | 	if err := t.Execute(&b, nil); err != nil { | 
					
						
							|  |  |  | 		return "", err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return b.String(), nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 06:26:09 +08:00
										 |  |  | // CheckDependencies checks the dependencies for a chart.
 | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | func CheckDependencies(ch *chart.Chart, reqs []*chart.Dependency) error { | 
					
						
							|  |  |  | 	var missing []string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OUTER: | 
					
						
							|  |  |  | 	for _, r := range reqs { | 
					
						
							|  |  |  | 		for _, d := range ch.Dependencies() { | 
					
						
							|  |  |  | 			if d.Name() == r.Name { | 
					
						
							|  |  |  | 				continue OUTER | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		missing = append(missing, r.Name) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(missing) > 0 { | 
					
						
							|  |  |  | 		return errors.Errorf("found in Chart.yaml, but missing in charts/ directory: %s", strings.Join(missing, ", ")) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // LocateChart looks for a chart directory in known places, and returns either the full path or an error.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This does not ensure that the chart is well-formed; only that the requested filename exists.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Order of resolution:
 | 
					
						
							|  |  |  | // - relative to current working directory
 | 
					
						
							|  |  |  | // - if path is absolute or begins with '.', error out here
 | 
					
						
							|  |  |  | // - URL
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2019-08-27 01:21:52 +08:00
										 |  |  | // If 'verify' was set on ChartPathOptions, this will attempt to also verify the chart.
 | 
					
						
							| 
									
										
										
										
											2019-08-23 14:31:50 +08:00
										 |  |  | func (c *ChartPathOptions) LocateChart(name string, settings *cli.EnvSettings) (string, error) { | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	name = strings.TrimSpace(name) | 
					
						
							|  |  |  | 	version := strings.TrimSpace(c.Version) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, err := os.Stat(name); err == nil { | 
					
						
							|  |  |  | 		abs, err := filepath.Abs(name) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return abs, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if c.Verify { | 
					
						
							|  |  |  | 			if _, err := downloader.VerifyChart(abs, c.Keyring); err != nil { | 
					
						
							|  |  |  | 				return "", err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return abs, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if filepath.IsAbs(name) || strings.HasPrefix(name, ".") { | 
					
						
							|  |  |  | 		return name, errors.Errorf("path %q not found", name) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dl := downloader.ChartDownloader{ | 
					
						
							| 
									
										
										
										
											2019-01-14 16:11:21 +08:00
										 |  |  | 		Out:     os.Stdout, | 
					
						
							|  |  |  | 		Keyring: c.Keyring, | 
					
						
							|  |  |  | 		Getters: getter.All(settings), | 
					
						
							| 
									
										
										
										
											2019-06-20 05:59:07 +08:00
										 |  |  | 		Options: []getter.Option{ | 
					
						
							|  |  |  | 			getter.WithBasicAuth(c.Username, c.Password), | 
					
						
							| 
									
										
										
										
											2019-12-05 23:21:58 +08:00
										 |  |  | 			getter.WithTLSClientConfig(c.CertFile, c.KeyFile, c.CaFile), | 
					
						
							| 
									
										
										
										
											2019-06-20 05:59:07 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-08-28 18:26:21 +08:00
										 |  |  | 		RepositoryConfig: settings.RepositoryConfig, | 
					
						
							|  |  |  | 		RepositoryCache:  settings.RepositoryCache, | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if c.Verify { | 
					
						
							|  |  |  | 		dl.Verify = downloader.VerifyAlways | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if c.RepoURL != "" { | 
					
						
							|  |  |  | 		chartURL, err := repo.FindChartInAuthRepoURL(c.RepoURL, c.Username, c.Password, name, version, | 
					
						
							|  |  |  | 			c.CertFile, c.KeyFile, c.CaFile, getter.All(settings)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return "", err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		name = chartURL | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-27 01:21:52 +08:00
										 |  |  | 	if err := os.MkdirAll(settings.RepositoryCache, 0755); err != nil { | 
					
						
							| 
									
										
										
										
											2019-08-23 14:31:50 +08:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-27 01:21:52 +08:00
										 |  |  | 	filename, _, err := dl.DownloadTo(name, version, settings.RepositoryCache) | 
					
						
							| 
									
										
										
										
											2019-02-09 08:02:57 +08:00
										 |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		lname, err := filepath.Abs(filename) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return filename, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return lname, nil | 
					
						
							|  |  |  | 	} else if settings.Debug { | 
					
						
							|  |  |  | 		return filename, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return filename, errors.Errorf("failed to download %q (hint: running `helm repo update` may help)", name) | 
					
						
							|  |  |  | } |