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

1766 - Git Tag and Chart Version assumed to match #1892

Merged
merged 18 commits into from
Jul 11, 2023
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
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
6 changes: 4 additions & 2 deletions src/extensions/bigbang/bigbang.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,12 +84,14 @@ func Run(YOLO bool, tmpPaths types.ComponentPaths, c types.ZarfComponent) (types
}
}

bbRepo := fmt.Sprintf("%s@%s", cfg.Repo, cfg.Version)

// Configure helm to pull down the Big Bang chart.
helmCfg := helm.Helm{
Chart: types.ZarfChart{
Name: bb,
Namespace: bb,
URL: cfg.Repo,
URL: bbRepo,
Version: cfg.Version,
ValuesFiles: cfg.ValuesFiles,
GitPath: "./chart",
Expand Down Expand Up @@ -494,7 +496,7 @@ func findImagesforBBChartRepo(repo string, values chartutil.Values) (images []st

chart := types.ZarfChart{
Name: repo,
URL: matches[0],
URL: repo,
Version: matches[1],
GitPath: "chart",
}
Expand Down
2 changes: 2 additions & 0 deletions src/internal/packager/helm/common.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import (
"github.com/defenseunicorns/zarf/src/types"
"helm.sh/helm/v3/pkg/action"
"helm.sh/helm/v3/pkg/chart"
"helm.sh/helm/v3/pkg/cli"
)

// Helm is a config object for working with helm charts.
Expand All @@ -25,6 +26,7 @@ type Helm struct {
Cluster *cluster.Cluster
Cfg *types.PackagerConfig
KubeVersion string
Settings *cli.EnvSettings

actionConfig *action.Configuration
}
Expand Down
107 changes: 76 additions & 31 deletions src/internal/packager/helm/repo.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,12 @@ import (
"fmt"
"os"
"path/filepath"
"strings"

"github.com/defenseunicorns/zarf/src/config"
"github.com/defenseunicorns/zarf/src/internal/packager/git"
"github.com/defenseunicorns/zarf/src/pkg/message"
"github.com/defenseunicorns/zarf/src/pkg/transform"
"helm.sh/helm/v3/pkg/action"
"helm.sh/helm/v3/pkg/cli"
"helm.sh/helm/v3/pkg/registry"
Expand All @@ -22,65 +24,82 @@ import (
"helm.sh/helm/v3/pkg/repo"
)

// PackageChart creates a chart archive from a path to a chart on the host os and builds chart dependencies
func (h *Helm) PackageChart(destination string) error {
cmwylie19 marked this conversation as resolved.
Show resolved Hide resolved
if len(h.Chart.URL) > 0 {
url, refPlain, err := transform.GitTransformURLSplitRef(h.Chart.URL)
cmwylie19 marked this conversation as resolved.
Show resolved Hide resolved
// check if the chart is a git url with a ref (if an error is returned url will be empty)
isGitURL := strings.HasSuffix(url, ".git")
if err != nil {
message.Debugf("unable to parse the url, continuing with %s", h.Chart.URL)
}

if isGitURL {
// if it is a git url append chart version as if its a tag
if refPlain == "" {
h.Chart.URL = fmt.Sprintf("%s@%s", h.Chart.URL, h.Chart.Version)
}

_, err = h.PackageChartFromGit(destination)

if err != nil {
return fmt.Errorf("error creating chart archive, unable to pull the chart from git: %s", err.Error())
}
} else {
h.DownloadPublishedChart(destination)
}

} else {
_, err := h.PackageChartFromLocalFiles(destination)
if err != nil {
return fmt.Errorf("error creating chart archive, unable to package the chart: %s", err.Error())
}
}
return nil
}

// PackageChartFromLocalFiles creates a chart archive from a path to a chart on the host os.
func (h *Helm) PackageChartFromLocalFiles(destination string) string {
func (h *Helm) PackageChartFromLocalFiles(destination string) (string, error) {
spinner := message.NewProgressSpinner("Processing helm chart %s:%s from %s", h.Chart.Name, h.Chart.Version, h.Chart.LocalPath)
defer spinner.Stop()

// Validate the chart
_, err := loader.LoadDir(h.Chart.LocalPath)
if err != nil {
spinner.Fatalf(err, "Validation failed for chart from %s (%s)", h.Chart.LocalPath, err.Error())
spinner.Errorf(err, "Validation failed for chart from %s (%s)", h.Chart.LocalPath, err.Error())
return "", err
}

h.buildChartDependencies(spinner)
client := action.NewPackage()

client.Destination = destination
path, err := client.Run(h.Chart.LocalPath, nil)

if err != nil {
spinner.Fatalf(err, "Helm is unable to save the archive and create the package %s", path)
spinner.Errorf(err, "Helm is unable to save the archive and create the package %s", path)
return "", err
}

spinner.Success()

return path
return path, nil
}

// PackageChartFromGit is a special implementation of chart archiving that supports the https://p1.dso.mil/#/products/big-bang/ model.
func (h *Helm) PackageChartFromGit(destination string) (string, error) {
spinner := message.NewProgressSpinner("Processing helm chart %s", h.Chart.Name)
defer spinner.Stop()

client := action.NewPackage()

// Retrieve the repo containing the chart
gitPath, err := h.DownloadChartFromGitToTemp(spinner)
if err != nil {
return "", err
}
defer os.RemoveAll(gitPath)

// Set the directory for the chart
chartPath := filepath.Join(gitPath, h.Chart.GitPath)

// Validate the chart
if _, err := loader.LoadDir(chartPath); err != nil {
spinner.Errorf(err, "Validation failed for chart %s (%s)", h.Chart.Name, err.Error())
return "", err
}

// Tell helm where to save the archive and create the package
client.Destination = destination
name, err := client.Run(chartPath, nil)
if err != nil {
spinner.Errorf(err, "Helm is unable to save the archive and create the package %s", name)
return "", err
}

spinner.Success()

return name, nil
// Set the directory for the chart and package it
h.Chart.LocalPath = filepath.Join(gitPath, h.Chart.GitPath)
return h.PackageChartFromLocalFiles(destination)
}

// DownloadPublishedChart loads a specific chart version from a remote repo.
Expand Down Expand Up @@ -154,14 +173,40 @@ func (h *Helm) DownloadChartFromGitToTemp(spinner *message.Spinner) (string, err
// Create the Git configuration and download the repo
gitCfg := git.NewWithSpinner(h.Cfg.State.GitServer, spinner)

gitRepoWithRef := fmt.Sprintf("%s@%s", h.Chart.URL, h.Chart.Version)

// Download the git repo to a temporary directory
err := gitCfg.DownloadRepoToTemp(gitRepoWithRef)
err := gitCfg.DownloadRepoToTemp(h.Chart.URL)
if err != nil {
spinner.Errorf(err, "Unable to download the git repo %s", gitRepoWithRef)
spinner.Errorf(err, "Unable to download the git repo %s", h.Chart.URL)
return "", err
}

return gitCfg.GitPath, nil
}

// buildChartDependencies builds the helm chart dependencies
func (h *Helm) buildChartDependencies(spinner *message.Spinner) error {
regClient, err := registry.NewClient(registry.ClientOptEnableCache(true))
if err != nil {
spinner.Fatalf(err, "Unable to create a new registry client")
}
h.Settings = cli.New()
man := &downloader.Manager{
Out: os.Stdout,
ChartPath: h.Chart.LocalPath,
Getters: getter.All(h.Settings),
RegistryClient: regClient,

RepositoryConfig: h.Settings.RepositoryConfig,
RepositoryCache: h.Settings.RepositoryCache,
Debug: false,
}
// Verify the chart
man.Verify = downloader.VerifyIfPossible

// Build the deps from the helm chart
err = man.Build()
if e, ok := err.(downloader.ErrRepoNotFound); ok {
return fmt.Errorf("%s. Please add the missing repos via 'helm repo add'", e.Error())
}
return nil
}
7 changes: 4 additions & 3 deletions src/internal/packager/helm/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -66,13 +66,14 @@ func (h *Helm) parseChartValues() (map[string]any, error) {
func (h *Helm) createActionConfig(namespace string, spinner *message.Spinner) error {
// Initialize helm SDK
actionConfig := new(action.Configuration)
settings := cli.New()
// Set the setings for the helm SDK
h.Settings = cli.New()

// Set the namespace for helm
settings.SetNamespace(namespace)
h.Settings.SetNamespace(namespace)

// Setup K8s connection
err := actionConfig.Init(settings.RESTClientGetter(), namespace, "", spinner.Updatef)
err := actionConfig.Init(h.Settings.RESTClientGetter(), namespace, "", spinner.Updatef)

// Set the actionConfig is the received Helm pointer
h.actionConfig = actionConfig
Expand Down
18 changes: 4 additions & 14 deletions src/pkg/packager/create.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import (
"fmt"
"os"
"path/filepath"
"regexp"
"strconv"
"strings"
"time"
Expand Down Expand Up @@ -326,8 +325,7 @@ func (p *Packager) addComponent(index int, component types.ZarfComponent, isSkel

// If any helm charts are defined, process them.
for chartIdx, chart := range component.Charts {
re := regexp.MustCompile(`\.git$`)
isGitURL := re.MatchString(chart.URL)

helmCfg := helm.Helm{
Chart: chart,
Cfg: p.cfg,
Expand All @@ -343,18 +341,10 @@ func (p *Packager) addComponent(index int, component types.ZarfComponent, isSkel
}

p.cfg.Pkg.Components[index].Charts[chartIdx].LocalPath = rel
} else if isGitURL {
_, err = helmCfg.PackageChartFromGit(componentPath.Charts)
if err != nil {
return fmt.Errorf("error creating chart archive, unable to pull the chart from git: %s", err.Error())
}
} else if len(chart.URL) > 0 {
helmCfg.DownloadPublishedChart(componentPath.Charts)
} else {
path := helmCfg.PackageChartFromLocalFiles(componentPath.Charts)
zarfFilename := fmt.Sprintf("%s-%s.tgz", chart.Name, chart.Version)
if !strings.HasSuffix(path, zarfFilename) {
return fmt.Errorf("error creating chart archive, user provided chart name and/or version does not match given chart")
err := helmCfg.PackageChart(componentPath.Charts)
if err != nil {
return err
}
}

Expand Down
18 changes: 5 additions & 13 deletions src/pkg/packager/prepare.go
Original file line number Diff line number Diff line change
Expand Up @@ -113,27 +113,19 @@ func (p *Packager) FindImages(baseDir, repoHelmChartPath string, kubeVersionOver
if len(component.Charts) > 0 {
_ = utils.CreateDirectory(componentPath.Charts, 0700)
_ = utils.CreateDirectory(componentPath.Values, 0700)
re := regexp.MustCompile(`\.git$`)

for _, chart := range component.Charts {
isGitURL := re.MatchString(chart.URL)

helmCfg := helm.Helm{
Chart: chart,
Cfg: p.cfg,
}

helmCfg.Cfg.State = types.ZarfState{}
if isGitURL {
path, err := helmCfg.PackageChartFromGit(componentPath.Charts)
if err != nil {
return fmt.Errorf("unable to download chart from git repo (%s): %w", chart.URL, err)
}
// track the actual chart path
chartOverrides[chart.Name] = path
} else if len(chart.URL) > 0 {
helmCfg.DownloadPublishedChart(componentPath.Charts)
} else {
helmCfg.PackageChartFromLocalFiles(componentPath.Charts)

err := helmCfg.PackageChart(componentPath.Charts)
if err != nil {
return fmt.Errorf("unable to package the chart %s: %s", chart.URL, err.Error())
}

for idx, path := range chart.ValuesFiles {
Expand Down
4 changes: 2 additions & 2 deletions src/pkg/transform/git.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,8 @@ import (
"github.com/defenseunicorns/zarf/src/pkg/utils/helpers"
)

// For further explanation: https://regex101.com/r/YxpfhC/3
var gitURLRegex = regexp.MustCompile(`^(?P<proto>[a-z]+:\/\/)(?P<hostPath>.+?)\/(?P<repo>[\w\-\.]+?)(?P<git>\.git)?(?P<atRef>@(?P<force>\+)?(?P<ref>[\/\+\w\-\.]+))?(?P<gitPath>\/(?P<gitPathId>info\/.*|git-upload-pack|git-receive-pack))?$`)
// For further explanation: https://regex101.com/r/YxpfhC/5
var gitURLRegex = regexp.MustCompile(`^(?P<proto>[a-z]+:\/\/)(?P<hostPath>.+?)\/(?P<repo>[\w\-\.]+?)?(?P<git>\.git)?(\/)?(?P<atRef>@(?P<force>\+)?(?P<ref>[\/\+\w\-\.]+))?(?P<gitPath>\/(?P<gitPathId>info\/.*|git-upload-pack|git-receive-pack))?$`)
cmwylie19 marked this conversation as resolved.
Show resolved Hide resolved

// MutateGitURLsInText changes the gitURL hostname to use the repository Zarf is configured to use.
func MutateGitURLsInText(logger Log, targetBaseURL string, text string, pushUser string) string {
Expand Down