From e30e4604bbabd8924000d08d77e49e90d2f0f759 Mon Sep 17 00:00:00 2001 From: Philip Laine Date: Wed, 9 Oct 2024 10:58:53 +0200 Subject: [PATCH 1/2] refactor: normal creator Signed-off-by: Philip Laine --- examples/manifests/zarf.yaml | 2 +- src/cmd/package.go | 23 +- src/internal/packager2/actions/actions.go | 320 +++++++++ src/internal/packager2/create.go | 72 ++ src/internal/packager2/filters/deploy.go | 189 ++++++ src/internal/packager2/filters/deploy_test.go | 212 ++++++ src/internal/packager2/filters/diff.go | 65 ++ src/internal/packager2/filters/diff_test.go | 68 ++ src/internal/packager2/filters/empty.go | 20 + src/internal/packager2/filters/empty_test.go | 32 + src/internal/packager2/filters/os.go | 39 ++ src/internal/packager2/filters/os_test.go | 39 ++ src/internal/packager2/filters/select.go | 41 ++ src/internal/packager2/filters/select_test.go | 92 +++ src/internal/packager2/filters/strat.go | 41 ++ src/internal/packager2/filters/strat_test.go | 59 ++ src/internal/packager2/filters/utils.go | 42 ++ src/internal/packager2/filters/utils_test.go | 87 +++ src/internal/packager2/layout/create.go | 616 +++++++++++++++++- src/internal/packager2/layout/create_test.go | 4 +- src/internal/packager2/layout/import.go | 7 +- src/internal/packager2/layout/oci.go | 14 +- src/internal/packager2/layout/package.go | 35 + src/internal/packager2/layout/sbom.go | 334 ++++++++++ src/internal/packager2/layout/sbom_test.go | 27 + .../zarf-skeleton-package/archive.tar | Bin 0 -> 20480 bytes .../zarf-skeleton-package/chart/.helmignore | 21 + .../zarf-skeleton-package/chart/Chart.yaml | 13 + .../zarf-skeleton-package/chart/LICENSE | 201 ++++++ .../zarf-skeleton-package/chart/NOTICE | 1 + .../chart/templates/NOTES.txt | 20 + .../chart/templates/_helpers.tpl | 69 ++ .../chart/templates/deployment.yaml | 205 ++++++ .../chart/templates/hpa.yaml | 41 ++ .../chart/templates/ingress.yaml | 41 ++ .../chart/templates/service.yaml | 36 + .../chart/templates/serviceaccount.yaml | 12 + .../zarf-skeleton-package/chart/values.yaml | 164 +++++ .../testdata/zarf-skeleton-package/data.txt | 1 + .../zarf-skeleton-package/deployment.yaml | 21 + .../zarf-skeleton-package/injection/data.txt | 1 + .../kustomize/kustomization.yaml | 2 + .../kustomize/namespace.yaml | 4 + .../zarf-skeleton-package/values.yaml | 5 + .../testdata/zarf-skeleton-package/zarf.yaml | 41 ++ .../packager2/layout/viewer/common.js | 56 ++ .../packager2/layout/viewer/compare.gohtml | 292 +++++++++ .../packager2/layout/viewer/compare.js | 126 ++++ .../packager2/layout/viewer/library.js | 8 + .../packager2/layout/viewer/styles.css | 242 +++++++ .../packager2/layout/viewer/template.gohtml | 277 ++++++++ .../packager2/layout/viewer/theme.css | 173 +++++ .../packager2/layout/viewer/viewer.js | 51 ++ src/test/e2e/00_use_cli_test.go | 1 - src/test/e2e/09_component_compose_test.go | 7 +- 55 files changed, 4561 insertions(+), 51 deletions(-) create mode 100644 src/internal/packager2/actions/actions.go create mode 100644 src/internal/packager2/create.go create mode 100644 src/internal/packager2/filters/deploy.go create mode 100644 src/internal/packager2/filters/deploy_test.go create mode 100644 src/internal/packager2/filters/diff.go create mode 100644 src/internal/packager2/filters/diff_test.go create mode 100644 src/internal/packager2/filters/empty.go create mode 100644 src/internal/packager2/filters/empty_test.go create mode 100644 src/internal/packager2/filters/os.go create mode 100644 src/internal/packager2/filters/os_test.go create mode 100644 src/internal/packager2/filters/select.go create mode 100644 src/internal/packager2/filters/select_test.go create mode 100644 src/internal/packager2/filters/strat.go create mode 100644 src/internal/packager2/filters/strat_test.go create mode 100644 src/internal/packager2/filters/utils.go create mode 100644 src/internal/packager2/filters/utils_test.go create mode 100644 src/internal/packager2/layout/sbom.go create mode 100644 src/internal/packager2/layout/sbom_test.go create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/archive.tar create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/.helmignore create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/Chart.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/LICENSE create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/NOTICE create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/NOTES.txt create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/_helpers.tpl create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/deployment.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/hpa.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/ingress.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/service.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/serviceaccount.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/values.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/data.txt create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/deployment.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/injection/data.txt create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/kustomization.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/namespace.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/values.yaml create mode 100644 src/internal/packager2/layout/testdata/zarf-skeleton-package/zarf.yaml create mode 100644 src/internal/packager2/layout/viewer/common.js create mode 100644 src/internal/packager2/layout/viewer/compare.gohtml create mode 100644 src/internal/packager2/layout/viewer/compare.js create mode 100644 src/internal/packager2/layout/viewer/library.js create mode 100644 src/internal/packager2/layout/viewer/styles.css create mode 100644 src/internal/packager2/layout/viewer/template.gohtml create mode 100644 src/internal/packager2/layout/viewer/theme.css create mode 100644 src/internal/packager2/layout/viewer/viewer.js diff --git a/examples/manifests/zarf.yaml b/examples/manifests/zarf.yaml index 7855f0a02b..6c97e00abd 100644 --- a/examples/manifests/zarf.yaml +++ b/examples/manifests/zarf.yaml @@ -39,7 +39,7 @@ components: kustomizations: # kustomizations can be specified relative to the `zarf.yaml` or as remoteBuild resources with the # following syntax: https://github.com/kubernetes-sigs/kustomize/blob/master/examples/remoteBuild.md: - - github.com/stefanprodan/podinfo//kustomize?ref=6.4.0 + - https://github.com/stefanprodan/podinfo//kustomize?ref=6.4.0 # while ?ref= is not a requirement, it is recommended to use a specific commit hash / git tag to # ensure that the kustomization is not changed in a way that breaks your deployment. # image discovery is supported in all manifests and charts using: diff --git a/src/cmd/package.go b/src/cmd/package.go index bac5d45d48..fa0eb05638 100644 --- a/src/cmd/package.go +++ b/src/cmd/package.go @@ -139,16 +139,19 @@ func (o *PackageCreateOptions) Run(cmd *cobra.Command, args []string) error { pkgConfig.CreateOpts.SetVariables = helpers.TransformAndMergeMap( v.GetStringMapString(common.VPkgCreateSet), pkgConfig.CreateOpts.SetVariables, strings.ToUpper) - pkgClient, err := packager.New(&pkgConfig, - packager.WithContext(ctx), - ) - if err != nil { - return err - } - defer pkgClient.ClearTempPaths() - - err = pkgClient.Create(ctx) - + opt := packager2.CreateOptions{ + Flavor: pkgConfig.CreateOpts.Flavor, + RegistryOverrides: pkgConfig.CreateOpts.RegistryOverrides, + SigningKeyPath: pkgConfig.CreateOpts.SigningKeyPath, + SigningKeyPassword: pkgConfig.CreateOpts.SigningKeyPassword, + SetVariables: pkgConfig.CreateOpts.SetVariables, + MaxPackageSizeMB: pkgConfig.CreateOpts.MaxPackageSizeMB, + SBOMOut: pkgConfig.CreateOpts.SBOMOutputDir, + SkipSBOM: pkgConfig.CreateOpts.SkipSBOM, + Output: pkgConfig.CreateOpts.Output, + DifferentialPackagePath: pkgConfig.CreateOpts.DifferentialPackagePath, + } + err := packager2.Create(cmd.Context(), pkgConfig.CreateOpts.BaseDir, opt) // NOTE(mkcp): LintErrors are rendered with a table var lintErr *lint.LintError if errors.As(err, &lintErr) { diff --git a/src/internal/packager2/actions/actions.go b/src/internal/packager2/actions/actions.go new file mode 100644 index 0000000000..cf831cb2b2 --- /dev/null +++ b/src/internal/packager2/actions/actions.go @@ -0,0 +1,320 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package actions contains functions for running component actions within Zarf packages. +package actions + +import ( + "context" + "fmt" + "path/filepath" + "regexp" + "runtime" + "strings" + "time" + + "github.com/defenseunicorns/pkg/helpers/v2" + "github.com/zarf-dev/zarf/src/api/v1alpha1" + "github.com/zarf-dev/zarf/src/internal/packager/template" + "github.com/zarf-dev/zarf/src/pkg/logger" + "github.com/zarf-dev/zarf/src/pkg/message" + "github.com/zarf-dev/zarf/src/pkg/utils" + "github.com/zarf-dev/zarf/src/pkg/utils/exec" + "github.com/zarf-dev/zarf/src/pkg/variables" +) + +// Run runs all provided actions. +func Run(ctx context.Context, basePath string, defaultCfg v1alpha1.ZarfComponentActionDefaults, actions []v1alpha1.ZarfComponentAction, variableConfig *variables.VariableConfig) error { + if variableConfig == nil { + variableConfig = template.GetZarfVariableConfig(ctx) + } + + for _, a := range actions { + if err := runAction(ctx, basePath, defaultCfg, a, variableConfig); err != nil { + return err + } + } + return nil +} + +// Run commands that a component has provided. +func runAction(ctx context.Context, basePath string, defaultCfg v1alpha1.ZarfComponentActionDefaults, action v1alpha1.ZarfComponentAction, variableConfig *variables.VariableConfig) error { + var cmdEscaped string + var err error + cmd := action.Cmd + l := logger.From(ctx) + start := time.Now() + + // If the action is a wait, convert it to a command. + if action.Wait != nil { + // If the wait has no timeout, set a default of 5 minutes. + if action.MaxTotalSeconds == nil { + fiveMin := 300 + action.MaxTotalSeconds = &fiveMin + } + + // Convert the wait to a command. + if cmd, err = convertWaitToCmd(ctx, *action.Wait, action.MaxTotalSeconds); err != nil { + return err + } + + // Mute the output because it will be noisy. + t := true + action.Mute = &t + + // Set the max retries to 0. + z := 0 + action.MaxRetries = &z + + // Not used for wait actions. + d := "" + action.Dir = &d + action.Env = []string{} + action.SetVariables = []v1alpha1.Variable{} + } + + if action.Description != "" { + cmdEscaped = action.Description + } else { + cmdEscaped = helpers.Truncate(cmd, 60, false) + } + + // TODO(mkcp): Remove message on logger release + spinner := message.NewProgressSpinner("Running \"%s\"", cmdEscaped) + // Persist the spinner output so it doesn't get overwritten by the command output. + spinner.EnablePreserveWrites() + l.Info("running command", "cmd", cmdEscaped) + + actionDefaults := actionGetCfg(ctx, defaultCfg, action, variableConfig.GetAllTemplates()) + actionDefaults.Dir = filepath.Join(basePath, actionDefaults.Dir) + + if cmd, err = actionCmdMutation(ctx, cmd, actionDefaults.Shell); err != nil { + spinner.Errorf(err, "Error mutating command: %s", cmdEscaped) + l.Error("error mutating command", "cmd", cmdEscaped, "err", err.Error()) + } + + duration := time.Duration(actionDefaults.MaxTotalSeconds) * time.Second + timeout := time.After(duration) + + // Keep trying until the max retries is reached. + // TODO: Refactor using go-retry +retryCmd: + for remaining := actionDefaults.MaxRetries + 1; remaining > 0; remaining-- { + // Perform the action run. + tryCmd := func(ctx context.Context) error { + // Try running the command and continue the retry loop if it fails. + stdout, stderr, err := actionRun(ctx, actionDefaults, cmd, spinner) + if err != nil { + if !actionDefaults.Mute { + l.Warn("action failed", "cmd", cmdEscaped, "stdout", stdout, "stderr", stderr) + } + return err + } + if !actionDefaults.Mute { + l.Info("action succeeded", "cmd", cmdEscaped, "stdout", stdout, "stderr", stderr) + } + + outTrimmed := strings.TrimSpace(stdout) + + // If an output variable is defined, set it. + for _, v := range action.SetVariables { + variableConfig.SetVariable(v.Name, outTrimmed, v.Sensitive, v.AutoIndent, v.Type) + if err := variableConfig.CheckVariablePattern(v.Name, v.Pattern); err != nil { + return err + } + } + + // If the action has a wait, change the spinner message to reflect that on success. + if action.Wait != nil { + spinner.Successf("Wait for \"%s\" succeeded", cmdEscaped) + l.Debug("wait for action succeeded", "cmd", cmdEscaped, "duration", time.Since(start)) + return nil + } + + spinner.Successf("Completed \"%s\"", cmdEscaped) + l.Debug("completed action", "cmd", cmdEscaped, "duration", time.Since(start)) + + // If the command ran successfully, continue to the next action. + return nil + } + + // If no timeout is set, run the command and return or continue retrying. + if actionDefaults.MaxTotalSeconds < 1 { + spinner.Updatef("Waiting for \"%s\" (no timeout)", cmdEscaped) + l.Info("waiting for action (no timeout)", "cmd", cmdEscaped) + if err := tryCmd(ctx); err != nil { + continue retryCmd + } + + return nil + } + + // Run the command on repeat until success or timeout. + spinner.Updatef("Waiting for \"%s\" (timeout: %ds)", cmdEscaped, actionDefaults.MaxTotalSeconds) + l.Info("waiting for action", "cmd", cmdEscaped, "timeout", actionDefaults.MaxTotalSeconds) + select { + // On timeout break the loop to abort. + case <-timeout: + break retryCmd + + // Otherwise, try running the command. + default: + ctx, cancel := context.WithTimeout(ctx, duration) + defer cancel() + if err := tryCmd(ctx); err != nil { + continue retryCmd + } + + return nil + } + } + + select { + case <-timeout: + // If we reached this point, the timeout was reached or command failed with no retries. + if actionDefaults.MaxTotalSeconds < 1 { + return fmt.Errorf("command %q failed after %d retries", cmdEscaped, actionDefaults.MaxRetries) + } else { + return fmt.Errorf("command %q timed out after %d seconds", cmdEscaped, actionDefaults.MaxTotalSeconds) + } + default: + // If we reached this point, the retry limit was reached. + return fmt.Errorf("command %q failed after %d retries", cmdEscaped, actionDefaults.MaxRetries) + } +} + +// convertWaitToCmd will return the wait command if it exists, otherwise it will return the original command. +func convertWaitToCmd(_ context.Context, wait v1alpha1.ZarfComponentActionWait, timeout *int) (string, error) { + // Build the timeout string. + timeoutString := fmt.Sprintf("--timeout %ds", *timeout) + + // If the action has a wait, build a cmd from that instead. + cluster := wait.Cluster + if cluster != nil { + ns := cluster.Namespace + if ns != "" { + ns = fmt.Sprintf("-n %s", ns) + } + + // Build a call to the zarf tools wait-for command. + return fmt.Sprintf("./zarf tools wait-for %s %s %s %s %s", + cluster.Kind, cluster.Name, cluster.Condition, ns, timeoutString), nil + } + + network := wait.Network + if network != nil { + // Make sure the protocol is lower case. + network.Protocol = strings.ToLower(network.Protocol) + + // If the protocol is http and no code is set, default to 200. + if strings.HasPrefix(network.Protocol, "http") && network.Code == 0 { + network.Code = 200 + } + + // Build a call to the zarf tools wait-for command. + return fmt.Sprintf("./zarf tools wait-for %s %s %d %s", + network.Protocol, network.Address, network.Code, timeoutString), nil + } + + return "", fmt.Errorf("wait action is missing a cluster or network") +} + +// Perform some basic string mutations to make commands more useful. +func actionCmdMutation(ctx context.Context, cmd string, shellPref v1alpha1.Shell) (string, error) { + zarfCommand, err := utils.GetFinalExecutableCommand() + if err != nil { + return cmd, err + } + + // Try to patch the zarf binary path in case the name isn't exactly "./zarf". + cmd = strings.ReplaceAll(cmd, "./zarf ", zarfCommand+" ") + + // Make commands 'more' compatible with Windows OS PowerShell + if runtime.GOOS == "windows" && (exec.IsPowershell(shellPref.Windows) || shellPref.Windows == "") { + // Replace "touch" with "New-Item" on Windows as it's a common command, but not POSIX so not aliased by M$. + // See https://mathieubuisson.github.io/powershell-linux-bash/ & + // http://web.cs.ucla.edu/~miryung/teaching/EE461L-Spring2012/labs/posix.html for more details. + cmd = regexp.MustCompile(`^touch `).ReplaceAllString(cmd, `New-Item `) + + // Convert any ${ZARF_VAR_*} or $ZARF_VAR_* to ${env:ZARF_VAR_*} or $env:ZARF_VAR_* respectively (also TF_VAR_*). + // https://regex101.com/r/xk1rkw/1 + envVarRegex := regexp.MustCompile(`(?P\${?(?P(ZARF|TF)_VAR_([a-zA-Z0-9_-])+)}?)`) + get, err := helpers.MatchRegex(envVarRegex, cmd) + if err == nil { + newCmd := strings.ReplaceAll(cmd, get("envIndicator"), fmt.Sprintf("$Env:%s", get("varName"))) + // TODO(mkcp): Remove message on logger release + message.Debugf("Converted command \"%s\" to \"%s\" t", cmd, newCmd) + logger.From(ctx).Debug("converted command", "cmd", cmd, "newCmd", newCmd) + cmd = newCmd + } + } + + return cmd, nil +} + +// Merge the ActionSet defaults with the action config. +func actionGetCfg(_ context.Context, cfg v1alpha1.ZarfComponentActionDefaults, a v1alpha1.ZarfComponentAction, vars map[string]*variables.TextTemplate) v1alpha1.ZarfComponentActionDefaults { + if a.Mute != nil { + cfg.Mute = *a.Mute + } + + // Default is no timeout, but add a timeout if one is provided. + if a.MaxTotalSeconds != nil { + cfg.MaxTotalSeconds = *a.MaxTotalSeconds + } + + if a.MaxRetries != nil { + cfg.MaxRetries = *a.MaxRetries + } + + if a.Dir != nil { + cfg.Dir = *a.Dir + } + + if len(a.Env) > 0 { + cfg.Env = append(cfg.Env, a.Env...) + } + + if a.Shell != nil { + cfg.Shell = *a.Shell + } + + // Add variables to the environment. + for k, v := range vars { + // Remove # from env variable name. + k = strings.ReplaceAll(k, "#", "") + // Make terraform variables available to the action as TF_VAR_lowercase_name. + k1 := strings.ReplaceAll(strings.ToLower(k), "zarf_var", "TF_VAR") + cfg.Env = append(cfg.Env, fmt.Sprintf("%s=%s", k, v.Value)) + cfg.Env = append(cfg.Env, fmt.Sprintf("%s=%s", k1, v.Value)) + } + + return cfg +} + +func actionRun(ctx context.Context, cfg v1alpha1.ZarfComponentActionDefaults, cmd string, spinner *message.Spinner) (string, string, error) { + l := logger.From(ctx) + shell, shellArgs := exec.GetOSShell(cfg.Shell) + + // TODO(mkcp): Remove message on logger release + message.Debugf("Running command in %s: %s", shell, cmd) + l.Debug("running command", "shell", shell, "cmd", cmd) + + execCfg := exec.Config{ + Env: cfg.Env, + Dir: cfg.Dir, + } + + if !cfg.Mute { + execCfg.Stdout = spinner + execCfg.Stderr = spinner + } + + stdout, stderr, err := exec.CmdWithContext(ctx, execCfg, shell, append(shellArgs, cmd)...) + // Dump final complete output (respect mute to prevent sensitive values from hitting the logs). + if !cfg.Mute { + // TODO(mkcp): Remove message on logger release + message.Debug(cmd, stdout, stderr) + } + return stdout, stderr, err +} diff --git a/src/internal/packager2/create.go b/src/internal/packager2/create.go new file mode 100644 index 0000000000..73171f55c4 --- /dev/null +++ b/src/internal/packager2/create.go @@ -0,0 +1,72 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +package packager2 + +import ( + "context" + + "github.com/defenseunicorns/pkg/helpers/v2" + "github.com/defenseunicorns/pkg/oci" + + "github.com/zarf-dev/zarf/src/config" + layout2 "github.com/zarf-dev/zarf/src/internal/packager2/layout" +) + +type CreateOptions struct { + Flavor string + RegistryOverrides map[string]string + SigningKeyPath string + SigningKeyPassword string + SetVariables map[string]string + MaxPackageSizeMB int + SBOMOut string + SkipSBOM bool + Output string + DifferentialPackagePath string +} + +func Create(ctx context.Context, packagePath string, opt CreateOptions) error { + createOpt := layout2.CreateOptions{ + Flavor: opt.Flavor, + RegistryOverrides: opt.RegistryOverrides, + SigningKeyPath: opt.SigningKeyPath, + SigningKeyPassword: opt.SigningKeyPassword, + SetVariables: opt.SetVariables, + SkipSBOM: opt.SkipSBOM, + DifferentialPackagePath: opt.DifferentialPackagePath, + } + pkgLayout, err := layout2.CreatePackage(ctx, packagePath, createOpt) + if err != nil { + return err + } + defer pkgLayout.Cleanup() + + if helpers.IsOCIURL(opt.Output) { + ref, err := layout2.ReferenceFromMetadata(opt.Output, pkgLayout.Pkg) + if err != nil { + return err + } + remote, err := layout2.NewRemote(ctx, ref, oci.PlatformForArch(config.GetArch())) + if err != nil { + return err + } + err = remote.Push(ctx, pkgLayout, config.CommonOptions.OCIConcurrency) + if err != nil { + return err + } + } else { + err = pkgLayout.Archive(ctx, opt.Output, opt.MaxPackageSizeMB) + if err != nil { + return err + } + } + + if opt.SBOMOut != "" { + _, err := pkgLayout.GetSBOM(opt.SBOMOut) + if err != nil { + return err + } + } + return nil +} diff --git a/src/internal/packager2/filters/deploy.go b/src/internal/packager2/filters/deploy.go new file mode 100644 index 0000000000..4b562d1c92 --- /dev/null +++ b/src/internal/packager2/filters/deploy.go @@ -0,0 +1,189 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "fmt" + "slices" + "strings" + + "github.com/agnivade/levenshtein" + "github.com/defenseunicorns/pkg/helpers/v2" + "github.com/zarf-dev/zarf/src/api/v1alpha1" + "github.com/zarf-dev/zarf/src/pkg/interactive" +) + +// ForDeploy creates a new deployment filter. +func ForDeploy(optionalComponents string, isInteractive bool) ComponentFilterStrategy { + requested := helpers.StringToSlice(optionalComponents) + + return &deploymentFilter{ + requested, + isInteractive, + } +} + +// deploymentFilter is the default filter for deployments. +type deploymentFilter struct { + requestedComponents []string + isInteractive bool +} + +// Errors for the deployment filter. +var ( + ErrMultipleSameGroup = fmt.Errorf("cannot specify multiple components from the same group") + ErrNoDefaultOrSelection = fmt.Errorf("no default or selected component found") + ErrNotFound = fmt.Errorf("no compatible components found") + ErrSelectionCanceled = fmt.Errorf("selection canceled") +) + +// Apply applies the filter. +func (f *deploymentFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + var selectedComponents []v1alpha1.ZarfComponent + groupedComponents := map[string][]v1alpha1.ZarfComponent{} + orderedComponentGroups := []string{} + + // Group the components by Name and Group while maintaining order + for _, component := range pkg.Components { + groupKey := component.Name + if component.DeprecatedGroup != "" { + groupKey = component.DeprecatedGroup + } + + if !slices.Contains(orderedComponentGroups, groupKey) { + orderedComponentGroups = append(orderedComponentGroups, groupKey) + } + + groupedComponents[groupKey] = append(groupedComponents[groupKey], component) + } + + isPartial := len(f.requestedComponents) > 0 && f.requestedComponents[0] != "" + + if isPartial { + matchedRequests := map[string]bool{} + + // NOTE: This does not use forIncludedComponents as it takes group, default and required status into account. + for _, groupKey := range orderedComponentGroups { + var groupDefault *v1alpha1.ZarfComponent + var groupSelected *v1alpha1.ZarfComponent + + for _, component := range groupedComponents[groupKey] { + // Ensure we have a local version of the component to point to (otherwise the pointer might change on us) + component := component + + selectState, matchedRequest := includedOrExcluded(component.Name, f.requestedComponents) + + if !component.IsRequired() { + if selectState == excluded { + // If the component was explicitly excluded, record the match and continue + matchedRequests[matchedRequest] = true + continue + } else if selectState == unknown && component.Default && groupDefault == nil { + // If the component is default but not included or excluded, remember the default + groupDefault = &component + } + } else { + // Force the selectState to included for Required components + selectState = included + } + + if selectState == included { + // If the component was explicitly included, record the match + matchedRequests[matchedRequest] = true + + // Then check for already selected groups + if groupSelected != nil { + return nil, fmt.Errorf("%w: group: %s selected: %s, %s", ErrMultipleSameGroup, component.DeprecatedGroup, groupSelected.Name, component.Name) + } + + // Then append to the final list + selectedComponents = append(selectedComponents, component) + groupSelected = &component + } + } + + // If nothing was selected from a group, handle the default + if groupSelected == nil && groupDefault != nil { + selectedComponents = append(selectedComponents, *groupDefault) + } else if len(groupedComponents[groupKey]) > 1 && groupSelected == nil && groupDefault == nil { + // If no default component was found, give up + componentNames := []string{} + for _, component := range groupedComponents[groupKey] { + componentNames = append(componentNames, component.Name) + } + return nil, fmt.Errorf("%w: choose from %s", ErrNoDefaultOrSelection, strings.Join(componentNames, ", ")) + } + } + + // Check that we have matched against all requests + for _, requestedComponent := range f.requestedComponents { + if _, ok := matchedRequests[requestedComponent]; !ok { + closeEnough := []string{} + for _, c := range pkg.Components { + d := levenshtein.ComputeDistance(c.Name, requestedComponent) + if d <= 5 { + closeEnough = append(closeEnough, c.Name) + } + } + return nil, fmt.Errorf("%w: %s, suggestions (%s)", ErrNotFound, requestedComponent, strings.Join(closeEnough, ", ")) + } + } + } else { + for _, groupKey := range orderedComponentGroups { + group := groupedComponents[groupKey] + if len(group) > 1 { + if f.isInteractive { + component, err := interactive.SelectChoiceGroup(group) + if err != nil { + return nil, fmt.Errorf("%w: %w", ErrSelectionCanceled, err) + } + selectedComponents = append(selectedComponents, component) + } else { + foundDefault := false + componentNames := []string{} + for _, component := range group { + // If the component is default, then use it + if component.Default { + selectedComponents = append(selectedComponents, component) + foundDefault = true + break + } + // Add each component name to the list + componentNames = append(componentNames, component.Name) + } + if !foundDefault { + // If no default component was found, give up + return nil, fmt.Errorf("%w: choose from %s", ErrNoDefaultOrSelection, strings.Join(componentNames, ", ")) + } + } + } else { + component := groupedComponents[groupKey][0] + + if component.IsRequired() { + selectedComponents = append(selectedComponents, component) + continue + } + + if f.isInteractive { + selected, err := interactive.SelectOptionalComponent(component) + if err != nil { + return nil, fmt.Errorf("%w: %w", ErrSelectionCanceled, err) + } + if selected { + selectedComponents = append(selectedComponents, component) + continue + } + } + + if component.Default { + selectedComponents = append(selectedComponents, component) + continue + } + } + } + } + + return selectedComponents, nil +} diff --git a/src/internal/packager2/filters/deploy_test.go b/src/internal/packager2/filters/deploy_test.go new file mode 100644 index 0000000000..f9899de228 --- /dev/null +++ b/src/internal/packager2/filters/deploy_test.go @@ -0,0 +1,212 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "fmt" + "strings" + "testing" + + "github.com/defenseunicorns/pkg/helpers/v2" + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +func componentFromQuery(t *testing.T, q string) v1alpha1.ZarfComponent { + c := v1alpha1.ZarfComponent{ + Name: q, + } + + conditions := strings.Split(q, "&&") + for _, cond := range conditions { + cond = strings.TrimSpace(cond) + switch cond { + case "default=true": + c.Default = true + case "default=false": + c.Default = false + case "required=": + c.Required = nil + case "required=false": + c.Required = helpers.BoolPtr(false) + case "required=true": + c.Required = helpers.BoolPtr(true) + default: + if strings.HasPrefix(cond, "group=") { + c.DeprecatedGroup = cond[6:] + continue + } + if strings.HasPrefix(cond, "idx=") { + continue + } + require.FailNow(t, "unknown condition", "unknown condition %q", cond) + } + } + + return c +} + +func componentMatrix(_ *testing.T) []v1alpha1.ZarfComponent { + var components []v1alpha1.ZarfComponent + + defaultValues := []bool{true, false} + requiredValues := []interface{}{nil, true, false} + // the duplicate groups are intentional + // this is to test group membership + default filtering + groupValues := []string{"", "foo", "foo", "foo", "bar", "bar", "bar"} + + for idx, groupValue := range groupValues { + for _, defaultValue := range defaultValues { + for _, requiredValue := range requiredValues { + name := strings.Builder{} + + // per validate rules, components in groups cannot be required + if requiredValue != nil && requiredValue.(bool) == true && groupValue != "" { + continue + } + + name.WriteString(fmt.Sprintf("required=%v", requiredValue)) + + if groupValue != "" { + name.WriteString(fmt.Sprintf(" && group=%s && idx=%d && default=%t", groupValue, idx, defaultValue)) + } else if defaultValue { + name.WriteString(" && default=true") + } + + if groupValue != "" { + // if there already exists a component in this group that is default, then set the default to false + // otherwise the filter will error + defaultAlreadyExists := false + if defaultValue { + for _, c := range components { + if c.DeprecatedGroup == groupValue && c.Default { + defaultAlreadyExists = true + break + } + } + } + if defaultAlreadyExists { + defaultValue = false + } + } + + c := v1alpha1.ZarfComponent{ + Name: name.String(), + Default: defaultValue, + DeprecatedGroup: groupValue, + } + + if requiredValue != nil { + c.Required = helpers.BoolPtr(requiredValue.(bool)) + } + + components = append(components, c) + } + } + } + + return components +} + +func TestDeployFilter_Apply(t *testing.T) { + possibilities := componentMatrix(t) + + tests := map[string]struct { + pkg v1alpha1.ZarfPackage + optionalComponents string + want []v1alpha1.ZarfComponent + expectedErr error + }{ + "Test when version is less than v0.33.0 w/ no optional components selected": { + pkg: v1alpha1.ZarfPackage{ + Build: v1alpha1.ZarfBuildData{ + Version: "v0.32.0", + }, + Components: possibilities, + }, + optionalComponents: "", + want: []v1alpha1.ZarfComponent{ + componentFromQuery(t, "required= && default=true"), + componentFromQuery(t, "required=true && default=true"), + componentFromQuery(t, "required=false && default=true"), + componentFromQuery(t, "required=true"), + componentFromQuery(t, "required= && group=foo && idx=1 && default=true"), + componentFromQuery(t, "required= && group=bar && idx=4 && default=true"), + }, + }, + "Test when version is less than v0.33.0 w/ some optional components selected": { + pkg: v1alpha1.ZarfPackage{ + Build: v1alpha1.ZarfBuildData{ + Version: "v0.32.0", + }, + Components: possibilities, + }, + optionalComponents: strings.Join([]string{"required=false", "required= && group=bar && idx=5 && default=false", "-required=true"}, ","), + want: []v1alpha1.ZarfComponent{ + componentFromQuery(t, "required= && default=true"), + componentFromQuery(t, "required=true && default=true"), + componentFromQuery(t, "required=false && default=true"), + // while "required=true" was deselected, it is still required + // therefore it should be included + componentFromQuery(t, "required=true"), + componentFromQuery(t, "required=false"), + componentFromQuery(t, "required= && group=foo && idx=1 && default=true"), + componentFromQuery(t, "required= && group=bar && idx=5 && default=false"), + }, + }, + "Test failing when group has no default and no selection was made": { + pkg: v1alpha1.ZarfPackage{ + Build: v1alpha1.ZarfBuildData{ + Version: "v0.32.0", + }, + Components: []v1alpha1.ZarfComponent{ + componentFromQuery(t, "group=foo && default=false"), + componentFromQuery(t, "group=foo && default=false"), + }, + }, + optionalComponents: "", + expectedErr: ErrNoDefaultOrSelection, + }, + "Test failing when multiple are selected from the same group": { + pkg: v1alpha1.ZarfPackage{ + Build: v1alpha1.ZarfBuildData{ + Version: "v0.32.0", + }, + Components: []v1alpha1.ZarfComponent{ + componentFromQuery(t, "group=foo && default=true"), + componentFromQuery(t, "group=foo && default=false"), + }, + }, + optionalComponents: strings.Join([]string{"group=foo && default=false", "group=foo && default=true"}, ","), + expectedErr: ErrMultipleSameGroup, + }, + "Test failing when no components are found that match the query": { + pkg: v1alpha1.ZarfPackage{ + Build: v1alpha1.ZarfBuildData{ + Version: "v0.32.0", + }, + Components: possibilities, + }, + optionalComponents: "nonexistent", + expectedErr: ErrNotFound, + }, + } + + for name, tt := range tests { + t.Run(name, func(t *testing.T) { + // we do not currently support interactive mode in unit tests + isInteractive := false + filter := ForDeploy(tt.optionalComponents, isInteractive) + + result, err := filter.Apply(tt.pkg) + if tt.expectedErr != nil { + require.ErrorIs(t, err, tt.expectedErr) + } else { + require.NoError(t, err) + } + require.Equal(t, tt.want, result) + }) + } +} diff --git a/src/internal/packager2/filters/diff.go b/src/internal/packager2/filters/diff.go new file mode 100644 index 0000000000..dfc2b2f5a9 --- /dev/null +++ b/src/internal/packager2/filters/diff.go @@ -0,0 +1,65 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +package filters + +import ( + "fmt" + + "github.com/go-git/go-git/v5/plumbing" + "github.com/zarf-dev/zarf/src/api/v1alpha1" + "github.com/zarf-dev/zarf/src/internal/git" + "github.com/zarf-dev/zarf/src/pkg/transform" +) + +// ByDifferentialData filters any images and repos already present in the reference package components. +func ByDifferentialData(images map[string]bool, repos map[string]bool) ComponentFilterStrategy { + return &differentialDataFilter{ + images: images, + repos: repos, + } +} + +type differentialDataFilter struct { + images map[string]bool + repos map[string]bool +} + +func (f *differentialDataFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + diffComponents := []v1alpha1.ZarfComponent{} + for _, component := range pkg.Components { + filteredImages := []string{} + for _, img := range component.Images { + imgRef, err := transform.ParseImageRef(img) + if err != nil { + return nil, fmt.Errorf("unable to parse image ref %s: %w", img, err) + } + imgTag := imgRef.TagOrDigest + includeImage := imgTag == ":latest" || imgTag == ":stable" || imgTag == ":nightly" + if includeImage || !f.images[img] { + filteredImages = append(filteredImages, img) + } + } + component.Images = filteredImages + + filteredRepos := []string{} + for _, repoURL := range component.Repos { + _, refPlain, err := transform.GitURLSplitRef(repoURL) + if err != nil { + return nil, err + } + var ref plumbing.ReferenceName + if refPlain != "" { + ref = git.ParseRef(refPlain) + } + includeRepo := ref == "" || (!ref.IsTag() && !plumbing.IsHash(refPlain)) + if includeRepo || !f.repos[repoURL] { + filteredRepos = append(filteredRepos, repoURL) + } + } + component.Repos = filteredRepos + + diffComponents = append(diffComponents, component) + } + return diffComponents, nil +} diff --git a/src/internal/packager2/filters/diff_test.go b/src/internal/packager2/filters/diff_test.go new file mode 100644 index 0000000000..3c686351d0 --- /dev/null +++ b/src/internal/packager2/filters/diff_test.go @@ -0,0 +1,68 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +func TestCopyFilter(t *testing.T) { + pkg := v1alpha1.ZarfPackage{ + Components: []v1alpha1.ZarfComponent{ + { + Images: []string{ + "example.com/include-image-tag:latest", + "example.com/image-with-tag:v1", + "example.com/diff-image-with-tag:v1", + "example.com/image-with-digest@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + "example.com/diff-image-with-digest@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + "example.com/image-with-tag-and-digest:v1@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + "example.com/diff-image-with-tag-and-digest:v1@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + }, + Repos: []string{ + "https://example.com/no-ref.git", + "https://example.com/branch.git@refs/heads/main", + "https://example.com/tag.git@v1", + "https://example.com/diff-tag.git@v1", + "https://example.com/commit.git@524980951ff16e19dc25232e9aea8fd693989ba6", + "https://example.com/diff-commit.git@524980951ff16e19dc25232e9aea8fd693989ba6", + }, + }, + }, + } + differentialImages := map[string]bool{ + "example.com/include-image-tag:latest": true, + "example.com/diff-image-with-tag:v1": true, + "example.com/diff-image-with-digest@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855": true, + "example.com/diff-image-with-tag-and-digest:v1@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855": true, + } + differentialRepos := map[string]bool{ + "https://example.com/no-ref.git": true, + "https://example.com/branch.git@refs/heads/main": true, + "https://example.com/diff-tag.git@v1": true, + "https://example.com/diff-commit.git@524980951ff16e19dc25232e9aea8fd693989ba6": true, + } + + filter := ByDifferentialData(differentialImages, differentialRepos) + diffComponents, err := filter.Apply(pkg) + require.NoError(t, err) + + expectedImages := []string{ + "example.com/include-image-tag:latest", + "example.com/image-with-tag:v1", + "example.com/image-with-digest@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + "example.com/image-with-tag-and-digest:v1@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", + } + require.ElementsMatch(t, expectedImages, diffComponents[0].Images) + expectedRepos := []string{ + "https://example.com/no-ref.git", + "https://example.com/branch.git@refs/heads/main", + "https://example.com/tag.git@v1", + "https://example.com/commit.git@524980951ff16e19dc25232e9aea8fd693989ba6", + } + require.ElementsMatch(t, expectedRepos, diffComponents[0].Repos) +} diff --git a/src/internal/packager2/filters/empty.go b/src/internal/packager2/filters/empty.go new file mode 100644 index 0000000000..4729adc509 --- /dev/null +++ b/src/internal/packager2/filters/empty.go @@ -0,0 +1,20 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import "github.com/zarf-dev/zarf/src/api/v1alpha1" + +// Empty returns a filter that does nothing. +func Empty() ComponentFilterStrategy { + return &emptyFilter{} +} + +// emptyFilter is a filter that does nothing. +type emptyFilter struct{} + +// Apply returns the components unchanged. +func (f *emptyFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + return pkg.Components, nil +} diff --git a/src/internal/packager2/filters/empty_test.go b/src/internal/packager2/filters/empty_test.go new file mode 100644 index 0000000000..2b74597723 --- /dev/null +++ b/src/internal/packager2/filters/empty_test.go @@ -0,0 +1,32 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +func TestEmptyFilter_Apply(t *testing.T) { + components := []v1alpha1.ZarfComponent{ + { + Name: "component1", + }, + { + Name: "component2", + }, + } + pkg := v1alpha1.ZarfPackage{ + Components: components, + } + filter := Empty() + + result, err := filter.Apply(pkg) + + require.NoError(t, err) + require.Equal(t, components, result) +} diff --git a/src/internal/packager2/filters/os.go b/src/internal/packager2/filters/os.go new file mode 100644 index 0000000000..2bc7dffa2e --- /dev/null +++ b/src/internal/packager2/filters/os.go @@ -0,0 +1,39 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "errors" + + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +// ByLocalOS creates a new filter that filters components based on local (runtime) OS. +func ByLocalOS(localOS string) ComponentFilterStrategy { + return &localOSFilter{localOS} +} + +// localOSFilter filters components based on local (runtime) OS. +type localOSFilter struct { + localOS string +} + +// ErrLocalOSRequired is returned when localOS is not set. +var ErrLocalOSRequired = errors.New("localOS is required") + +// Apply applies the filter. +func (f *localOSFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + if f.localOS == "" { + return nil, ErrLocalOSRequired + } + + filtered := []v1alpha1.ZarfComponent{} + for _, component := range pkg.Components { + if component.Only.LocalOS == "" || component.Only.LocalOS == f.localOS { + filtered = append(filtered, component) + } + } + return filtered, nil +} diff --git a/src/internal/packager2/filters/os_test.go b/src/internal/packager2/filters/os_test.go new file mode 100644 index 0000000000..c2e022a752 --- /dev/null +++ b/src/internal/packager2/filters/os_test.go @@ -0,0 +1,39 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" + "github.com/zarf-dev/zarf/src/pkg/lint" +) + +func TestLocalOSFilter(t *testing.T) { + pkg := v1alpha1.ZarfPackage{} + for _, os := range lint.SupportedOS() { + pkg.Components = append(pkg.Components, v1alpha1.ZarfComponent{ + Only: v1alpha1.ZarfComponentOnlyTarget{ + LocalOS: os, + }, + }) + } + + for _, os := range lint.SupportedOS() { + filter := ByLocalOS(os) + result, err := filter.Apply(pkg) + if os == "" { + require.ErrorIs(t, err, ErrLocalOSRequired) + } else { + require.NoError(t, err) + } + for _, component := range result { + if component.Only.LocalOS != "" { + require.Equal(t, os, component.Only.LocalOS) + } + } + } +} diff --git a/src/internal/packager2/filters/select.go b/src/internal/packager2/filters/select.go new file mode 100644 index 0000000000..fafc8c64fa --- /dev/null +++ b/src/internal/packager2/filters/select.go @@ -0,0 +1,41 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "github.com/defenseunicorns/pkg/helpers/v2" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +// BySelectState creates a new simple included filter. +func BySelectState(optionalComponents string) ComponentFilterStrategy { + requested := helpers.StringToSlice(optionalComponents) + + return &selectStateFilter{ + requested, + } +} + +// selectStateFilter sorts based purely on the internal included state of the component. +type selectStateFilter struct { + requestedComponents []string +} + +// Apply applies the filter. +func (f *selectStateFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + isPartial := len(f.requestedComponents) > 0 && f.requestedComponents[0] != "" + result := []v1alpha1.ZarfComponent{} + for _, component := range pkg.Components { + selectState := included + if isPartial { + selectState, _ = includedOrExcluded(component.Name, f.requestedComponents) + } + if selectState != included { + continue + } + result = append(result, component) + } + return result, nil +} diff --git a/src/internal/packager2/filters/select_test.go b/src/internal/packager2/filters/select_test.go new file mode 100644 index 0000000000..a384ce56e8 --- /dev/null +++ b/src/internal/packager2/filters/select_test.go @@ -0,0 +1,92 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +func Test_selectStateFilter_Apply(t *testing.T) { + tests := []struct { + name string + requestedComponents string + components []v1alpha1.ZarfComponent + expectedResult []v1alpha1.ZarfComponent + expectedError error + }{ + { + name: "Test when requestedComponents is empty", + requestedComponents: "", + components: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + {Name: "component3"}, + }, + expectedResult: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + {Name: "component3"}, + }, + expectedError: nil, + }, + { + name: "Test when requestedComponents contains a valid component name", + requestedComponents: "component2", + components: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + {Name: "component3"}, + }, + expectedResult: []v1alpha1.ZarfComponent{ + {Name: "component2"}, + }, + expectedError: nil, + }, + { + name: "Test when requestedComponents contains an excluded component name", + requestedComponents: "comp*, -component2", + components: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + {Name: "component3"}, + }, + expectedResult: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component3"}, + }, + expectedError: nil, + }, + { + name: "Test when requestedComponents contains a glob pattern", + requestedComponents: "comp*", + components: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + {Name: "other"}, + }, + expectedResult: []v1alpha1.ZarfComponent{ + {Name: "component1"}, + {Name: "component2"}, + }, + expectedError: nil, + }, + } + + for _, tc := range tests { + t.Run(tc.name, func(t *testing.T) { + filter := BySelectState(tc.requestedComponents) + + result, err := filter.Apply(v1alpha1.ZarfPackage{ + Components: tc.components, + }) + + require.Equal(t, tc.expectedResult, result) + require.Equal(t, tc.expectedError, err) + }) + } +} diff --git a/src/internal/packager2/filters/strat.go b/src/internal/packager2/filters/strat.go new file mode 100644 index 0000000000..b63f39bd42 --- /dev/null +++ b/src/internal/packager2/filters/strat.go @@ -0,0 +1,41 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "fmt" + + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +// ComponentFilterStrategy is a strategy interface for filtering components. +type ComponentFilterStrategy interface { + Apply(v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) +} + +// comboFilter is a filter that applies a sequence of filters. +type comboFilter struct { + filters []ComponentFilterStrategy +} + +// Apply applies the filter. +func (f *comboFilter) Apply(pkg v1alpha1.ZarfPackage) ([]v1alpha1.ZarfComponent, error) { + result := pkg + + for _, filter := range f.filters { + components, err := filter.Apply(result) + if err != nil { + return nil, fmt.Errorf("error applying filter %T: %w", filter, err) + } + result.Components = components + } + + return result.Components, nil +} + +// Combine creates a new filter that applies a sequence of filters. +func Combine(filters ...ComponentFilterStrategy) ComponentFilterStrategy { + return &comboFilter{filters} +} diff --git a/src/internal/packager2/filters/strat_test.go b/src/internal/packager2/filters/strat_test.go new file mode 100644 index 0000000000..69c39beff9 --- /dev/null +++ b/src/internal/packager2/filters/strat_test.go @@ -0,0 +1,59 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/api/v1alpha1" +) + +func TestCombine(t *testing.T) { + f1 := BySelectState("*a*") + f2 := BySelectState("*bar, foo") + f3 := Empty() + + combo := Combine(f1, f2, f3) + + pkg := v1alpha1.ZarfPackage{ + Components: []v1alpha1.ZarfComponent{ + { + Name: "foo", + }, + { + Name: "bar", + }, + { + Name: "baz", + }, + { + Name: "foobar", + }, + }, + } + + expected := []v1alpha1.ZarfComponent{ + { + Name: "bar", + }, + { + Name: "foobar", + }, + } + + result, err := combo.Apply(pkg) + require.NoError(t, err) + require.Equal(t, expected, result) + + // Test error propagation + combo = Combine(f1, f2, ForDeploy("group with no default", false)) + pkg.Components = append(pkg.Components, v1alpha1.ZarfComponent{ + Name: "group with no default", + DeprecatedGroup: "g1", + }) + _, err = combo.Apply(pkg) + require.Error(t, err) +} diff --git a/src/internal/packager2/filters/utils.go b/src/internal/packager2/filters/utils.go new file mode 100644 index 0000000000..3b33f9b594 --- /dev/null +++ b/src/internal/packager2/filters/utils.go @@ -0,0 +1,42 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "path" + "strings" +) + +type selectState int + +const ( + unknown selectState = iota + included + excluded +) + +func includedOrExcluded(componentName string, requestedComponentNames []string) (selectState, string) { + // Check if the component has a leading dash indicating it should be excluded - this is done first so that exclusions precede inclusions + for _, requestedComponent := range requestedComponentNames { + if strings.HasPrefix(requestedComponent, "-") { + // If the component glob matches one of the requested components, then return true + // This supports globbing with "path" in order to have the same behavior across OSes (if we ever allow namespaced components with /) + if matched, _ := path.Match(strings.TrimPrefix(requestedComponent, "-"), componentName); matched { + return excluded, requestedComponent + } + } + } + // Check if the component matches a glob pattern and should be included + for _, requestedComponent := range requestedComponentNames { + // If the component glob matches one of the requested components, then return true + // This supports globbing with "path" in order to have the same behavior across OSes (if we ever allow namespaced components with /) + if matched, _ := path.Match(requestedComponent, componentName); matched { + return included, requestedComponent + } + } + + // All other cases we don't know if we should include or exclude yet + return unknown, "" +} diff --git a/src/internal/packager2/filters/utils_test.go b/src/internal/packager2/filters/utils_test.go new file mode 100644 index 0000000000..f81b8e105f --- /dev/null +++ b/src/internal/packager2/filters/utils_test.go @@ -0,0 +1,87 @@ +// SPDX-License-Identifier: Apache-2.0 +// SPDX-FileCopyrightText: 2021-Present The Zarf Authors + +// Package filters contains core implementations of the ComponentFilterStrategy interface. +package filters + +import ( + "testing" + + "github.com/stretchr/testify/require" +) + +func Test_includedOrExcluded(t *testing.T) { + tests := []struct { + name string + componentName string + requestedComponentNames []string + wantState selectState + wantRequestedComponent string + }{ + { + name: "Test when component is excluded", + componentName: "example", + requestedComponentNames: []string{"-example"}, + wantState: excluded, + wantRequestedComponent: "-example", + }, + { + name: "Test when component is included", + componentName: "example", + requestedComponentNames: []string{"example"}, + wantState: included, + wantRequestedComponent: "example", + }, + { + name: "Test when component is not included or excluded", + componentName: "example", + requestedComponentNames: []string{"other"}, + wantState: unknown, + wantRequestedComponent: "", + }, + { + name: "Test when component is excluded and included", + componentName: "example", + requestedComponentNames: []string{"-example", "example"}, + wantState: excluded, + wantRequestedComponent: "-example", + }, + // interesting case, excluded wins + { + name: "Test when component is included and excluded", + componentName: "example", + requestedComponentNames: []string{"example", "-example"}, + wantState: excluded, + wantRequestedComponent: "-example", + }, + { + name: "Test when component is included via glob", + componentName: "example", + requestedComponentNames: []string{"ex*"}, + wantState: included, + wantRequestedComponent: "ex*", + }, + { + name: "Test when component is excluded via glob", + componentName: "example", + requestedComponentNames: []string{"-ex*"}, + wantState: excluded, + wantRequestedComponent: "-ex*", + }, + { + name: "Test when component is not found via glob", + componentName: "example", + requestedComponentNames: []string{"other*"}, + wantState: unknown, + wantRequestedComponent: "", + }, + } + + for _, tc := range tests { + t.Run(tc.name, func(t *testing.T) { + gotState, gotRequestedComponent := includedOrExcluded(tc.componentName, tc.requestedComponentNames) + require.Equal(t, tc.wantState, gotState) + require.Equal(t, tc.wantRequestedComponent, gotRequestedComponent) + }) + } +} diff --git a/src/internal/packager2/layout/create.go b/src/internal/packager2/layout/create.go index 5aaa446b4d..d459fc8e72 100644 --- a/src/internal/packager2/layout/create.go +++ b/src/internal/packager2/layout/create.go @@ -8,6 +8,8 @@ import ( "context" "crypto/sha256" "encoding/hex" + "encoding/json" + "errors" "fmt" "io" "io/fs" @@ -28,54 +30,201 @@ import ( "github.com/zarf-dev/zarf/src/api/v1alpha1" "github.com/zarf-dev/zarf/src/config" "github.com/zarf-dev/zarf/src/config/lang" + "github.com/zarf-dev/zarf/src/internal/git" "github.com/zarf-dev/zarf/src/internal/packager/helm" + "github.com/zarf-dev/zarf/src/internal/packager/images" "github.com/zarf-dev/zarf/src/internal/packager/kustomize" + actions2 "github.com/zarf-dev/zarf/src/internal/packager2/actions" + "github.com/zarf-dev/zarf/src/internal/packager2/filters" + "github.com/zarf-dev/zarf/src/pkg/interactive" "github.com/zarf-dev/zarf/src/pkg/lint" + "github.com/zarf-dev/zarf/src/pkg/logger" + "github.com/zarf-dev/zarf/src/pkg/message" + "github.com/zarf-dev/zarf/src/pkg/transform" "github.com/zarf-dev/zarf/src/pkg/utils" "github.com/zarf-dev/zarf/src/pkg/zoci" + "github.com/zarf-dev/zarf/src/types" ) // CreateOptions are the options for creating a skeleton package. type CreateOptions struct { - Flavor string - RegistryOverrides map[string]string - SigningKeyPath string - SigningKeyPassword string - SetVariables map[string]string + Flavor string + RegistryOverrides map[string]string + SigningKeyPath string + SigningKeyPassword string + SetVariables map[string]string + SkipSBOM bool + DifferentialPackagePath string } -// CreateSkeleton creates a skeleton package and returns the path to the created package. -func CreateSkeleton(ctx context.Context, packagePath string, opt CreateOptions) (string, error) { - b, err := os.ReadFile(filepath.Join(packagePath, ZarfYAML)) +func CreatePackage(ctx context.Context, packagePath string, opt CreateOptions) (*PackageLayout, error) { + l := logger.From(ctx) + l.Info("creating package", "path", packagePath) + + buildPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) if err != nil { - return "", err + return nil, err } - var pkg v1alpha1.ZarfPackage - err = goyaml.Unmarshal(b, &pkg) + + pkg, err := loadPackage(ctx, packagePath, opt.Flavor, opt.SetVariables) if err != nil { - return "", err + return nil, err } - buildPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) + + if opt.DifferentialPackagePath != "" { + l.Debug("creating differential package", "differential", opt.DifferentialPackagePath) + layoutOpt := PackageLayoutOptions{ + SkipSignatureValidation: true, + } + diffPkgLayout, err := LoadFromTar(ctx, opt.DifferentialPackagePath, layoutOpt) + if err != nil { + return nil, err + } + allIncludedImagesMap := map[string]bool{} + allIncludedReposMap := map[string]bool{} + for _, component := range diffPkgLayout.Pkg.Components { + for _, image := range component.Images { + allIncludedImagesMap[image] = true + } + for _, repo := range component.Repos { + allIncludedReposMap[repo] = true + } + } + + pkg.Build.Differential = true + pkg.Build.DifferentialPackageVersion = diffPkgLayout.Pkg.Metadata.Version + + versionsMatch := diffPkgLayout.Pkg.Metadata.Version == pkg.Metadata.Version + if versionsMatch { + return nil, errors.New(lang.PkgCreateErrDifferentialSameVersion) + } + noVersionSet := diffPkgLayout.Pkg.Metadata.Version == "" || pkg.Metadata.Version == "" + if noVersionSet { + return nil, errors.New(lang.PkgCreateErrDifferentialNoVersion) + } + filter := filters.ByDifferentialData(allIncludedImagesMap, allIncludedReposMap) + pkg.Components, err = filter.Apply(pkg) + if err != nil { + return nil, err + } + } + + for _, component := range pkg.Components { + err := assemblePackageComponent(ctx, component, packagePath, buildPath) + if err != nil { + return nil, err + } + } + + componentImages := []transform.Image{} + for _, component := range pkg.Components { + for _, src := range component.Images { + refInfo, err := transform.ParseImageRef(src) + if err != nil { + return nil, fmt.Errorf("failed to create ref for image %s: %w", src, err) + } + if slices.Contains(componentImages, refInfo) { + continue + } + componentImages = append(componentImages, refInfo) + } + } + sbomImageList := []transform.Image{} + if len(componentImages) > 0 { + cachePath, err := config.GetAbsCachePath() + if err != nil { + return nil, err + } + pullCfg := images.PullConfig{ + DestinationDirectory: filepath.Join(buildPath, ImagesDir), + ImageList: componentImages, + Arch: pkg.Metadata.Architecture, + RegistryOverrides: opt.RegistryOverrides, + CacheDirectory: filepath.Join(cachePath, ImagesDir), + } + pulled, err := images.Pull(ctx, pullCfg) + if err != nil { + return nil, err + } + for info, img := range pulled { + ok, err := utils.OnlyHasImageLayers(img) + if err != nil { + return nil, fmt.Errorf("failed to validate %s is an image and not an artifact: %w", info, err) + } + if ok { + sbomImageList = append(sbomImageList, info) + } + } + + // Sort images index to make build reproducible. + err = utils.SortImagesIndex(filepath.Join(buildPath, ImagesDir)) + if err != nil { + return nil, err + } + } + + l.Info("composed components successfully") + + if !opt.SkipSBOM { + l.Info("generating SBOM") + err = generateSBOM(ctx, pkg, buildPath, sbomImageList) + if err != nil { + return nil, err + } + } + + checksumContent, checksumSha, err := getChecksum(buildPath) if err != nil { - return "", err + return nil, err } + checksumPath := filepath.Join(buildPath, Checksums) + err = os.WriteFile(checksumPath, []byte(checksumContent), helpers.ReadWriteUser) + if err != nil { + return nil, err + } + pkg.Metadata.AggregateChecksum = checksumSha - pkg.Metadata.Architecture = config.GetArch() + pkg = recordPackageMetadata(pkg, opt.Flavor, opt.RegistryOverrides) - pkg, err = resolveImports(ctx, pkg, packagePath, pkg.Metadata.Architecture, opt.Flavor, map[string]interface{}{}) + b, err := goyaml.Marshal(pkg) if err != nil { - return "", err + return nil, err + } + err = os.WriteFile(filepath.Join(buildPath, ZarfYAML), b, helpers.ReadWriteUser) + if err != nil { + return nil, err } + err = signPackage(buildPath, opt.SigningKeyPath, opt.SigningKeyPassword) + if err != nil { + return nil, err + } + + pkgLayout, err := LoadFromDir(ctx, buildPath, PackageLayoutOptions{SkipSignatureValidation: true}) + if err != nil { + return nil, err + } + + l.Info("package created") + + return pkgLayout, nil +} + +// CreateSkeleton creates a skeleton package and returns the path to the created package. +func CreateSkeleton(ctx context.Context, packagePath string, opt CreateOptions) (string, error) { + pkg, err := loadPackage(ctx, packagePath, opt.Flavor, nil) + if err != nil { + return "", err + } pkg.Metadata.Architecture = zoci.SkeletonArch - err = validate(pkg, packagePath, opt.SetVariables) + buildPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) if err != nil { return "", err } for _, component := range pkg.Components { - err := assembleComponent(component, packagePath, buildPath) + err := assembleSkeletonComponent(component, packagePath, buildPath) if err != nil { return "", err } @@ -94,7 +243,7 @@ func CreateSkeleton(ctx context.Context, packagePath string, opt CreateOptions) pkg = recordPackageMetadata(pkg, opt.Flavor, opt.RegistryOverrides) - b, err = goyaml.Marshal(pkg) + b, err := goyaml.Marshal(pkg) if err != nil { return "", err } @@ -111,6 +260,33 @@ func CreateSkeleton(ctx context.Context, packagePath string, opt CreateOptions) return buildPath, nil } +func loadPackage(ctx context.Context, packagePath, flavor string, setVariables map[string]string) (v1alpha1.ZarfPackage, error) { + b, err := os.ReadFile(filepath.Join(packagePath, ZarfYAML)) + if err != nil { + return v1alpha1.ZarfPackage{}, err + } + pkg, err := ParseZarfPackage(b) + if err != nil { + return v1alpha1.ZarfPackage{}, err + } + pkg.Metadata.Architecture = config.GetArch(pkg.Metadata.Architecture) + pkg, err = resolveImports(ctx, pkg, packagePath, pkg.Metadata.Architecture, flavor, map[string]interface{}{}) + if err != nil { + return v1alpha1.ZarfPackage{}, err + } + if setVariables != nil { + pkg, _, err = fillActiveTemplate(ctx, pkg, setVariables) + if err != nil { + return v1alpha1.ZarfPackage{}, err + } + } + err = validate(pkg, packagePath, setVariables) + if err != nil { + return v1alpha1.ZarfPackage{}, err + } + return pkg, nil +} + func validate(pkg v1alpha1.ZarfPackage, packagePath string, setVariables map[string]string) error { err := lint.ValidatePackage(pkg) if err != nil { @@ -130,7 +306,204 @@ func validate(pkg v1alpha1.ZarfPackage, packagePath string, setVariables map[str } } -func assembleComponent(component v1alpha1.ZarfComponent, packagePath, buildPath string) error { +func assemblePackageComponent(ctx context.Context, component v1alpha1.ZarfComponent, packagePath, buildPath string) error { + tmpBuildPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) + if err != nil { + return err + } + defer os.RemoveAll(tmpBuildPath) + compBuildPath := filepath.Join(tmpBuildPath, component.Name) + err = os.MkdirAll(compBuildPath, 0o700) + if err != nil { + return err + } + + onCreate := component.Actions.OnCreate + if err := actions2.Run(ctx, packagePath, onCreate.Defaults, onCreate.Before, nil); err != nil { + return fmt.Errorf("unable to run component before action: %w", err) + } + + // If any helm charts are defined, process them. + for _, chart := range component.Charts { + // TODO: Refactor helm builder + if chart.LocalPath != "" { + chart.LocalPath = filepath.Join(packagePath, chart.LocalPath) + } + oldValuesFiles := chart.ValuesFiles + valuesFiles := []string{} + for _, v := range chart.ValuesFiles { + valuesFiles = append(valuesFiles, filepath.Join(packagePath, v)) + } + chart.ValuesFiles = valuesFiles + helmCfg := helm.New(chart, filepath.Join(compBuildPath, string(ChartsComponentDir)), filepath.Join(compBuildPath, string(ValuesComponentDir))) + if err := helmCfg.PackageChart(ctx, filepath.Join(compBuildPath, string(ChartsComponentDir))); err != nil { + return err + } + chart.ValuesFiles = oldValuesFiles + } + + for filesIdx, file := range component.Files { + rel := filepath.Join(string(FilesComponentDir), strconv.Itoa(filesIdx), filepath.Base(file.Target)) + dst := filepath.Join(compBuildPath, rel) + destinationDir := filepath.Dir(dst) + + if helpers.IsURL(file.Source) { + if file.ExtractPath != "" { + // get the compressedFileName from the source + compressedFileName, err := helpers.ExtractBasePathFromURL(file.Source) + if err != nil { + return fmt.Errorf(lang.ErrFileNameExtract, file.Source, err.Error()) + } + tmpDir, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) + if err != nil { + return err + } + defer os.RemoveAll(tmpDir) + compressedFile := filepath.Join(tmpDir, compressedFileName) + + // If the file is an archive, download it to the componentPath.Temp + if err := utils.DownloadToFile(ctx, file.Source, compressedFile, component.DeprecatedCosignKeyPath); err != nil { + return fmt.Errorf(lang.ErrDownloading, file.Source, err.Error()) + } + err = archiver.Extract(compressedFile, file.ExtractPath, destinationDir) + if err != nil { + return fmt.Errorf(lang.ErrFileExtract, file.ExtractPath, compressedFileName, err.Error()) + } + } else { + if err := utils.DownloadToFile(ctx, file.Source, dst, component.DeprecatedCosignKeyPath); err != nil { + return fmt.Errorf(lang.ErrDownloading, file.Source, err.Error()) + } + } + } else { + if file.ExtractPath != "" { + if err := archiver.Extract(filepath.Join(packagePath, file.Source), file.ExtractPath, destinationDir); err != nil { + return fmt.Errorf(lang.ErrFileExtract, file.ExtractPath, file.Source, err.Error()) + } + } else { + if err := helpers.CreatePathAndCopy(filepath.Join(packagePath, file.Source), dst); err != nil { + return fmt.Errorf("unable to copy file %s: %w", file.Source, err) + } + } + } + + if file.ExtractPath != "" { + // Make sure dst reflects the actual file or directory. + updatedExtractedFileOrDir := filepath.Join(destinationDir, file.ExtractPath) + if updatedExtractedFileOrDir != dst { + if err := os.Rename(updatedExtractedFileOrDir, dst); err != nil { + return fmt.Errorf(lang.ErrWritingFile, dst, err) + } + } + } + + // Abort packaging on invalid shasum (if one is specified). + if file.Shasum != "" { + if err := helpers.SHAsMatch(dst, file.Shasum); err != nil { + return err + } + } + + if file.Executable || helpers.IsDir(dst) { + err := os.Chmod(dst, helpers.ReadWriteExecuteUser) + if err != nil { + return err + } + } else { + err := os.Chmod(dst, helpers.ReadWriteUser) + if err != nil { + return err + } + } + } + + for dataIdx, data := range component.DataInjections { + rel := filepath.Join(string(DataComponentDir), strconv.Itoa(dataIdx), filepath.Base(data.Target.Path)) + dst := filepath.Join(compBuildPath, rel) + + if helpers.IsURL(data.Source) { + if err := utils.DownloadToFile(ctx, data.Source, dst, component.DeprecatedCosignKeyPath); err != nil { + return fmt.Errorf(lang.ErrDownloading, data.Source, err.Error()) + } + } else { + if err := helpers.CreatePathAndCopy(filepath.Join(packagePath, data.Source), dst); err != nil { + return fmt.Errorf("unable to copy data injection %s: %s", data.Source, err.Error()) + } + } + } + + // Iterate over all manifests. + if len(component.Manifests) > 0 { + err := os.MkdirAll(filepath.Join(compBuildPath, string(ManifestsComponentDir)), 0o700) + if err != nil { + return err + } + } + for _, manifest := range component.Manifests { + for fileIdx, path := range manifest.Files { + rel := filepath.Join(string(ManifestsComponentDir), fmt.Sprintf("%s-%d.yaml", manifest.Name, fileIdx)) + dst := filepath.Join(compBuildPath, rel) + + // Copy manifests without any processing. + if helpers.IsURL(path) { + if err := utils.DownloadToFile(ctx, path, dst, component.DeprecatedCosignKeyPath); err != nil { + return fmt.Errorf(lang.ErrDownloading, path, err.Error()) + } + } else { + if err := helpers.CreatePathAndCopy(filepath.Join(packagePath, path), dst); err != nil { + return fmt.Errorf("unable to copy manifest %s: %w", path, err) + } + } + } + + for kustomizeIdx, path := range manifest.Kustomizations { + // Generate manifests from kustomizations and place in the package. + kname := fmt.Sprintf("kustomization-%s-%d.yaml", manifest.Name, kustomizeIdx) + rel := filepath.Join(string(ManifestsComponentDir), kname) + dst := filepath.Join(compBuildPath, rel) + + if !helpers.IsURL(path) { + path = filepath.Join(packagePath, path) + } + if err := kustomize.Build(path, dst, manifest.KustomizeAllowAnyDirectory); err != nil { + return fmt.Errorf("unable to build kustomization %s: %w", path, err) + } + } + } + + // Load all specified git repos. + for _, url := range component.Repos { + // Pull all the references if there is no `@` in the string. + _, err := git.Clone(ctx, filepath.Join(compBuildPath, string(RepoComponentDir)), url, false) + if err != nil { + return fmt.Errorf("unable to pull git repo %s: %w", url, err) + } + } + + if err := actions2.Run(ctx, packagePath, onCreate.Defaults, onCreate.After, nil); err != nil { + return fmt.Errorf("unable to run component after action: %w", err) + } + + // Write the tar component. + entries, err := os.ReadDir(compBuildPath) + if err != nil { + return err + } + if len(entries) == 0 { + return nil + } + tarPath := filepath.Join(buildPath, "components", fmt.Sprintf("%s.tar", component.Name)) + err = os.MkdirAll(filepath.Join(buildPath, "components"), 0o700) + if err != nil { + return err + } + err = createReproducibleTarballFromDir(compBuildPath, component.Name, tarPath, false) + if err != nil { + return err + } + return nil +} + +func assembleSkeletonComponent(component v1alpha1.ZarfComponent, packagePath, buildPath string) error { tmpBuildPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) if err != nil { return err @@ -279,7 +652,7 @@ func assembleComponent(component v1alpha1.ZarfComponent, packagePath, buildPath if err != nil { return err } - err = createReproducibleTarballFromDir(compBuildPath, component.Name, tarPath) + err = createReproducibleTarballFromDir(compBuildPath, component.Name, tarPath, true) if err != nil { return err } @@ -384,7 +757,7 @@ func signPackage(dirPath, signingKeyPath, signingKeyPassword string) error { return nil } -func createReproducibleTarballFromDir(dirPath, dirPrefix, tarballPath string) error { +func createReproducibleTarballFromDir(dirPath, dirPrefix, tarballPath string, overrideMode bool) error { tb, err := os.Create(tarballPath) if err != nil { return fmt.Errorf("error creating tarball: %w", err) @@ -431,7 +804,9 @@ func createReproducibleTarballFromDir(dirPath, dirPrefix, tarballPath string) er // that when unpackaged files from packages created on Windows and Linux will have the same permissions. // The &^ operator called AND NOT sets the bits to 0 in the left hand if the right hand bits are 1. // https://medium.com/learning-the-go-programming-language/bit-hacking-with-go-e0acee258827 - header.Mode = header.Mode &^ 0o077 + if overrideMode { + header.Mode = header.Mode &^ 0o077 + } // Ensure the header's name is correctly set relative to the base directory name, err := filepath.Rel(dirPath, filePath) @@ -463,3 +838,196 @@ func createReproducibleTarballFromDir(dirPath, dirPrefix, tarballPath string) er return nil }) } + +func fillActiveTemplate(ctx context.Context, pkg v1alpha1.ZarfPackage, setVariables map[string]string) (v1alpha1.ZarfPackage, []string, error) { + templateMap := map[string]string{} + warnings := []string{} + + promptAndSetTemplate := func(templatePrefix string, deprecated bool) error { + yamlTemplates, err := utils.FindYamlTemplates(&pkg, templatePrefix, "###") + if err != nil { + return err + } + + for key := range yamlTemplates { + if deprecated { + warnings = append(warnings, fmt.Sprintf(lang.PkgValidateTemplateDeprecation, key, key, key)) + } + + _, present := setVariables[key] + if !present && !config.CommonOptions.Confirm { + setVal, err := interactive.PromptVariable(ctx, v1alpha1.InteractiveVariable{ + Variable: v1alpha1.Variable{Name: key}, + }) + if err != nil { + return err + } + setVariables[key] = setVal + } else if !present { + return fmt.Errorf("template %q must be '--set' when using the '--confirm' flag", key) + } + } + + for key, value := range setVariables { + templateMap[fmt.Sprintf("%s%s###", templatePrefix, key)] = value + } + + return nil + } + + // update the component templates on the package + if err := reloadComponentTemplatesInPackage(&pkg); err != nil { + return v1alpha1.ZarfPackage{}, nil, err + } + + if err := promptAndSetTemplate(v1alpha1.ZarfPackageTemplatePrefix, false); err != nil { + return v1alpha1.ZarfPackage{}, nil, err + } + // [DEPRECATION] Set the Package Variable syntax as well for backward compatibility + if err := promptAndSetTemplate(v1alpha1.ZarfPackageVariablePrefix, true); err != nil { + return v1alpha1.ZarfPackage{}, nil, err + } + + // Add special variable for the current package architecture + templateMap[v1alpha1.ZarfPackageArch] = pkg.Metadata.Architecture + + if err := utils.ReloadYamlTemplate(&pkg, templateMap); err != nil { + return v1alpha1.ZarfPackage{}, nil, err + } + + return pkg, warnings, nil +} + +// reloadComponentTemplate appends ###ZARF_COMPONENT_NAME### for the component, assigns value, and reloads +// Any instance of ###ZARF_COMPONENT_NAME### within a component will be replaced with that components name +func reloadComponentTemplate(component *v1alpha1.ZarfComponent) error { + mappings := map[string]string{} + mappings[v1alpha1.ZarfComponentName] = component.Name + err := utils.ReloadYamlTemplate(component, mappings) + if err != nil { + return err + } + return nil +} + +// reloadComponentTemplatesInPackage appends ###ZARF_COMPONENT_NAME### for each component, assigns value, and reloads +func reloadComponentTemplatesInPackage(zarfPackage *v1alpha1.ZarfPackage) error { + // iterate through components to and find all ###ZARF_COMPONENT_NAME, assign to component Name and value + for i := range zarfPackage.Components { + if err := reloadComponentTemplate(&zarfPackage.Components[i]); err != nil { + return err + } + } + return nil +} + +func splitFile(srcPath string, chunkSize int) (err error) { + srcFile, err := os.Open(srcPath) + if err != nil { + return err + } + // Ensure we close our sourcefile, even if we error out. + defer func() { + err2 := srcFile.Close() + // Ignore if file is already closed + if !errors.Is(err2, os.ErrClosed) { + err = errors.Join(err, err2) + } + }() + + fi, err := srcFile.Stat() + if err != nil { + return err + } + + title := fmt.Sprintf("[0/%d] MB bytes written", fi.Size()/1000/1000) + progressBar := message.NewProgressBar(fi.Size(), title) + defer func(progressBar *message.ProgressBar) { + err2 := progressBar.Close() + err = errors.Join(err, err2) + }(progressBar) + + hash := sha256.New() + fileCount := 0 + // TODO(mkcp): The inside of this loop should be wrapped in a closure so we can close the destination file each + // iteration as soon as we're done writing. + for { + path := fmt.Sprintf("%s.part%03d", srcPath, fileCount+1) + dstFile, err := os.OpenFile(path, os.O_CREATE|os.O_RDWR, helpers.ReadAllWriteUser) + if err != nil { + return err + } + defer func(dstFile *os.File) { + err2 := dstFile.Close() + // Ignore if file is already closed + if !errors.Is(err2, os.ErrClosed) { + err = errors.Join(err, err2) + } + }(dstFile) + + written, copyErr := io.CopyN(dstFile, srcFile, int64(chunkSize)) + if copyErr != nil && !errors.Is(copyErr, io.EOF) { + return err + } + progressBar.Add(int(written)) + title := fmt.Sprintf("[%d/%d] MB bytes written", progressBar.GetCurrent()/1000/1000, fi.Size()/1000/1000) + progressBar.Updatef(title) + + _, err = dstFile.Seek(0, io.SeekStart) + if err != nil { + return err + } + _, err = io.Copy(hash, dstFile) + if err != nil { + return err + } + + // EOF error could be returned on 0 bytes written. + if written == 0 { + // NOTE(mkcp): We have to close the file before removing it or windows will break with a file-in-use err. + err = dstFile.Close() + if err != nil { + return err + } + err = os.Remove(path) + if err != nil { + return err + } + break + } + + fileCount++ + if errors.Is(copyErr, io.EOF) { + break + } + } + + // Remove original file + // NOTE(mkcp): We have to close the file before removing or windows can break with a file-in-use err. + err = srcFile.Close() + if err != nil { + return err + } + err = os.Remove(srcPath) + if err != nil { + return err + } + + // Write header file + data := types.ZarfSplitPackageData{ + Count: fileCount, + Bytes: fi.Size(), + Sha256Sum: fmt.Sprintf("%x", hash.Sum(nil)), + } + b, err := json.Marshal(data) + if err != nil { + return fmt.Errorf("unable to marshal the split package data: %w", err) + } + path := fmt.Sprintf("%s.part000", srcPath) + if err := os.WriteFile(path, b, helpers.ReadAllWriteUser); err != nil { + return fmt.Errorf("unable to write the file %s: %w", path, err) + } + progressBar.Successf("Package split across %d files", fileCount+1) + + return nil +} diff --git a/src/internal/packager2/layout/create_test.go b/src/internal/packager2/layout/create_test.go index 7e29bb875a..b16d6ace08 100644 --- a/src/internal/packager2/layout/create_test.go +++ b/src/internal/packager2/layout/create_test.go @@ -24,7 +24,7 @@ func TestCreateSkeleton(t *testing.T) { lint.ZarfSchema = testutil.LoadSchema(t, "../../../../zarf.schema.json") opt := CreateOptions{} - path, err := CreateSkeleton(ctx, "./testdata/zarf-package", opt) + path, err := CreateSkeleton(ctx, "./testdata/zarf-skeleton-package", opt) require.NoError(t, err) pkgPath := layout.New(path) @@ -100,7 +100,7 @@ func TestCreateReproducibleTarballFromDir(t *testing.T) { require.NoError(t, err) tarPath := filepath.Join(t.TempDir(), "data.tar") - err = createReproducibleTarballFromDir(tmpDir, "", tarPath) + err = createReproducibleTarballFromDir(tmpDir, "", tarPath, true) require.NoError(t, err) shaSum, err := helpers.GetSHA256OfFile(tarPath) diff --git a/src/internal/packager2/layout/import.go b/src/internal/packager2/layout/import.go index c5933420a0..2415e2399d 100644 --- a/src/internal/packager2/layout/import.go +++ b/src/internal/packager2/layout/import.go @@ -29,7 +29,7 @@ func resolveImports(ctx context.Context, pkg v1alpha1.ZarfPackage, packagePath, components := []v1alpha1.ZarfComponent{} for _, component := range pkg.Components { - if !compatibleComponent(component, pkg.Metadata.Architecture, flavor) { + if !compatibleComponent(component, arch, flavor) { continue } @@ -46,10 +46,11 @@ func resolveImports(ctx context.Context, pkg v1alpha1.ZarfPackage, packagePath, var importedPkg v1alpha1.ZarfPackage if component.Import.Path != "" { importPath := filepath.Join(packagePath, component.Import.Path) - if _, ok := seenImports[importPath]; ok { + importKey := fmt.Sprintf("%s-%s", component.Name, importPath) + if _, ok := seenImports[importKey]; ok { return v1alpha1.ZarfPackage{}, fmt.Errorf("package %s imported in cycle by %s", filepath.ToSlash(importPath), filepath.ToSlash(packagePath)) } - seenImports[importPath] = nil + seenImports[importKey] = nil b, err := os.ReadFile(filepath.Join(importPath, layout.ZarfYAML)) if err != nil { return v1alpha1.ZarfPackage{}, err diff --git a/src/internal/packager2/layout/oci.go b/src/internal/packager2/layout/oci.go index b1c21bd5b3..ec4f43042b 100644 --- a/src/internal/packager2/layout/oci.go +++ b/src/internal/packager2/layout/oci.go @@ -56,7 +56,9 @@ func NewRemote(ctx context.Context, url string, platform ocispec.Platform, mods } // Push pushes the given package layout to the remote registry. -func (r *Remote) Push(ctx context.Context, pkgLayout PackageLayout, concurrency int) (err error) { +func (r *Remote) Push(ctx context.Context, pkgLayout *PackageLayout, concurrency int) (err error) { + logger.From(ctx).Info("pushing package to registry", "destination", r.orasRemote.Repo().Reference.String()) + src, err := file.New("") if err != nil { return err @@ -104,8 +106,8 @@ func (r *Remote) Push(ctx context.Context, pkgLayout PackageLayout, concurrency return nil } -func ReferenceFromMetadata(registryLocation string, metadata *v1alpha1.ZarfMetadata, build *v1alpha1.ZarfBuildData) (string, error) { - if len(metadata.Version) == 0 { +func ReferenceFromMetadata(registryLocation string, pkg v1alpha1.ZarfPackage) (string, error) { + if len(pkg.Metadata.Version) == 0 { return "", errors.New("version is required for publishing") } if !strings.HasSuffix(registryLocation, "/") { @@ -113,9 +115,9 @@ func ReferenceFromMetadata(registryLocation string, metadata *v1alpha1.ZarfMetad } registryLocation = strings.TrimPrefix(registryLocation, helpers.OCIURLPrefix) - raw := fmt.Sprintf("%s%s:%s", registryLocation, metadata.Name, metadata.Version) - if build != nil && build.Flavor != "" { - raw = fmt.Sprintf("%s-%s", raw, build.Flavor) + raw := fmt.Sprintf("%s%s:%s", registryLocation, pkg.Metadata.Name, pkg.Metadata.Version) + if pkg.Build.Flavor != "" { + raw = fmt.Sprintf("%s-%s", raw, pkg.Build.Flavor) } ref, err := registry.ParseReference(raw) diff --git a/src/internal/packager2/layout/package.go b/src/internal/packager2/layout/package.go index 708fd19b41..5bd75b7cfc 100644 --- a/src/internal/packager2/layout/package.go +++ b/src/internal/packager2/layout/package.go @@ -24,6 +24,9 @@ import ( "github.com/zarf-dev/zarf/src/api/v1alpha1" "github.com/zarf-dev/zarf/src/config" + "github.com/zarf-dev/zarf/src/pkg/logger" + "github.com/zarf-dev/zarf/src/pkg/message" + "github.com/zarf-dev/zarf/src/pkg/packager/sources" "github.com/zarf-dev/zarf/src/pkg/transform" "github.com/zarf-dev/zarf/src/pkg/utils" ) @@ -189,6 +192,38 @@ func (p *PackageLayout) GetImage(ref transform.Image) (registryv1.Image, error) return nil, fmt.Errorf("unable to find the image %s", ref.Reference) } +func (p *PackageLayout) Archive(ctx context.Context, dirPath string, maxPackageSize int) error { + packageName := fmt.Sprintf("%s%s", sources.NameFromMetadata(&p.Pkg, false), sources.PkgSuffix(p.Pkg.Metadata.Uncompressed)) + tarballPath := filepath.Join(dirPath, packageName) + err := os.Remove(tarballPath) + if err != nil && !errors.Is(err, os.ErrNotExist) { + return err + } + message.Notef("Saving package to path %s", tarballPath) + logger.From(ctx).Info("writing package to disk", "path", tarballPath) + err = archiver.Archive([]string{p.dirPath + string(os.PathSeparator)}, tarballPath) + if err != nil { + return fmt.Errorf("unable to create package: %w", err) + } + fi, err := os.Stat(tarballPath) + if err != nil { + return fmt.Errorf("unable to read the package archive: %w", err) + } + // Convert Megabytes to bytes. + chunkSize := maxPackageSize * 1000 * 1000 + // If a chunk size was specified and the package is larger than the chunk size, split it into chunks. + if maxPackageSize > 0 && fi.Size() > int64(chunkSize) { + if fi.Size()/int64(chunkSize) > 999 { + return fmt.Errorf("unable to split the package archive into multiple files: must be less than 1,000 files") + } + err := splitFile(tarballPath, chunkSize) + if err != nil { + return fmt.Errorf("unable to split the package archive into multiple files: %w", err) + } + } + return nil +} + // Files returns a map off all the files in the package. func (p *PackageLayout) Files() (map[string]string, error) { files := map[string]string{} diff --git a/src/internal/packager2/layout/sbom.go b/src/internal/packager2/layout/sbom.go new file mode 100644 index 0000000000..d7b71599cb --- /dev/null +++ b/src/internal/packager2/layout/sbom.go @@ -0,0 +1,334 @@ +package layout + +import ( + "context" + "embed" + "encoding/json" + "fmt" + "html/template" + "os" + "path/filepath" + "regexp" + "strconv" + + "github.com/anchore/stereoscope/pkg/file" + "github.com/anchore/stereoscope/pkg/image" + "github.com/anchore/syft/syft" + "github.com/anchore/syft/syft/artifact" + syftFile "github.com/anchore/syft/syft/file" + "github.com/anchore/syft/syft/format" + "github.com/anchore/syft/syft/format/syftjson" + "github.com/anchore/syft/syft/linux" + "github.com/anchore/syft/syft/pkg" + "github.com/anchore/syft/syft/sbom" + "github.com/anchore/syft/syft/source" + "github.com/anchore/syft/syft/source/directorysource" + "github.com/anchore/syft/syft/source/filesource" + "github.com/anchore/syft/syft/source/stereoscopesource" + "github.com/defenseunicorns/pkg/helpers/v2" + v1 "github.com/google/go-containerregistry/pkg/v1" + "github.com/mholt/archiver/v3" + + "github.com/zarf-dev/zarf/src/api/v1alpha1" + "github.com/zarf-dev/zarf/src/config" + "github.com/zarf-dev/zarf/src/pkg/transform" + "github.com/zarf-dev/zarf/src/pkg/utils" +) + +const componentPrefix = "zarf-component-" + +//go:embed viewer/* +var viewerAssets embed.FS +var transformRegex = regexp.MustCompile(`(?m)[^a-zA-Z0-9\.\-]`) + +func generateSBOM(ctx context.Context, pkg v1alpha1.ZarfPackage, buildPath string, images []transform.Image) error { + outputPath, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) + if err != nil { + return err + } + defer os.RemoveAll(outputPath) + + cachePath, err := config.GetAbsCachePath() + if err != nil { + return err + } + + componentSBOMs := []string{} + for _, comp := range pkg.Components { + if len(comp.Files) > 0 || len(comp.DataInjections) > 0 { + componentSBOMs = append(componentSBOMs, comp.Name) + } + } + jsonList, err := generateJSONList(componentSBOMs, images) + if err != nil { + return err + } + + for _, refInfo := range images { + img, err := utils.LoadOCIImage(filepath.Join(buildPath, string(ImagesDir)), refInfo) + if err != nil { + return err + } + b, err := createImageSBOM(ctx, cachePath, outputPath, img, refInfo.Reference) + if err != nil { + return err + } + err = createSBOMViewerAsset(outputPath, refInfo.Reference, b, jsonList) + if err != nil { + return err + } + } + + // Generate SBOM for each component + for _, comp := range pkg.Components { + if len(comp.DataInjections) == 0 && len(comp.Files) == 0 { + continue + } + jsonData, err := createFileSBOM(ctx, comp, outputPath, buildPath) + if err != nil { + return err + } + err = createSBOMViewerAsset(outputPath, fmt.Sprintf("%s%s", componentPrefix, comp.Name), jsonData, jsonList) + if err != nil { + return err + } + } + + // Include the compare tool if there are any image SBOMs OR component SBOMs + err = createSBOMCompareAsset(outputPath) + if err != nil { + return err + } + + err = createReproducibleTarballFromDir(outputPath, "", filepath.Join(buildPath, "sboms.tar"), false) + if err != nil { + return err + } + + return nil +} + +func createImageSBOM(ctx context.Context, cachePath, outputPath string, img v1.Image, src string) ([]byte, error) { + imageCachePath := filepath.Join(cachePath, ImagesDir) + err := os.MkdirAll(imageCachePath, helpers.ReadWriteExecuteUser) + if err != nil { + return nil, err + } + + refInfo, err := transform.ParseImageRef(src) + if err != nil { + return nil, fmt.Errorf("failed to create ref for image %s: %w", src, err) + } + syftImage := image.NewImage(img, file.NewTempDirGenerator("zarf"), imageCachePath, image.WithTags(refInfo.Reference)) + err = syftImage.Read() + if err != nil { + return nil, err + } + cfg := getDefaultSyftConfig() + syftSrc := stereoscopesource.New(syftImage, stereoscopesource.ImageConfig{ + Reference: refInfo.Reference, + }) + sbom, err := syft.CreateSBOM(ctx, syftSrc, cfg) + if err != nil { + return nil, err + } + jsonData, err := format.Encode(*sbom, syftjson.NewFormatEncoder()) + if err != nil { + return nil, err + } + + normalizedName := getNormalizedFileName(fmt.Sprintf("%s.json", refInfo.Reference)) + path := filepath.Join(outputPath, normalizedName) + err = os.WriteFile(path, jsonData, 0o666) + if err != nil { + return nil, err + } + return jsonData, nil +} + +func createFileSBOM(ctx context.Context, component v1alpha1.ZarfComponent, outputPath, buildPath string) ([]byte, error) { + tmpDir, err := utils.MakeTempDir(config.CommonOptions.TempDirectory) + if err != nil { + return nil, err + } + defer os.RemoveAll(tmpDir) + tarPath := filepath.Join(buildPath, ComponentsDir, component.Name) + ".tar" + err = archiver.Unarchive(tarPath, tmpDir) + if err != nil { + return nil, err + } + sbomFiles := []string{} + appendSBOMFiles := func(path string) error { + if helpers.IsDir(path) { + files, err := helpers.RecursiveFileList(path, nil, false) + if err != nil { + return err + } + sbomFiles = append(sbomFiles, files...) + } else { + sbomFiles = append(sbomFiles, path) + } + return nil + } + for i, file := range component.Files { + path := filepath.Join(tmpDir, component.Name, string(FilesComponentDir), strconv.Itoa(i), filepath.Base(file.Target)) + err := appendSBOMFiles(path) + if err != nil { + return nil, err + } + } + for i, data := range component.DataInjections { + path := filepath.Join(tmpDir, component.Name, string(DataComponentDir), strconv.Itoa(i), filepath.Base(data.Target.Path)) + err := appendSBOMFiles(path) + if err != nil { + return nil, err + } + } + + parentSource, err := directorysource.NewFromPath(tmpDir) + if err != nil { + return nil, err + } + catalog := pkg.NewCollection() + relationships := []artifact.Relationship{} + for _, sbomFile := range sbomFiles { + fileSrc, err := filesource.NewFromPath(sbomFile) + if err != nil { + return nil, err + } + + cfg := getDefaultSyftConfig() + sbom, err := syft.CreateSBOM(ctx, fileSrc, cfg) + if err != nil { + return nil, err + } + + for pkg := range sbom.Artifacts.Packages.Enumerate() { + containsSource := false + + // See if the source locations for this package contain the file Zarf indexed + for _, location := range pkg.Locations.ToSlice() { + if location.RealPath == fileSrc.Describe().Metadata.(source.FileMetadata).Path { + containsSource = true + } + } + + // If the locations do not contain the source file (i.e. the package was inside a tarball), add the file source + if !containsSource { + sourceLocation := syftFile.NewLocation(fileSrc.Describe().Metadata.(source.FileMetadata).Path) + pkg.Locations.Add(sourceLocation) + } + + catalog.Add(pkg) + } + + for _, r := range sbom.Relationships { + relationships = append(relationships, artifact.Relationship{ + From: parentSource, + To: r.To, + Type: r.Type, + Data: r.Data, + }) + } + } + artifact := sbom.SBOM{ + Descriptor: sbom.Descriptor{ + Name: "zarf", + Version: config.CLIVersion, + }, + Source: parentSource.Describe(), + Artifacts: sbom.Artifacts{ + Packages: catalog, + LinuxDistribution: &linux.Release{}, + }, + Relationships: relationships, + } + jsonData, err := format.Encode(artifact, syftjson.NewFormatEncoder()) + if err != nil { + return nil, err + } + + filename := fmt.Sprintf("%s%s.json", componentPrefix, component.Name) + path := filepath.Join(outputPath, getNormalizedFileName(filename)) + err = os.WriteFile(path, jsonData, 0o666) + if err != nil { + return nil, err + } + return jsonData, nil +} + +func createSBOMViewerAsset(outputDir, identifier string, jsonData, jsonList []byte) error { + filename := fmt.Sprintf("sbom-viewer-%s.html", getNormalizedFileName(identifier)) + return createSBOMHTML(outputDir, filename, "viewer/template.gohtml", jsonData, jsonList) +} + +func createSBOMCompareAsset(outputDir string) error { + return createSBOMHTML(outputDir, "compare.html", "viewer/compare.gohtml", nil, nil) +} + +func createSBOMHTML(outputDir, filename, goTemplate string, jsonData, jsonList []byte) error { + path := filepath.Join(outputDir, getNormalizedFileName(filename)) + file, err := os.Create(path) + if err != nil { + return err + } + defer file.Close() + tplData := struct { + ThemeCSS template.CSS + ViewerCSS template.CSS + List template.JS + Data template.JS + LibraryJS template.JS + CommonJS template.JS + ViewerJS template.JS + CompareJS template.JS + }{ + ThemeCSS: loadFileCSS("theme.css"), + ViewerCSS: loadFileCSS("styles.css"), + List: template.JS(jsonList), + Data: template.JS(jsonData), + LibraryJS: loadFileJS("library.js"), + CommonJS: loadFileJS("common.js"), + ViewerJS: loadFileJS("viewer.js"), + CompareJS: loadFileJS("compare.js"), + } + tpl, err := template.ParseFS(viewerAssets, goTemplate) + if err != nil { + return err + } + return tpl.Execute(file, tplData) +} + +func loadFileCSS(name string) template.CSS { + data, _ := viewerAssets.ReadFile("viewer/" + name) + return template.CSS(data) +} + +func loadFileJS(name string) template.JS { + data, _ := viewerAssets.ReadFile("viewer/" + name) + return template.JS(data) +} + +func getNormalizedFileName(identifier string) string { + return transformRegex.ReplaceAllString(identifier, "_") +} + +func generateJSONList(components []string, imageList []transform.Image) ([]byte, error) { + var jsonList []string + for _, refInfo := range imageList { + normalized := getNormalizedFileName(refInfo.Reference) + jsonList = append(jsonList, normalized) + } + for _, k := range components { + normalized := getNormalizedFileName(fmt.Sprintf("%s%s", componentPrefix, k)) + jsonList = append(jsonList, normalized) + } + return json.Marshal(jsonList) +} + +func getDefaultSyftConfig() *syft.CreateSBOMConfig { + cfg := syft.DefaultCreateSBOMConfig() + cfg.ToolName = "zarf" + cfg.ToolVersion = config.CLIVersion + return cfg +} diff --git a/src/internal/packager2/layout/sbom_test.go b/src/internal/packager2/layout/sbom_test.go new file mode 100644 index 0000000000..9c8edabcbc --- /dev/null +++ b/src/internal/packager2/layout/sbom_test.go @@ -0,0 +1,27 @@ +package layout + +import ( + "os" + "path/filepath" + "testing" + + "github.com/google/go-containerregistry/pkg/v1/empty" + "github.com/stretchr/testify/require" + "github.com/zarf-dev/zarf/src/test/testutil" +) + +func TestCreateImageSBOM(t *testing.T) { + t.Parallel() + + ctx := testutil.TestContext(t) + + outputPath := t.TempDir() + img := empty.Image + b, err := createImageSBOM(ctx, t.TempDir(), outputPath, img, "docker.io/foo/bar:latest") + require.NoError(t, err) + require.NotEmpty(t, b) + + fileContent, err := os.ReadFile(filepath.Join(outputPath, "docker.io_foo_bar_latest.json")) + require.NoError(t, err) + require.Equal(t, fileContent, b) +} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/archive.tar b/src/internal/packager2/layout/testdata/zarf-skeleton-package/archive.tar new file mode 100644 index 0000000000000000000000000000000000000000..cbbd80680cd84b59686c8b82b86a1b1e1ea4b20f GIT binary patch literal 20480 zcmeIu!3x4K3;@u6%6>p;-PV0iQDFlQV@~n&?P&-*c-rmdkd!2Z^2%}@r}zDID{DEg zpEVp1u}1E?P)i#6_*-l1($y4Fmxun6V{TK3sqIq=O+4T)yOXcyT9&2>^Ef@Gc{kn~ z|K{7G2LS>E2oNAZfB*pk1PBlyK!5-N0t5&UAV7cs0RjXF5FkK+009C72oNAZfB*pk u1PBlyK!5-N0t5&UAV7cs0RjXF5FkK+009C72oNAZfB*pk1PBn=3w#07z!#MO literal 0 HcmV?d00001 diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/.helmignore b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/.helmignore @@ -0,0 +1,21 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/Chart.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/Chart.yaml new file mode 100644 index 0000000000..0ae3bfd45f --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/Chart.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +version: 6.4.0 +appVersion: 6.4.0 +name: podinfo +engine: gotpl +description: Podinfo Helm chart for Kubernetes +home: https://github.com/stefanprodan/podinfo +maintainers: +- email: stefanprodan@users.noreply.github.com + name: stefanprodan +sources: +- https://github.com/stefanprodan/podinfo +kubeVersion: ">=1.23.0-0" diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/LICENSE b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/LICENSE new file mode 100644 index 0000000000..1b92ec15f9 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright 2018 Stefan Prodan. All rights reserved. + + 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. diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/NOTICE b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/NOTICE new file mode 100644 index 0000000000..5b0414f8c2 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/NOTICE @@ -0,0 +1 @@ +All files from this chart are from https://github.com/stefanprodan/podinfo/tree/6.4.0/charts/podinfo. diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/NOTES.txt b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/NOTES.txt new file mode 100644 index 0000000000..d8329725ef --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/NOTES.txt @@ -0,0 +1,20 @@ +1. Get the application URL by running these commands: +{{- if .Values.ingress.enabled }} +{{- range $host := .Values.ingress.hosts }} + {{- range .paths }} + http{{ if $.Values.ingress.tls }}s{{ end }}://{{ $host.host }}{{ .path }} + {{- end }} +{{- end }} +{{- else if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "podinfo.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc -w {{ template "podinfo.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "podinfo.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.service.externalPort }} +{{- else if contains "ClusterIP" .Values.service.type }} + echo "Visit http://127.0.0.1:8080 to use your application" + kubectl -n {{ .Release.Namespace }} port-forward deploy/{{ template "podinfo.fullname" . }} 8080:{{ .Values.service.externalPort }} +{{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/_helpers.tpl b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/_helpers.tpl new file mode 100644 index 0000000000..1f5a052871 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/_helpers.tpl @@ -0,0 +1,69 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "podinfo.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If release name contains chart name it will be used as a full name. +*/}} +{{- define "podinfo.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "podinfo.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "podinfo.labels" -}} +helm.sh/chart: {{ include "podinfo.chart" . }} +{{ include "podinfo.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "podinfo.selectorLabels" -}} +app.kubernetes.io/name: {{ include "podinfo.fullname" . }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "podinfo.serviceAccountName" -}} +{{- if .Values.serviceAccount.enabled }} +{{- default (include "podinfo.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +Create the name of the tls secret for secure port +*/}} +{{- define "podinfo.tlsSecretName" -}} +{{- $fullname := include "podinfo.fullname" . -}} +{{- default (printf "%s-tls" $fullname) .Values.tls.secretName }} +{{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/deployment.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/deployment.yaml new file mode 100644 index 0000000000..87ed373534 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/deployment.yaml @@ -0,0 +1,205 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "podinfo.fullname" . }} + labels: + {{- include "podinfo.labels" . | nindent 4 }} +spec: + {{- if not .Values.hpa.enabled }} + replicas: {{ .Values.replicaCount }} + {{- end }} + strategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + selector: + matchLabels: + {{- include "podinfo.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "podinfo.selectorLabels" . | nindent 8 }} + annotations: + prometheus.io/scrape: "true" + prometheus.io/port: "{{ .Values.service.httpPort }}" + {{- range $key, $value := .Values.podAnnotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + spec: + terminationGracePeriodSeconds: 30 + {{- if .Values.serviceAccount.enabled }} + serviceAccountName: {{ template "podinfo.serviceAccountName" . }} + {{- end }} + containers: + - name: {{ .Chart.Name }} + image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + {{- if .Values.securityContext }} + securityContext: + {{- toYaml .Values.securityContext | nindent 12 }} + {{- else if (or .Values.service.hostPort .Values.tls.hostPort) }} + securityContext: + allowPrivilegeEscalation: true + capabilities: + drop: + - ALL + add: + - NET_BIND_SERVICE + {{- end }} + command: + - ./podinfo + - --port={{ .Values.service.httpPort | default 9898 }} + {{- if .Values.host }} + - --host={{ .Values.host }} + {{- end }} + {{- if .Values.tls.enabled }} + - --secure-port={{ .Values.tls.port }} + {{- end }} + {{- if .Values.tls.certPath }} + - --cert-path={{ .Values.tls.certPath }} + {{- end }} + {{- if .Values.service.metricsPort }} + - --port-metrics={{ .Values.service.metricsPort }} + {{- end }} + {{- if .Values.service.grpcPort }} + - --grpc-port={{ .Values.service.grpcPort }} + {{- end }} + {{- if .Values.service.grpcService }} + - --grpc-service-name={{ .Values.service.grpcService }} + {{- end }} + {{- range .Values.backends }} + - --backend-url={{ . }} + {{- end }} + {{- if .Values.cache }} + - --cache-server={{ .Values.cache }} + {{- else if .Values.redis.enabled }} + - --cache-server=tcp://{{ template "podinfo.fullname" . }}-redis:6379 + {{- end }} + - --level={{ .Values.logLevel }} + - --random-delay={{ .Values.faults.delay }} + - --random-error={{ .Values.faults.error }} + {{- if .Values.faults.unhealthy }} + - --unhealthy + {{- end }} + {{- if .Values.faults.unready }} + - --unready + {{- end }} + {{- if .Values.h2c.enabled }} + - --h2c + {{- end }} + env: + {{- if .Values.ui.message }} + - name: PODINFO_UI_MESSAGE + value: {{ quote .Values.ui.message }} + {{- end }} + {{- if .Values.ui.logo }} + - name: PODINFO_UI_LOGO + value: {{ .Values.ui.logo }} + {{- end }} + {{- if .Values.ui.color }} + - name: PODINFO_UI_COLOR + value: {{ quote .Values.ui.color }} + {{- end }} + {{- if .Values.backend }} + - name: PODINFO_BACKEND_URL + value: {{ .Values.backend }} + {{- end }} + ports: + - name: http + containerPort: {{ .Values.service.httpPort | default 9898 }} + protocol: TCP + {{- if .Values.service.hostPort }} + hostPort: {{ .Values.service.hostPort }} + {{- end }} + {{- if .Values.tls.enabled }} + - name: https + containerPort: {{ .Values.tls.port | default 9899 }} + protocol: TCP + {{- if .Values.tls.hostPort }} + hostPort: {{ .Values.tls.hostPort }} + {{- end }} + {{- end }} + {{- if .Values.service.metricsPort }} + - name: http-metrics + containerPort: {{ .Values.service.metricsPort }} + protocol: TCP + {{- end }} + {{- if .Values.service.grpcPort }} + - name: grpc + containerPort: {{ .Values.service.grpcPort }} + protocol: TCP + {{- end }} + {{- if .Values.probes.startup.enable }} + startupProbe: + exec: + command: + - podcli + - check + - http + - localhost:{{ .Values.service.httpPort | default 9898 }}/healthz + {{- with .Values.probes.startup }} + initialDelaySeconds: {{ .initialDelaySeconds | default 1 }} + timeoutSeconds: {{ .timeoutSeconds | default 5 }} + failureThreshold: {{ .failureThreshold | default 3 }} + successThreshold: {{ .successThreshold | default 1 }} + periodSeconds: {{ .periodSeconds | default 10 }} + {{- end }} + {{- end }} + livenessProbe: + exec: + command: + - podcli + - check + - http + - localhost:{{ .Values.service.httpPort | default 9898 }}/healthz + {{- with .Values.probes.liveness }} + initialDelaySeconds: {{ .initialDelaySeconds | default 1 }} + timeoutSeconds: {{ .timeoutSeconds | default 5 }} + failureThreshold: {{ .failureThreshold | default 3 }} + successThreshold: {{ .successThreshold | default 1 }} + periodSeconds: {{ .periodSeconds | default 10 }} + {{- end }} + readinessProbe: + exec: + command: + - podcli + - check + - http + - localhost:{{ .Values.service.httpPort | default 9898 }}/readyz + {{- with .Values.probes.readiness }} + initialDelaySeconds: {{ .initialDelaySeconds | default 1 }} + timeoutSeconds: {{ .timeoutSeconds | default 5 }} + failureThreshold: {{ .failureThreshold | default 3 }} + successThreshold: {{ .successThreshold | default 1 }} + periodSeconds: {{ .periodSeconds | default 10 }} + {{- end }} + volumeMounts: + - name: data + mountPath: /data + {{- if .Values.tls.enabled }} + - name: tls + mountPath: {{ .Values.tls.certPath | default "/data/cert" }} + readOnly: true + {{- end }} + resources: +{{ toYaml .Values.resources | indent 12 }} + {{- with .Values.nodeSelector }} + nodeSelector: +{{ toYaml . | indent 8 }} + {{- end }} + {{- with .Values.affinity }} + affinity: +{{ toYaml . | indent 8 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: +{{ toYaml . | indent 8 }} + {{- end }} + volumes: + - name: data + emptyDir: {} + {{- if .Values.tls.enabled }} + - name: tls + secret: + secretName: {{ template "podinfo.tlsSecretName" . }} + {{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/hpa.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/hpa.yaml new file mode 100644 index 0000000000..f2fb8df1b8 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/hpa.yaml @@ -0,0 +1,41 @@ +{{- if .Values.hpa.enabled -}} +apiVersion: autoscaling/v2 +kind: HorizontalPodAutoscaler +metadata: + name: {{ template "podinfo.fullname" . }} + labels: + {{- include "podinfo.labels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ template "podinfo.fullname" . }} + minReplicas: {{ .Values.replicaCount }} + maxReplicas: {{ .Values.hpa.maxReplicas }} + metrics: + {{- if .Values.hpa.cpu }} + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: {{ .Values.hpa.cpu }} + {{- end }} + {{- if .Values.hpa.memory }} + - type: Resource + resource: + name: memory + target: + type: AverageValue + averageValue: {{ .Values.hpa.memory }} + {{- end }} + {{- if .Values.hpa.requests }} + - type: Pods + pods: + metric: + name: http_requests + target: + type: AverageValue + averageValue: {{ .Values.hpa.requests }} + {{- end }} +{{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/ingress.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/ingress.yaml new file mode 100644 index 0000000000..93f9ae437a --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/ingress.yaml @@ -0,0 +1,41 @@ +{{- if .Values.ingress.enabled -}} +{{- $fullName := include "podinfo.fullname" . -}} +{{- $svcPort := .Values.service.externalPort -}} +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: {{ $fullName }} + labels: + {{- include "podinfo.labels" . | nindent 4 }} + {{- with .Values.ingress.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ingressClassName: {{ .Values.ingress.className }} + {{- if .Values.ingress.tls }} + tls: + {{- range .Values.ingress.tls }} + - hosts: + {{- range .hosts }} + - {{ . | quote }} + {{- end }} + secretName: {{ .secretName }} + {{- end }} + {{- end }} + rules: + {{- range .Values.ingress.hosts }} + - host: {{ .host | quote }} + http: + paths: + {{- range .paths }} + - path: {{ .path }} + pathType: {{ .pathType }} + backend: + service: + name: {{ $fullName }} + port: + number: {{ $svcPort }} + {{- end }} + {{- end }} +{{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/service.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/service.yaml new file mode 100644 index 0000000000..6014e78853 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/service.yaml @@ -0,0 +1,36 @@ +{{- if .Values.service.enabled -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "podinfo.fullname" . }} + labels: + {{- include "podinfo.labels" . | nindent 4 }} +{{- with .Values.service.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + type: {{ .Values.service.type }} + ports: + - port: {{ .Values.service.externalPort }} + targetPort: http + protocol: TCP + name: http + {{- if (and (eq .Values.service.type "NodePort") (not (empty .Values.service.nodePort))) }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + {{- if .Values.tls.enabled }} + - port: {{ .Values.tls.port | default 9899 }} + targetPort: https + protocol: TCP + name: https + {{- end }} + {{- if .Values.service.grpcPort }} + - port: {{ .Values.service.grpcPort }} + targetPort: grpc + protocol: TCP + name: grpc + {{- end }} + selector: + {{- include "podinfo.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/serviceaccount.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/serviceaccount.yaml new file mode 100644 index 0000000000..d39b798967 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/templates/serviceaccount.yaml @@ -0,0 +1,12 @@ +{{- if .Values.serviceAccount.enabled -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ template "podinfo.serviceAccountName" . }} + labels: + {{- include "podinfo.labels" . | nindent 4 }} +{{- with .Values.serviceAccount.imagePullSecrets }} +imagePullSecrets: + {{- toYaml . | nindent 2 }} +{{- end -}} +{{- end -}} diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/values.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/values.yaml new file mode 100644 index 0000000000..89b2bd9129 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/chart/values.yaml @@ -0,0 +1,164 @@ +# Default values for podinfo. + +replicaCount: 1 +logLevel: info +host: #0.0.0.0 +backend: #http://backend-podinfo:9898/echo +backends: [] + +image: + repository: ghcr.io/stefanprodan/podinfo + tag: 6.4.0 + pullPolicy: IfNotPresent + +ui: + color: "#34577c" + message: "" + logo: "" + +# failure conditions +faults: + delay: false + error: false + unhealthy: false + unready: false + testFail: false + testTimeout: false + +# Kubernetes Service settings +service: + enabled: true + annotations: {} + type: ClusterIP + metricsPort: 9797 + httpPort: 9898 + externalPort: 9898 + grpcPort: 9999 + grpcService: podinfo + nodePort: 31198 + # the port used to bind the http port to the host + # NOTE: requires privileged container with NET_BIND_SERVICE capability -- this is useful for testing + # in local clusters such as kind without port forwarding + hostPort: + +# enable h2c protocol (non-TLS version of HTTP/2) +h2c: + enabled: false + +# enable tls on the podinfo service +tls: + enabled: false + # the name of the secret used to mount the certificate key pair + secretName: + # the path where the certificate key pair will be mounted + certPath: /data/cert + # the port used to host the tls endpoint on the service + port: 9899 + # the port used to bind the tls port to the host + # NOTE: requires privileged container with NET_BIND_SERVICE capability -- this is useful for testing + # in local clusters such as kind without port forwarding + hostPort: + +# create a certificate manager certificate (cert-manager required) +certificate: + create: false + # the issuer used to issue the certificate + issuerRef: + kind: ClusterIssuer + name: self-signed + # the hostname / subject alternative names for the certificate + dnsNames: + - podinfo + +# metrics-server add-on required +hpa: + enabled: false + maxReplicas: 10 + # average total CPU usage per pod (1-100) + cpu: + # average memory usage per pod (100Mi-1Gi) + memory: + # average http requests per second per pod (k8s-prometheus-adapter) + requests: + +# Redis address in the format tcp://: +cache: "" +# Redis deployment +redis: + enabled: false + repository: redis + tag: 7.0.7 + +serviceAccount: + # Specifies whether a service account should be created + enabled: false + # The name of the service account to use. + # If not set and create is true, a name is generated using the fullname template + name: + # List of image pull secrets if pulling from private registries + imagePullSecrets: [] + +# set container security context +securityContext: {} + +ingress: + enabled: false + className: "" + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + hosts: + - host: podinfo.local + paths: + - path: / + pathType: ImplementationSpecific + tls: [] + # - secretName: chart-example-tls + # hosts: + # - chart-example.local + +linkerd: + profile: + enabled: false + +# create Prometheus Operator monitor +serviceMonitor: + enabled: false + interval: 15s + additionalLabels: {} + +resources: + limits: + requests: + cpu: 1m + memory: 16Mi + +nodeSelector: {} + +tolerations: [] + +affinity: {} + +podAnnotations: {} + +# https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes +probes: + readiness: + initialDelaySeconds: 1 + timeoutSeconds: 5 + failureThreshold: 3 + successThreshold: 1 + periodSeconds: 10 + liveness: + initialDelaySeconds: 1 + timeoutSeconds: 5 + failureThreshold: 3 + successThreshold: 1 + periodSeconds: 10 + startup: + enable: false + initialDelaySeconds: 10 + timeoutSeconds: 5 + failureThreshold: 20 + successThreshold: 1 + periodSeconds: 10 diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/data.txt b/src/internal/packager2/layout/testdata/zarf-skeleton-package/data.txt new file mode 100644 index 0000000000..557db03de9 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/data.txt @@ -0,0 +1 @@ +Hello World diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/deployment.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/deployment.yaml new file mode 100644 index 0000000000..685c17aa68 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/deployment.yaml @@ -0,0 +1,21 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: nginx-deployment + labels: + app: nginx +spec: + replicas: 3 + selector: + matchLabels: + app: nginx + template: + metadata: + labels: + app: nginx + spec: + containers: + - name: nginx + image: nginx:1.14.2 + ports: + - containerPort: 80 diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/injection/data.txt b/src/internal/packager2/layout/testdata/zarf-skeleton-package/injection/data.txt new file mode 100644 index 0000000000..1269488f7f --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/injection/data.txt @@ -0,0 +1 @@ +data diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/kustomization.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/kustomization.yaml new file mode 100644 index 0000000000..736967b1a3 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/kustomization.yaml @@ -0,0 +1,2 @@ +resources: + - namespace.yaml diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/namespace.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/namespace.yaml new file mode 100644 index 0000000000..7c265c0193 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/kustomize/namespace.yaml @@ -0,0 +1,4 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: test diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/values.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/values.yaml new file mode 100644 index 0000000000..f86a45afe7 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/values.yaml @@ -0,0 +1,5 @@ +ui: + color: "#0d133d" + message: "greetings from podinfo (as deployed by Zarf)" + # Replace the githubusercontent URL for the airgap + logo: "" diff --git a/src/internal/packager2/layout/testdata/zarf-skeleton-package/zarf.yaml b/src/internal/packager2/layout/testdata/zarf-skeleton-package/zarf.yaml new file mode 100644 index 0000000000..695ac11604 --- /dev/null +++ b/src/internal/packager2/layout/testdata/zarf-skeleton-package/zarf.yaml @@ -0,0 +1,41 @@ +kind: ZarfPackageConfig +metadata: + name: test + version: v0.0.1 +components: + - name: helm-charts + required: true + charts: + - name: podinfo-local + version: 6.4.0 + namespace: podinfo-from-local-chart + localPath: chart + valuesFiles: + - values.yaml + - name: files + required: true + files: + - source: data.txt + target: data.txt + - source: archive.tar + extractPath: archive-data.txt + target: archive-data.txt + - name: data-injections + required: true + dataInjections: + - source: injection + target: + namespace: test + selector: app=test + container: test + path: /test + compress: true + - name: manifests + required: true + manifests: + - name: deployment + namespace: httpd + files: + - deployment.yaml + kustomizations: + - kustomize diff --git a/src/internal/packager2/layout/viewer/common.js b/src/internal/packager2/layout/viewer/common.js new file mode 100644 index 0000000000..37c63a4e32 --- /dev/null +++ b/src/internal/packager2/layout/viewer/common.js @@ -0,0 +1,56 @@ +const sbomSelector = document.getElementById('sbom-selector'); +const distroInfo = document.getElementById('distro-info'); +const modal = document.getElementById('modal'); +const modalFader = document.getElementById('modal-fader'); +const modalTitle = document.getElementById('modal-title'); +const modalContent = document.getElementById('modal-content'); +const artifactsTable = document.createElement('table'); +const mailtoMaintainerReplace = ` |  $1`; + +document.body.appendChild(artifactsTable); + +function fileList(files, artifactName) { + if (files) { + const list = (files || []).map((file) => file.path || '').filter((test) => test); + + if (list.length > 0) { + flatList = list.sort().join('
'); + return `${list.length} files`; + } + } + + return '-'; +} + +function choose(path) { + if (path !== '-') { + window.location.href = encodeURIComponent(`sbom-viewer-${path}.html`); + } +} + +function exportCSV(path) { + if (window.dt) { + window.dt.export({ + type: 'csv', + filename: path + }); + } else { + showModal('Unable to Export', 'No data in current table'); + } +} + +function showModal(title, list) { + modalTitle.innerText = `Files for ${title}`; + modalContent.innerHTML = list; + modalFader.className = 'active'; + modal.className = 'active'; +} + +function hideModal() { + modalFader.className = ''; + modal.className = ''; + modalTitle.innerText = ''; + modalContent.innerHTML = ''; +} diff --git a/src/internal/packager2/layout/viewer/compare.gohtml b/src/internal/packager2/layout/viewer/compare.gohtml new file mode 100644 index 0000000000..afda6fdb2d --- /dev/null +++ b/src/internal/packager2/layout/viewer/compare.gohtml @@ -0,0 +1,292 @@ + + + + + + + Zarf SBOM Comparison + + + + + +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Zarf SBOM Comparison +
+
+ +
+ +
+
+
+

+ + + + +
+
+

Old File

+ +
+
+

New File

+ +
+
+

 

+ +
+
+
+

 

+ +
+
+ + + + + diff --git a/src/internal/packager2/layout/viewer/compare.js b/src/internal/packager2/layout/viewer/compare.js new file mode 100644 index 0000000000..27b462d87e --- /dev/null +++ b/src/internal/packager2/layout/viewer/compare.js @@ -0,0 +1,126 @@ +const leftJsonPicker = document.getElementById('leftJson'); +const rightJsonPicker = document.getElementById('rightJson'); + +function initSelector() { + sbomSelector.add(new Option('-', '-', true, true)); + + ZARF_SBOM_LIST.sort().forEach((item) => { + sbomSelector.add(new Option(item, item, false, false)); + }); +} + +function compare() { + if ( + document.getElementById('leftJson').files.length == 0 || + document.getElementById('rightJson').files.length == 0 + ) { + showModal('Unable to Compare', 'You must select 2 files from the file browsers'); + return; + } + + let leftJson = document.getElementById('leftJson').files[0]; + let rightJson = document.getElementById('rightJson').files[0]; + + let leftReader = new FileReader(); + leftReader.readAsText(leftJson); + + leftReader.onload = function () { + try { + let leftData = JSON.parse(leftReader.result); + const leftMap = {}; + leftData.artifacts.map((artifact) => { + if (!leftMap[artifact.name]) { + leftMap[artifact.name] = {}; + } + leftMap[artifact.name][artifact.version] = artifact; + }); + + let rightReader = new FileReader(); + rightReader.readAsText(rightJson); + + rightReader.onload = function () { + try { + let rightData = JSON.parse(rightReader.result); + const rightMap = {}; + rightData.artifacts.map((artifact) => { + if (!rightMap[artifact.name]) { + rightMap[artifact.name] = {}; + } + rightMap[artifact.name][artifact.version] = artifact; + }); + + let differences = []; + rightData.artifacts.map((artifact) => { + if (!leftMap[artifact.name]) { + artifact.zarfDiff = 'Added'; + differences.push(artifact); + } else if (!leftMap[artifact.name][artifact.version]) { + artifact.zarfDiff = 'Changed'; + oldVersion = Object.keys(leftMap[artifact.name])[0]; + artifact.version = oldVersion + ' -> ' + artifact.version; + differences.push(artifact); + } + }); + + leftData.artifacts.map((artifact) => { + if (!rightMap[artifact.name]) { + artifact.zarfDiff = 'Removed'; + differences.push(artifact); + } + }); + + loadDataTable(differences, artifactsTable); + } catch (e) { + showModal('Unable to Compare', 'You must select 2 Syft JSON files'); + } + }; + } catch (e) { + showModal('Unable to Compare', 'You must select 2 Syft JSON files'); + } + }; +} + +function loadDataTable(artifacts, dataTable) { + const transformedData = artifacts.map((artifact) => { + return [ + diff(artifact.zarfDiff), + artifact.type, + artifact.name, + artifact.version, + fileList(artifact.locations, artifact.name), + (artifact.metadata && fileList(artifact.metadata.files, artifact.name)) || '-', + (artifact.metadata && artifact.metadata.description) || '-', + ((artifact.metadata && artifact.metadata.maintainer) || '-').replace( + /\u003c(.*)\u003e/, + mailtoMaintainerReplace + ), + (artifact.metadata && artifact.metadata.installedSize) || '-' + ]; + }); + + const data = { + headings: ['Difference', 'Type', 'Name', 'Version', 'Sources', 'Package Files', 'Notes', 'Maintainer', 'Size'], + data: transformedData + }; + + if (window.dt) { + window.dt.destroy(); + } + + window.dt = new simpleDatatables.DataTable(dataTable, { + data, + perPage: 20 + }); +} + +function diff(diffTag) { + return `${diffTag}`; +} + +function getCompareName() { + leftFilename = leftJsonPicker.value.split('/').pop().split('\\').pop(); + rightFilename = rightJsonPicker.value.split('/').pop().split('\\').pop(); + return leftFilename.replace(/\.json$/, '') + '-' + rightFilename.replace(/\.json$/, ''); +} + +initSelector(); diff --git a/src/internal/packager2/layout/viewer/library.js b/src/internal/packager2/layout/viewer/library.js new file mode 100644 index 0000000000..422ec0f6db --- /dev/null +++ b/src/internal/packager2/layout/viewer/library.js @@ -0,0 +1,8 @@ +/** + * Minified by jsDelivr using Terser v5.9.0. + * Original file: /npm/simple-datatables@3.2.0/dist/umd/simple-datatables.js + * + * Do NOT use SRI with dynamically generated files! More information: https://www.jsdelivr.com/using-sri-with-dynamic-files + */ + !function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{("undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this).simpleDatatables=t()}}((function(){return function t(e,s,i){function a(r,o){if(!s[r]){if(!e[r]){var h="function"==typeof require&&require;if(!o&&h)return h(r,!0);if(n)return n(r,!0);var l=new Error("Cannot find module '"+r+"'");throw l.code="MODULE_NOT_FOUND",l}var d=s[r]={exports:{}};e[r][0].call(d.exports,(function(t){return a(e[r][1][t]||t)}),d,d.exports,t,e,s,i)}return s[r].exports}for(var n="function"==typeof require&&require,r=0;r=e?t:""+Array(e+1-i.length).join(s)+t},b={s:m,z:function(t){var e=-t.utcOffset(),s=Math.abs(e),i=Math.floor(s/60),a=s%60;return(e<=0?"+":"-")+m(i,2,"0")+":"+m(a,2,"0")},m:function t(e,s){if(e.date()68?1900:2e3)},o=function(t){return function(e){this[t]=+e}},h=[/[+-]\d\d:?(\d\d)?|Z/,function(t){(this.zone||(this.zone={})).offset=function(t){if(!t)return 0;if("Z"===t)return 0;var e=t.match(/([+-]|\d\d)/g),s=60*e[1]+(+e[2]||0);return 0===s?0:"+"===e[0]?-s:s}(t)}],l=function(t){var e=n[t];return e&&(e.indexOf?e:e.s.concat(e.f))},d=function(t,e){var s,i=n.meridiem;if(i){for(var a=1;a<=24;a+=1)if(t.indexOf(i(a,0,e))>-1){s=a>12;break}}else s=t===(e?"pm":"PM");return s},c={A:[a,function(t){this.afternoon=d(t,!1)}],a:[a,function(t){this.afternoon=d(t,!0)}],S:[/\d/,function(t){this.milliseconds=100*+t}],SS:[s,function(t){this.milliseconds=10*+t}],SSS:[/\d{3}/,function(t){this.milliseconds=+t}],s:[i,o("seconds")],ss:[i,o("seconds")],m:[i,o("minutes")],mm:[i,o("minutes")],H:[i,o("hours")],h:[i,o("hours")],HH:[i,o("hours")],hh:[i,o("hours")],D:[i,o("day")],DD:[s,o("day")],Do:[a,function(t){var e=n.ordinal,s=t.match(/\d+/);if(this.day=s[0],e)for(var i=1;i<=31;i+=1)e(i).replace(/\[|\]/g,"")===t&&(this.day=i)}],M:[i,o("month")],MM:[s,o("month")],MMM:[a,function(t){var e=l("months"),s=(l("monthsShort")||e.map((function(t){return t.substr(0,3)}))).indexOf(t)+1;if(s<1)throw new Error;this.month=s%12||s}],MMMM:[a,function(t){var e=l("months").indexOf(t)+1;if(e<1)throw new Error;this.month=e%12||e}],Y:[/[+-]?\d+/,o("year")],YY:[s,function(t){this.year=r(t)}],YYYY:[/\d{4}/,o("year")],Z:h,ZZ:h};function u(s){var i,a;i=s,a=n&&n.formats;for(var r=(s=i.replace(/(\[[^\]]+])|(LTS?|l{1,4}|L{1,4})/g,(function(e,s,i){var n=i&&i.toUpperCase();return s||a[i]||t[i]||a[n].replace(/(\[[^\]]+])|(MMMM|MM|DD|dddd)/g,(function(t,e,s){return e||s.slice(1)}))}))).match(e),o=r.length,h=0;h-1)return new Date(("X"===e?1e3:1)*t);var i=u(e)(t),a=i.year,n=i.month,r=i.day,o=i.hours,h=i.minutes,l=i.seconds,d=i.milliseconds,c=i.zone,p=new Date,f=r||(a||n?1:p.getDate()),g=a||p.getFullYear(),m=0;a&&!n||(m=n>0?n-1:p.getMonth());var b=o||0,y=h||0,v=l||0,w=d||0;return c?new Date(Date.UTC(g,m,f,b,y,v,w+60*c.offset*1e3)):s?new Date(Date.UTC(g,m,f,b,y,v,w)):new Date(g,m,f,b,y,v,w)}catch(t){return new Date("")}}(e,o,i),this.init(),c&&!0!==c&&(this.$L=this.locale(c).$L),d&&e!=this.format(o)&&(this.$d=new Date("")),n={}}else if(o instanceof Array)for(var p=o.length,f=1;f<=p;f+=1){r[1]=o[f-1];var g=s.apply(this,r);if(g.isValid()){this.$d=g.$d,this.$L=g.$L,this.init();break}f===p&&(this.$d=new Date(""))}else a.call(this,t)}}}()}));i.extend(a),s.parseDate=(t,e)=>{let s=!1;if(e)switch(e){case"ISO_8601":s=t;break;case"RFC_2822":s=i(t.slice(5),"DD MMM YYYY HH:mm:ss ZZ").unix();break;case"MYSQL":s=i(t,"YYYY-MM-DD hh:mm:ss").unix();break;case"UNIX":s=i(t).unix();break;default:s=i(t,e,!0).valueOf()}return s}}).call(this)}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{}],2:[function(t,e,s){"use strict";Object.defineProperty(s,"__esModule",{value:!0});const i=t=>"[object Object]"===Object.prototype.toString.call(t),a=(t,e)=>{const s=document.createElement(t);if(e&&"object"==typeof e)for(const t in e)"html"===t?s.innerHTML=e[t]:s.setAttribute(t,e[t]);return s},n=t=>{t instanceof NodeList?t.forEach((t=>n(t))):t.innerHTML=""},r=(t,e,s)=>a("li",{class:t,html:`${s}`}),o=(t,e)=>{let s,i;1===e?(s=0,i=t.length):-1===e&&(s=t.length-1,i=-1);for(let a=!0;a;){a=!1;for(let n=s;n!=i;n+=e)if(t[n+e]&&t[n].value>t[n+e].value){const s=t[n],i=t[n+e],r=s;t[n]=i,t[n+e]=r,a=!0}}return t};class h{constructor(t,e){return this.dt=t,this.rows=e,this}build(t){const e=a("tr");let s=this.dt.headings;return s.length||(s=t.map((()=>""))),s.forEach(((s,i)=>{const n=a("td");t[i]&&t[i].length||(t[i]=""),n.innerHTML=t[i],n.data=t[i],e.appendChild(n)})),e}render(t){return t}add(t){if(Array.isArray(t)){const e=this.dt;Array.isArray(t[0])?t.forEach((t=>{e.data.push(this.build(t))})):e.data.push(this.build(t)),e.data.length&&(e.hasRows=!0),this.update(),e.columns().rebuild()}}remove(t){const e=this.dt;Array.isArray(t)?(t.sort(((t,e)=>e-t)),t.forEach((t=>{e.data.splice(t,1)}))):"all"==t?e.data=[]:e.data.splice(t,1),e.data.length||(e.hasRows=!1),this.update(),e.columns().rebuild()}update(){this.dt.data.forEach(((t,e)=>{t.dataIndex=e}))}findRowIndex(t,e){return this.dt.data.findIndex((s=>s.children[t].innerText.toLowerCase().includes(String(e).toLowerCase())))}findRow(t,e){const s=this.findRowIndex(t,e);if(s<0)return{index:-1,row:null,cols:[]};const i=this.dt.data[s];return{index:s,row:i,cols:[...i.cells].map((t=>t.innerHTML))}}updateRow(t,e){const s=this.build(e);this.dt.data.splice(t,1,s),this.update(),this.dt.columns().rebuild()}}class l{constructor(t){return this.dt=t,this}swap(t){if(t.length&&2===t.length){const e=[];this.dt.headings.forEach(((t,s)=>{e.push(s)}));const s=t[0],i=t[1],a=e[i];e[i]=e[s],e[s]=a,this.order(e)}}order(t){let e,s,i,a,n,r,o;const h=[[],[],[],[]],l=this.dt;t.forEach(((t,i)=>{n=l.headings[t],r="false"!==n.getAttribute("data-sortable"),e=n.cloneNode(!0),e.originalCellIndex=i,e.sortable=r,h[0].push(e),l.hiddenColumns.includes(t)||(s=n.cloneNode(!0),s.originalCellIndex=i,s.sortable=r,h[1].push(s))})),l.data.forEach(((e,s)=>{i=e.cloneNode(!1),a=e.cloneNode(!1),i.dataIndex=a.dataIndex=s,null!==e.searchIndex&&void 0!==e.searchIndex&&(i.searchIndex=a.searchIndex=e.searchIndex),t.forEach((t=>{o=e.cells[t].cloneNode(!0),o.data=e.cells[t].data,i.appendChild(o),l.hiddenColumns.includes(t)||(o=e.cells[t].cloneNode(!0),o.data=e.cells[t].data,a.appendChild(o))})),h[2].push(i),h[3].push(a)})),l.headings=h[0],l.activeHeadings=h[1],l.data=h[2],l.activeRows=h[3],l.update()}hide(t){if(t.length){const e=this.dt;t.forEach((t=>{e.hiddenColumns.includes(t)||e.hiddenColumns.push(t)})),this.rebuild()}}show(t){if(t.length){let e;const s=this.dt;t.forEach((t=>{e=s.hiddenColumns.indexOf(t),e>-1&&s.hiddenColumns.splice(e,1)})),this.rebuild()}}visible(t){let e;const s=this.dt;return t=t||s.headings.map((t=>t.originalCellIndex)),isNaN(t)?Array.isArray(t)&&(e=[],t.forEach((t=>{e.push(!s.hiddenColumns.includes(t))}))):e=!s.hiddenColumns.includes(t),e}add(t){let e;const s=document.createElement("th");if(!this.dt.headings.length)return this.dt.insert({headings:[t.heading],data:t.data.map((t=>[t]))}),void this.rebuild();this.dt.hiddenHeader?s.innerHTML="":t.heading.nodeName?s.appendChild(t.heading):s.innerHTML=t.heading,this.dt.headings.push(s),this.dt.data.forEach(((s,i)=>{t.data[i]&&(e=document.createElement("td"),t.data[i].nodeName?e.appendChild(t.data[i]):e.innerHTML=t.data[i],e.data=e.innerHTML,t.render&&(e.innerHTML=t.render.call(this,e.data,e,s)),s.appendChild(e))})),t.type&&s.setAttribute("data-type",t.type),t.format&&s.setAttribute("data-format",t.format),t.hasOwnProperty("sortable")&&(s.sortable=t.sortable,s.setAttribute("data-sortable",!0===t.sortable?"true":"false")),this.rebuild(),this.dt.renderHeader()}remove(t){Array.isArray(t)?(t.sort(((t,e)=>e-t)),t.forEach((t=>this.remove(t)))):(this.dt.headings.splice(t,1),this.dt.data.forEach((e=>{e.removeChild(e.cells[t])}))),this.rebuild()}filter(t,e,s,i){const a=this.dt;if(a.filterState||(a.filterState={originalData:a.data}),!a.filterState[t]){const e=[...i,()=>!0];a.filterState[t]=function(){let t=0;return()=>e[t++%e.length]}()}const n=a.filterState[t](),r=Array.from(a.filterState.originalData).filter((e=>{const s=e.cells[t],i=s.hasAttribute("data-content")?s.getAttribute("data-content"):s.innerText;return"function"==typeof n?n(i):i===n}));a.data=r,a.data.length?(this.rebuild(),a.update()):(a.clear(),a.hasRows=!1,a.setMessage(a.options.labels.noRows)),s||a.emit("datatable.sort",t,e)}sort(e,s,i){const a=this.dt;if(a.hasHeadings&&(e<0||e>a.headings.length))return!1;const n=a.options.filters&&a.options.filters[a.headings[e].textContent];if(n&&0!==n.length)return void this.filter(e,s,i,n);a.sorting=!0,i||a.emit("datatable.sorting",e,s);let r=a.data;const h=[],l=[];let d=0,c=0;const u=a.headings[e],p=[];if("date"===u.getAttribute("data-type")){let e=!1;u.hasAttribute("data-format")&&(e=u.getAttribute("data-format")),p.push(Promise.resolve().then((function(){return t("./date-170bba30.js")})).then((({parseDate:t})=>s=>t(s,e))))}Promise.all(p).then((t=>{const n=t[0];let p,f;Array.from(r).forEach((t=>{const s=t.cells[e],i=s.hasAttribute("data-content")?s.getAttribute("data-content"):s.innerText;let a;a=n?n(i):"string"==typeof i?i.replace(/(\$|,|\s|%)/g,""):i,parseFloat(a)==a?l[c++]={value:Number(a),row:t}:h[d++]={value:"string"==typeof i?i.toLowerCase():i,row:t}})),s||(s=u.classList.contains("asc")?"desc":"asc"),"desc"==s?(p=o(h,-1),f=o(l,-1),u.classList.remove("asc"),u.classList.add("desc")):(p=o(l,1),f=o(h,1),u.classList.remove("desc"),u.classList.add("asc")),a.lastTh&&u!=a.lastTh&&(a.lastTh.classList.remove("desc"),a.lastTh.classList.remove("asc")),a.lastTh=u,r=p.concat(f),a.data=[];const g=[];r.forEach(((t,e)=>{a.data.push(t.row),null!==t.row.searchIndex&&void 0!==t.row.searchIndex&&g.push(e)})),a.searchData=g,this.rebuild(),a.update(),i||a.emit("datatable.sort",e,s)}))}rebuild(){let t,e,s,i;const a=this.dt,n=[];a.activeRows=[],a.activeHeadings=[],a.headings.forEach(((t,e)=>{t.originalCellIndex=e,t.sortable="false"!==t.getAttribute("data-sortable"),a.hiddenColumns.includes(e)||a.activeHeadings.push(t)})),a.data.forEach(((r,o)=>{t=r.cloneNode(!1),e=r.cloneNode(!1),t.dataIndex=e.dataIndex=o,null!==r.searchIndex&&void 0!==r.searchIndex&&(t.searchIndex=e.searchIndex=r.searchIndex),Array.from(r.cells).forEach((n=>{s=n.cloneNode(!0),s.data=n.data,t.appendChild(s),a.hiddenColumns.includes(s.cellIndex)||(i=s.cloneNode(!0),i.data=s.data,e.appendChild(i))})),n.push(t),a.activeRows.push(e)})),a.data=n,a.update()}}const d=function(t){let e=!1,s=!1;if((t=t||this.options.data).headings){e=a("thead");const s=a("tr");t.headings.forEach((t=>{const e=a("th",{html:t});s.appendChild(e)})),e.appendChild(s)}t.data&&t.data.length&&(s=a("tbody"),t.data.forEach((e=>{if(t.headings&&t.headings.length!==e.length)throw new Error("The number of rows do not match the number of headings.");const i=a("tr");e.forEach((t=>{const e=a("td",{html:t});i.appendChild(e)})),s.appendChild(i)}))),e&&(null!==this.dom.tHead&&this.dom.removeChild(this.dom.tHead),this.dom.appendChild(e)),s&&(this.dom.tBodies.length&&this.dom.removeChild(this.dom.tBodies[0]),this.dom.appendChild(s))},c={sortable:!0,searchable:!0,paging:!0,perPage:10,perPageSelect:[5,10,15,20,25],nextPrev:!0,firstLast:!1,prevText:"‹",nextText:"›",firstText:"«",lastText:"»",ellipsisText:"…",ascText:"â–´",descText:"â–¾",truncatePager:!0,pagerDelta:2,scrollY:"",fixedColumns:!0,fixedHeight:!1,header:!0,hiddenHeader:!1,footer:!1,labels:{placeholder:"Search...",perPage:"{select} entries per page",noRows:"No entries found",noResults:"No results match your search query",info:"Showing {start} to {end} of {rows} entries"},layout:{top:"{select}{search}",bottom:"{info}{pager}"}};class u{constructor(t,e={}){const s="string"==typeof t?document.querySelector(t):t;if(this.options={...c,...e,layout:{...c.layout,...e.layout},labels:{...c.labels,...e.labels}},this.initialized=!1,this.initialLayout=s.innerHTML,this.initialSortable=this.options.sortable,this.options.header||(this.options.sortable=!1),null===s.tHead&&(!this.options.data||this.options.data&&!this.options.data.headings)&&(this.options.sortable=!1),s.tBodies.length&&!s.tBodies[0].rows.length&&this.options.data&&!this.options.data.data)throw new Error("You seem to be using the data option, but you've not defined any rows.");this.dom=s,this.table=this.dom,this.listeners={onResize:t=>this.onResize(t)},this.init()}static extend(t,e){"function"==typeof e?u.prototype[t]=e:u[t]=e}init(t){if(this.initialized||this.dom.classList.contains("dataTable-table"))return!1;Object.assign(this.options,t||{}),this.currentPage=1,this.onFirstPage=!0,this.hiddenColumns=[],this.columnRenderers=[],this.selectedColumns=[],this.render(),setTimeout((()=>{this.emit("datatable.init"),this.initialized=!0,this.options.plugins&&Object.entries(this.options.plugins).forEach((([t,e])=>{this[t]&&"function"==typeof this[t]&&(this[t]=this[t](e,{createElement:a}),e.enabled&&this[t].init&&"function"==typeof this[t].init&&this[t].init())}))}),10)}render(t){if(t){switch(t){case"page":this.renderPage();break;case"pager":this.renderPager();break;case"header":this.renderHeader()}return!1}const e=this.options;let s="";if(e.data&&d.call(this),this.body=this.dom.tBodies[0],this.head=this.dom.tHead,this.foot=this.dom.tFoot,this.body||(this.body=a("tbody"),this.dom.appendChild(this.body)),this.hasRows=this.body.rows.length>0,!this.head){const t=a("thead"),s=a("tr");this.hasRows&&(Array.from(this.body.rows[0].cells).forEach((()=>{s.appendChild(a("th"))})),t.appendChild(s)),this.head=t,this.dom.insertBefore(this.head,this.body),this.hiddenHeader=e.hiddenHeader}if(this.headings=[],this.hasHeadings=this.head.rows.length>0,this.hasHeadings&&(this.header=this.head.rows[0],this.headings=[].slice.call(this.header.cells)),e.header||this.head&&this.dom.removeChild(this.dom.tHead),e.footer?this.head&&!this.foot&&(this.foot=a("tfoot",{html:this.head.innerHTML}),this.dom.appendChild(this.foot)):this.foot&&this.dom.removeChild(this.dom.tFoot),this.wrapper=a("div",{class:"dataTable-wrapper dataTable-loading"}),s+="
",s+=e.layout.top,s+="
",e.scrollY.length?s+=`
`:s+="
",s+="
",s+=e.layout.bottom,s+="
",s=s.replace("{info}",e.paging?"
":""),e.paging&&e.perPageSelect){let t="
";const i=a("select",{class:"dataTable-selector"});e.perPageSelect.forEach((t=>{const s=t===e.perPage,a=new Option(t,t,s,s);i.add(a)})),t=t.replace("{select}",i.outerHTML),s=s.replace("{select}",t)}else s=s.replace("{select}","");if(e.searchable){const t=``;s=s.replace("{search}",t)}else s=s.replace("{search}","");this.hasHeadings&&this.render("header"),this.dom.classList.add("dataTable-table");const i=a("nav",{class:"dataTable-pagination"}),n=a("ul",{class:"dataTable-pagination-list"});i.appendChild(n),s=s.replace(/\{pager\}/g,i.outerHTML),this.wrapper.innerHTML=s,this.container=this.wrapper.querySelector(".dataTable-container"),this.pagers=this.wrapper.querySelectorAll(".dataTable-pagination-list"),this.label=this.wrapper.querySelector(".dataTable-info"),this.dom.parentNode.replaceChild(this.wrapper,this.dom),this.container.appendChild(this.dom),this.rect=this.dom.getBoundingClientRect(),this.data=Array.from(this.body.rows),this.activeRows=this.data.slice(),this.activeHeadings=this.headings.slice(),this.update(),this.setColumns(),this.fixHeight(),this.fixColumns(),e.header||this.wrapper.classList.add("no-header"),e.footer||this.wrapper.classList.add("no-footer"),e.sortable&&this.wrapper.classList.add("sortable"),e.searchable&&this.wrapper.classList.add("searchable"),e.fixedHeight&&this.wrapper.classList.add("fixed-height"),e.fixedColumns&&this.wrapper.classList.add("fixed-columns"),this.bindEvents()}renderPage(){if(this.hasHeadings&&(n(this.header),this.activeHeadings.forEach((t=>this.header.appendChild(t)))),this.hasRows&&this.totalPages){this.currentPage>this.totalPages&&(this.currentPage=1);const t=this.currentPage-1,e=document.createDocumentFragment();this.pages[t].forEach((t=>e.appendChild(this.rows().render(t)))),this.clear(e),this.onFirstPage=1===this.currentPage,this.onLastPage=this.currentPage===this.lastPage}else this.setMessage(this.options.labels.noRows);let t,e=0,s=0,i=0;if(this.totalPages&&(e=this.currentPage-1,s=e*this.options.perPage,i=s+this.pages[e].length,s+=1,t=this.searching?this.searchData.length:this.data.length),this.label&&this.options.labels.info.length){const e=this.options.labels.info.replace("{start}",s).replace("{end}",i).replace("{page}",this.currentPage).replace("{pages}",this.totalPages).replace("{rows}",t);this.label.innerHTML=t?e:""}1==this.currentPage&&this.fixHeight()}renderPager(){if(n(this.pagers),this.totalPages>1){const t="pager",e=document.createDocumentFragment(),s=this.onFirstPage?1:this.currentPage-1,i=this.onLastPage?this.totalPages:this.currentPage+1;this.options.firstLast&&e.appendChild(r(t,1,this.options.firstText)),this.options.nextPrev&&!this.onFirstPage&&e.appendChild(r(t,s,this.options.prevText));let n=this.links;this.options.truncatePager&&(n=((t,e,s,i,n)=>{let r;const o=2*(i=i||2);let h=e-i,l=e+i;const d=[],c=[];e<4-i+o?l=3+o:e>s-(3-i+o)&&(h=s-(2+o));for(let e=1;e<=s;e++)if(1==e||e==s||e>=h&&e<=l){const s=t[e-1];s.classList.remove("active"),d.push(s)}return d.forEach((e=>{const s=e.children[0].getAttribute("data-page");if(r){const e=r.children[0].getAttribute("data-page");if(s-e==2)c.push(t[e]);else if(s-e!=1){const t=a("li",{class:"ellipsis",html:`${n}`});c.push(t)}}c.push(e),r=e})),c})(this.links,this.currentPage,this.pages.length,this.options.pagerDelta,this.options.ellipsisText)),this.links[this.currentPage-1].classList.add("active"),n.forEach((t=>{t.classList.remove("active"),e.appendChild(t)})),this.links[this.currentPage-1].classList.add("active"),this.options.nextPrev&&!this.onLastPage&&e.appendChild(r(t,i,this.options.nextText)),this.options.firstLast&&e.appendChild(r(t,this.totalPages,this.options.lastText)),this.pagers.forEach((t=>{t.appendChild(e.cloneNode(!0))}))}}renderHeader(){this.labels=[],this.headings&&this.headings.length&&this.headings.forEach(((t,e)=>{if(this.labels[e]=t.textContent,t.firstElementChild&&t.firstElementChild.classList.contains("dataTable-sorter")&&(t.innerHTML=t.firstElementChild.innerHTML),t.sortable="false"!==t.getAttribute("data-sortable"),t.originalCellIndex=e,this.options.sortable&&t.sortable){const e=a("a",{href:"#",class:"dataTable-sorter",html:t.innerHTML});t.innerHTML="",t.setAttribute("data-sortable",""),t.appendChild(e)}})),this.fixColumns()}bindEvents(){const t=this.options;if(t.perPageSelect){const e=this.wrapper.querySelector(".dataTable-selector");e&&e.addEventListener("change",(()=>{t.perPage=parseInt(e.value,10),this.update(),this.fixHeight(),this.emit("datatable.perpage",t.perPage)}),!1)}t.searchable&&(this.input=this.wrapper.querySelector(".dataTable-input"),this.input&&this.input.addEventListener("keyup",(()=>this.search(this.input.value)),!1)),this.wrapper.addEventListener("click",(e=>{const s=e.target.closest("a");s&&"a"===s.nodeName.toLowerCase()&&(s.hasAttribute("data-page")?(this.page(s.getAttribute("data-page")),e.preventDefault()):t.sortable&&s.classList.contains("dataTable-sorter")&&"false"!=s.parentNode.getAttribute("data-sortable")&&(this.columns().sort(this.headings.indexOf(s.parentNode)),e.preventDefault()))}),!1),window.addEventListener("resize",this.listeners.onResize)}onResize(){this.rect=this.container.getBoundingClientRect(),this.rect.width&&this.fixColumns()}setColumns(t){t||this.data.forEach((t=>{Array.from(t.cells).forEach((t=>{t.data=t.innerHTML}))})),this.options.columns&&this.headings.length&&this.options.columns.forEach((t=>{Array.isArray(t.select)||(t.select=[t.select]),t.hasOwnProperty("render")&&"function"==typeof t.render&&(this.selectedColumns=this.selectedColumns.concat(t.select),this.columnRenderers.push({columns:t.select,renderer:t.render})),t.select.forEach((e=>{const s=this.headings[e];t.type&&s.setAttribute("data-type",t.type),t.format&&s.setAttribute("data-format",t.format),t.hasOwnProperty("sortable")&&s.setAttribute("data-sortable",t.sortable),t.hasOwnProperty("hidden")&&!1!==t.hidden&&this.columns().hide([e]),t.hasOwnProperty("sort")&&1===t.select.length&&this.columns().sort(t.select[0],t.sort,!0)}))})),this.hasRows&&(this.data.forEach(((t,e)=>{t.dataIndex=e,Array.from(t.cells).forEach((t=>{t.data=t.innerHTML}))})),this.selectedColumns.length&&this.data.forEach((t=>{Array.from(t.cells).forEach(((e,s)=>{this.selectedColumns.includes(s)&&this.columnRenderers.forEach((i=>{i.columns.includes(s)&&(e.innerHTML=i.renderer.call(this,e.data,e,t))}))}))})),this.columns().rebuild()),this.render("header")}destroy(){this.dom.innerHTML=this.initialLayout,this.dom.classList.remove("dataTable-table"),this.wrapper.parentNode.replaceChild(this.dom,this.wrapper),this.initialized=!1,window.removeEventListener("resize",this.listeners.onResize)}update(){this.wrapper.classList.remove("dataTable-empty"),this.paginate(this),this.render("page"),this.links=[];let t=this.pages.length;for(;t--;){const e=t+1;this.links[t]=r(0===t?"active":"",e,e)}this.sorting=!1,this.render("pager"),this.rows().update(),this.emit("datatable.update")}paginate(){const t=this.options.perPage;let e=this.activeRows;return this.searching&&(e=[],this.searchData.forEach((t=>e.push(this.activeRows[t])))),this.options.paging?this.pages=e.map(((s,i)=>i%t==0?e.slice(i,i+t):null)).filter((t=>t)):this.pages=[e],this.totalPages=this.lastPage=this.pages.length,this.totalPages}fixColumns(){if((this.options.scrollY.length||this.options.fixedColumns)&&this.activeHeadings&&this.activeHeadings.length){let t,e=!1;if(this.columnWidths=[],this.dom.tHead){if(this.options.scrollY.length&&(e=a("thead"),e.appendChild(a("tr")),e.style.height="0px",this.headerTable&&(this.dom.tHead=this.headerTable.tHead)),this.activeHeadings.forEach((t=>{t.style.width=""})),this.activeHeadings.forEach(((t,s)=>{const i=t.offsetWidth,n=i/this.rect.width*100;if(t.style.width=`${n}%`,this.columnWidths[s]=i,this.options.scrollY.length){const t=a("th");e.firstElementChild.appendChild(t),t.style.width=`${n}%`,t.style.paddingTop="0",t.style.paddingBottom="0",t.style.border="0"}})),this.options.scrollY.length){const t=this.dom.parentElement;if(!this.headerTable){this.headerTable=a("table",{class:"dataTable-table"});const e=a("div",{class:"dataTable-headercontainer"});e.appendChild(this.headerTable),t.parentElement.insertBefore(e,t)}const s=this.dom.tHead;this.dom.replaceChild(e,s),this.headerTable.tHead=s,this.headerTable.parentElement.style.paddingRight=`${this.headerTable.clientWidth-this.dom.clientWidth+parseInt(this.headerTable.parentElement.style.paddingRight||"0",10)}px`,t.scrollHeight>t.clientHeight&&(t.style.overflowY="scroll")}}else{t=[],e=a("thead");const s=a("tr");Array.from(this.dom.tBodies[0].rows[0].cells).forEach((()=>{const e=a("th");s.appendChild(e),t.push(e)})),e.appendChild(s),this.dom.insertBefore(e,this.body);const i=[];t.forEach(((t,e)=>{const s=t.offsetWidth,a=s/this.rect.width*100;i.push(a),this.columnWidths[e]=s})),this.data.forEach((t=>{Array.from(t.cells).forEach(((t,e)=>{this.columns(t.cellIndex).visible()&&(t.style.width=`${i[e]}%`)}))})),this.dom.removeChild(e)}}}fixHeight(){this.options.fixedHeight&&(this.container.style.height=null,this.rect=this.container.getBoundingClientRect(),this.container.style.height=`${this.rect.height}px`)}search(t){return!!this.hasRows&&(t=t.toLowerCase(),this.currentPage=1,this.searching=!0,this.searchData=[],t.length?(this.clear(),this.data.forEach(((e,s)=>{const i=this.searchData.includes(e);t.split(" ").reduce(((t,s)=>{let i=!1,a=null,n=null;for(let t=0;tthis.pages.length||t<0)&&(this.render("page"),this.render("pager"),void this.emit("datatable.page",t)))}sortColumn(t,e){this.columns().sort(t,e)}insert(t){let e=[];if(i(t)){if(t.headings&&!this.hasHeadings&&!this.hasRows){const e=a("tr");t.headings.forEach((t=>{const s=a("th",{html:t});e.appendChild(s)})),this.head.appendChild(e),this.header=e,this.headings=[].slice.call(e.cells),this.hasHeadings=!0,this.options.sortable=this.initialSortable,this.render("header"),this.activeHeadings=this.headings.slice()}t.data&&Array.isArray(t.data)&&(e=t.data)}else Array.isArray(t)&&t.forEach((t=>{const s=[];Object.entries(t).forEach((([t,e])=>{const i=this.labels.indexOf(t);i>-1&&(s[i]=e)})),e.push(s)}));e.length&&(this.rows().add(e),this.hasRows=!0),this.update(),this.setColumns(),this.fixColumns()}refresh(){this.options.searchable&&(this.input.value="",this.searching=!1),this.currentPage=1,this.onFirstPage=!0,this.update(),this.emit("datatable.refresh")}clear(t){this.body&&n(this.body);let e=this.body;this.body||(e=this.dom),t&&("string"==typeof t&&(document.createDocumentFragment().innerHTML=t),e.appendChild(t))}export(t){if(!this.hasHeadings&&!this.hasRows)return!1;const e=this.activeHeadings;let s=[];const a=[];let n,r,o,h;if(!i(t))return!1;const l={download:!0,skipColumn:[],lineDelimiter:"\n",columnDelimiter:",",tableName:"myTable",replacer:null,space:4,...t};if(l.type){if("txt"!==l.type&&"csv"!==l.type||(s[0]=this.header),l.selection)if(isNaN(l.selection)){if(Array.isArray(l.selection))for(n=0;n{e.data[i]=[];const a=t.split(s.columnDelimiter);a.length&&a.forEach((t=>{e.data[i].push(t)}))})))}else if("json"===s.type){const t=(t=>{let e=!1;try{e=JSON.parse(t)}catch(t){return!1}return!(null===e||!Array.isArray(e)&&!i(e))&&e})(s.data);t&&(e={headings:[],data:[]},t.forEach(((t,s)=>{e.data[s]=[],Object.entries(t).forEach((([t,i])=>{e.headings.includes(t)||e.headings.push(t),e.data[s].push(i)}))})))}i(s.data)&&(e=s.data),e&&this.insert(e)}return!1}print(){const t=this.activeHeadings,e=this.activeRows,s=a("table"),i=a("thead"),n=a("tbody"),r=a("tr");t.forEach((t=>{r.appendChild(a("th",{html:t.textContent}))})),i.appendChild(r),e.forEach((t=>{const e=a("tr");Array.from(t.cells).forEach((t=>{e.appendChild(a("td",{html:t.textContent}))})),n.appendChild(e)})),s.appendChild(i),s.appendChild(n);const o=window.open();o.document.body.appendChild(s),o.print()}setMessage(t){let e=1;this.hasRows?e=this.data[0].cells.length:this.activeHeadings.length&&(e=this.activeHeadings.length),this.wrapper.classList.add("dataTable-empty"),this.label&&(this.label.innerHTML=""),this.totalPages=0,this.render("pager"),this.clear(a("tr",{html:`${t}`}))}columns(t){return new l(this,t)}rows(t){return new h(this,t)}on(t,e){this.events=this.events||{},this.events[t]=this.events[t]||[],this.events[t].push(e)}off(t,e){this.events=this.events||{},t in this.events!=0&&this.events[t].splice(this.events[t].indexOf(e),1)}emit(t){if(this.events=this.events||{},t in this.events!=0)for(let e=0;e + + + + + + Zarf SBOM Viewer + + + + + +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Zarf SBOM Viewer +
+
+ +
+ +
+
+
+ + + + + +
+

+
+ +
+ + + + + diff --git a/src/internal/packager2/layout/viewer/theme.css b/src/internal/packager2/layout/viewer/theme.css new file mode 100644 index 0000000000..ed2b7fd9a4 --- /dev/null +++ b/src/internal/packager2/layout/viewer/theme.css @@ -0,0 +1,173 @@ +.dataTable-wrapper.no-header .dataTable-container { + border-top: 1px solid #d9d9d9; +} + +.dataTable-wrapper.no-footer .dataTable-container { + border-bottom: 1px solid #d9d9d9; +} + +.dataTable-top, +.dataTable-bottom { + padding: 8px 10px; +} + +.dataTable-top > nav:first-child, +.dataTable-top > div:first-child, +.dataTable-bottom > nav:first-child, +.dataTable-bottom > div:first-child { + float: left; +} + +.dataTable-top > nav:last-child, +.dataTable-top > div:last-child, +.dataTable-bottom > nav:last-child, +.dataTable-bottom > div:last-child { + float: right; +} + +.dataTable-selector { + padding: 6px; +} + +.dataTable-input { + padding: 6px 12px; +} + +.dataTable-info { + margin: 7px 0; +} + +/* PAGER */ +.dataTable-pagination ul { + margin: 0; + padding-left: 0; +} + +.dataTable-pagination li { + list-style: none; + float: left; +} + +.dataTable-pagination a { + border: 1px solid transparent; + float: left; + margin-left: 2px; + padding: 6px 12px; + position: relative; + text-decoration: none; + color: #333; +} + +.dataTable-pagination a:hover { + background-color: #d9d9d9; +} + +.dataTable-pagination .active a, +.dataTable-pagination .active a:focus, +.dataTable-pagination .active a:hover { + background-color: #d9d9d9; + cursor: default; +} + +.dataTable-pagination .ellipsis a, +.dataTable-pagination .disabled a, +.dataTable-pagination .disabled a:focus, +.dataTable-pagination .disabled a:hover { + cursor: not-allowed; +} + +.dataTable-pagination .disabled a, +.dataTable-pagination .disabled a:focus, +.dataTable-pagination .disabled a:hover { + cursor: not-allowed; + opacity: 0.4; +} + +.dataTable-pagination .pager a { + font-weight: bold; +} + +/* TABLE */ +.dataTable-table { + max-width: 100%; + width: 100%; + border-spacing: 0; + border-collapse: separate; +} + +.dataTable-table > tbody > tr > td, +.dataTable-table > tbody > tr > th, +.dataTable-table > tfoot > tr > td, +.dataTable-table > tfoot > tr > th, +.dataTable-table > thead > tr > td, +.dataTable-table > thead > tr > th { + vertical-align: top; + padding: 8px 10px; +} + +.dataTable-table > thead > tr > th { + vertical-align: bottom; + text-align: left; + border-bottom: 1px solid #d9d9d9; +} + +.dataTable-table > tfoot > tr > th { + vertical-align: bottom; + text-align: left; + border-top: 1px solid #d9d9d9; +} + +.dataTable-table th { + vertical-align: bottom; + text-align: left; +} + +.dataTable-table th a { + text-decoration: none; + color: inherit; +} + +.dataTable-sorter { + display: inline-block; + height: 100%; + position: relative; + width: 100%; +} + +.dataTable-sorter::before, +.dataTable-sorter::after { + content: ""; + height: 0; + width: 0; + position: absolute; + right: 4px; + border-left: 4px solid transparent; + border-right: 4px solid transparent; + opacity: 0.2; +} + +.dataTable-sorter::before { + border-top: 4px solid #000; + bottom: 0px; +} + +.dataTable-sorter::after { + border-bottom: 4px solid #000; + border-top: 4px solid transparent; + top: 0px; +} + +.asc .dataTable-sorter::after, +.desc .dataTable-sorter::before { + opacity: 0.6; +} + +.dataTables-empty { + text-align: center; +} + +.dataTable-top::after, .dataTable-bottom::after { + clear: both; + content: " "; + display: table; +} \ No newline at end of file diff --git a/src/internal/packager2/layout/viewer/viewer.js b/src/internal/packager2/layout/viewer/viewer.js new file mode 100644 index 0000000000..17475c78d3 --- /dev/null +++ b/src/internal/packager2/layout/viewer/viewer.js @@ -0,0 +1,51 @@ +function initSelector() { + const url = /sbom-viewer-(.*).html*$/gim.exec(window.location.href)[1]; + + ZARF_SBOM_LIST.sort().forEach((item) => { + let selected = url === item ? 'selected' : ''; + sbomSelector.add(new Option(item, item, selected, selected)); + }); +} + +function initData() { + const payload = ZARF_SBOM_DATA; + + const transformedData = payload.artifacts.map((artifact) => { + return [ + artifact.type, + artifact.name, + artifact.version, + fileList(artifact.locations, artifact.name), + (artifact.metadata && fileList(artifact.metadata.files, artifact.name)) || '-', + (artifact.metadata && artifact.metadata.description) || '-', + ((artifact.metadata && artifact.metadata.maintainer) || '-').replace( + /\u003c(.*)\u003e/, + mailtoMaintainerReplace + ), + (artifact.metadata && artifact.metadata.installedSize) || '-' + ]; + }); + + const data = { + headings: ['Type', 'Name', 'Version', 'Sources', 'Package Files', 'Notes', 'Maintainer', 'Size'], + data: transformedData + }; + + if (window.dt) { + window.dt.destroy(); + } + + distroInfo.innerHTML = payload.distro.prettyName || 'No Base Image Detected'; + + window.dt = new simpleDatatables.DataTable(artifactsTable, { + data, + perPage: 20 + }); +} + +function compare() { + window.location.href = 'compare.html'; +} + +initSelector(); +initData(); diff --git a/src/test/e2e/00_use_cli_test.go b/src/test/e2e/00_use_cli_test.go index cfaf340436..1f2a3e2ecc 100644 --- a/src/test/e2e/00_use_cli_test.go +++ b/src/test/e2e/00_use_cli_test.go @@ -137,7 +137,6 @@ func TestUseCLI(t *testing.T) { tmpdir := t.TempDir() cacheDir := filepath.Join(t.TempDir(), ".cache-location") stdOut, stdErr, err := e2e.Zarf(t, "package", "create", "examples/dos-games", "--zarf-cache", cacheDir, "--tmpdir", tmpdir, "--log-level=debug", "-o=build", "--confirm") - require.Contains(t, stdErr, tmpdir, "The other tmp path should show as being created") require.NoError(t, err, stdOut, stdErr) files, err := os.ReadDir(filepath.Join(cacheDir, "images")) diff --git a/src/test/e2e/09_component_compose_test.go b/src/test/e2e/09_component_compose_test.go index d0e23f08f0..da654362cc 100644 --- a/src/test/e2e/09_component_compose_test.go +++ b/src/test/e2e/09_component_compose_test.go @@ -108,13 +108,13 @@ func TestFullComposability(t *testing.T) { - files/service.yaml kustomizations: - files - - files + - files/ - name: connect-service-two namespace: podinfo-compose-two files: - files/service.yaml kustomizations: - - files + - files/ charts: - name: podinfo-compose version: 6.4.0 @@ -160,8 +160,7 @@ func TestFullComposability(t *testing.T) { before: - dir: sub-package cmd: ls - - dir: . - cmd: ls + - cmd: ls onDeploy: after: - cmd: cat coffee-ipsum.txt From 8df4a7016e4c91b8660081f22844bc6d117099ff Mon Sep 17 00:00:00 2001 From: Philip Laine Date: Wed, 9 Oct 2024 10:58:53 +0200 Subject: [PATCH 2/2] refactor: normal creator Signed-off-by: Philip Laine --- examples/manifests/zarf.yaml | 2 +- src/internal/packager2/layout/import.go | 4 +++- src/internal/packager2/layout/import_test.go | 2 +- 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/examples/manifests/zarf.yaml b/examples/manifests/zarf.yaml index 6c97e00abd..4bead0b8c7 100644 --- a/examples/manifests/zarf.yaml +++ b/examples/manifests/zarf.yaml @@ -39,7 +39,7 @@ components: kustomizations: # kustomizations can be specified relative to the `zarf.yaml` or as remoteBuild resources with the # following syntax: https://github.com/kubernetes-sigs/kustomize/blob/master/examples/remoteBuild.md: - - https://github.com/stefanprodan/podinfo//kustomize?ref=6.4.0 + - https://github.com/stefanprodan/podinfo/kustomize?ref=6.4.0 # while ?ref= is not a requirement, it is recommended to use a specific commit hash / git tag to # ensure that the kustomization is not changed in a way that breaks your deployment. # image discovery is supported in all manifests and charts using: diff --git a/src/internal/packager2/layout/import.go b/src/internal/packager2/layout/import.go index 2415e2399d..af7d2ad7a4 100644 --- a/src/internal/packager2/layout/import.go +++ b/src/internal/packager2/layout/import.go @@ -7,6 +7,7 @@ import ( "context" "errors" "fmt" + "maps" "os" "path/filepath" "slices" @@ -48,8 +49,9 @@ func resolveImports(ctx context.Context, pkg v1alpha1.ZarfPackage, packagePath, importPath := filepath.Join(packagePath, component.Import.Path) importKey := fmt.Sprintf("%s-%s", component.Name, importPath) if _, ok := seenImports[importKey]; ok { - return v1alpha1.ZarfPackage{}, fmt.Errorf("package %s imported in cycle by %s", filepath.ToSlash(importPath), filepath.ToSlash(packagePath)) + return v1alpha1.ZarfPackage{}, fmt.Errorf("package %s imported in cycle by %s in component %s", filepath.ToSlash(importPath), filepath.ToSlash(packagePath), component.Name) } + seenImports = maps.Clone(seenImports) seenImports[importKey] = nil b, err := os.ReadFile(filepath.Join(importPath, layout.ZarfYAML)) if err != nil { diff --git a/src/internal/packager2/layout/import_test.go b/src/internal/packager2/layout/import_test.go index e66712adbf..04d84a6d6b 100644 --- a/src/internal/packager2/layout/import_test.go +++ b/src/internal/packager2/layout/import_test.go @@ -29,7 +29,7 @@ func TestResolveImportsCircular(t *testing.T) { require.NoError(t, err) _, err = resolveImports(ctx, pkg, "./testdata/import/first", "", "", map[string]interface{}{}) - require.EqualError(t, err, "package testdata/import/second imported in cycle by testdata/import/third") + require.EqualError(t, err, "package testdata/import/second imported in cycle by testdata/import/third in component component") } func TestValidateComponentCompose(t *testing.T) {