From 1e5a0e7647d328ecdc429ebc2717d70e1e17b7b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Cl=C3=A9ment=20Legrand?= Date: Sat, 14 Oct 2023 00:01:52 +0200 Subject: [PATCH] add plugin tar inclusion MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Clément Legrand --- USERS.md | 1 + .../commands/argocd_repo_server.go | 3 ++ cmd/argocd/commands/app.go | 2 +- cmpserver/plugin/plugin_test.go | 6 +-- .../config-management-plugins.md | 12 ++++- .../server-commands/argocd-repo-server.md | 1 + docs/user-guide/commands/argocd_app_diff.md | 2 +- go.sum | 2 + .../argocd-repo-server-deployment.yaml | 6 +++ manifests/core-install.yaml | 6 +++ manifests/ha/install.yaml | 6 +++ manifests/ha/namespace-install.yaml | 6 +++ manifests/install.yaml | 6 +++ manifests/namespace-install.yaml | 6 +++ reposerver/repository/repository.go | 42 ++++++++++------- reposerver/repository/repository_test.go | 6 +-- util/app/discovery/discovery.go | 22 ++++----- util/app/discovery/discovery_test.go | 14 +++--- util/cmp/stream.go | 8 ++-- util/cmp/stream_test.go | 6 +-- util/io/files/tar.go | 7 +-- util/io/files/tar_test.go | 47 ++++++++++++++++--- util/io/files/testdata/app/git/index | 0 .../src/domain/service/deploy/helmfile.yaml | 0 .../src/domain/service/deploy/template.tpl | 0 25 files changed, 157 insertions(+), 60 deletions(-) create mode 100644 util/io/files/testdata/app/git/index create mode 100644 util/io/files/testdata/app/src/domain/service/deploy/helmfile.yaml create mode 100644 util/io/files/testdata/app/src/domain/service/deploy/template.tpl diff --git a/USERS.md b/USERS.md index 3f164796d099f1..6a64917b15aabe 100644 --- a/USERS.md +++ b/USERS.md @@ -121,6 +121,7 @@ Currently, the following organizations are **officially** using Argo CD: 1. [Grupo MasMovil](https://grupomasmovil.com/en/) 1. [Handelsbanken](https://www.handelsbanken.se) 1. [Healy](https://www.healyworld.net) +1. [Heetch](https://www.heetch.com) 1. [Helio](https://helio.exchange) 1. [Hetki](https://hetki.ai) 1. [hipages](https://hipages.com.au/) diff --git a/cmd/argocd-repo-server/commands/argocd_repo_server.go b/cmd/argocd-repo-server/commands/argocd_repo_server.go index 84b50e7cd5ab92..c4cbbcf4859629 100644 --- a/cmd/argocd-repo-server/commands/argocd_repo_server.go +++ b/cmd/argocd-repo-server/commands/argocd_repo_server.go @@ -64,6 +64,7 @@ func NewCommand() *cobra.Command { disableTLS bool maxCombinedDirectoryManifestsSize string cmpTarExcludedGlobs []string + cmpTarIncludedGlobs []string allowOutOfBoundsSymlinks bool streamedManifestMaxTarSize string streamedManifestMaxExtractedSize string @@ -121,6 +122,7 @@ func NewCommand() *cobra.Command { SubmoduleEnabled: gitSubmoduleEnabled, MaxCombinedDirectoryManifestsSize: maxCombinedDirectoryManifestsQuantity, CMPTarExcludedGlobs: cmpTarExcludedGlobs, + CMPTarIncludedGlobs: cmpTarIncludedGlobs, AllowOutOfBoundsSymlinks: allowOutOfBoundsSymlinks, StreamedManifestMaxExtractedSize: streamedManifestMaxExtractedSizeQuantity.ToDec().Value(), StreamedManifestMaxTarSize: streamedManifestMaxTarSizeQuantity.ToDec().Value(), @@ -204,6 +206,7 @@ func NewCommand() *cobra.Command { command.Flags().BoolVar(&disableTLS, "disable-tls", env.ParseBoolFromEnv("ARGOCD_REPO_SERVER_DISABLE_TLS", false), "Disable TLS on the gRPC endpoint") command.Flags().StringVar(&maxCombinedDirectoryManifestsSize, "max-combined-directory-manifests-size", env.StringFromEnv("ARGOCD_REPO_SERVER_MAX_COMBINED_DIRECTORY_MANIFESTS_SIZE", "10M"), "Max combined size of manifest files in a directory-type Application") command.Flags().StringArrayVar(&cmpTarExcludedGlobs, "plugin-tar-exclude", env.StringsFromEnv("ARGOCD_REPO_SERVER_PLUGIN_TAR_EXCLUSIONS", []string{}, ";"), "Globs to filter when sending tarballs to plugins.") + command.Flags().StringArrayVar(&cmpTarIncludedGlobs, "plugin-tar-include", env.StringsFromEnv("ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS", []string{}, ";"), "Globs to include files when sending tarballs to plugins.") command.Flags().BoolVar(&allowOutOfBoundsSymlinks, "allow-oob-symlinks", env.ParseBoolFromEnv("ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS", false), "Allow out-of-bounds symlinks in repositories (not recommended)") command.Flags().StringVar(&streamedManifestMaxTarSize, "streamed-manifest-max-tar-size", env.StringFromEnv("ARGOCD_REPO_SERVER_STREAMED_MANIFEST_MAX_TAR_SIZE", "100M"), "Maximum size of streamed manifest archives") command.Flags().StringVar(&streamedManifestMaxExtractedSize, "streamed-manifest-max-extracted-size", env.StringFromEnv("ARGOCD_REPO_SERVER_STREAMED_MANIFEST_MAX_EXTRACTED_SIZE", "1G"), "Maximum size of streamed manifest archives when extracted") diff --git a/cmd/argocd/commands/app.go b/cmd/argocd/commands/app.go index f18a4fb34fa324..6192138536dc8f 100644 --- a/cmd/argocd/commands/app.go +++ b/cmd/argocd/commands/app.go @@ -1136,7 +1136,7 @@ func NewApplicationDiffCommand(clientOpts *argocdclient.ClientOptions) *cobra.Co command.Flags().StringVar(&revision, "revision", "", "Compare live app to a particular revision") command.Flags().StringVar(&localRepoRoot, "local-repo-root", "/", "Path to the repository root. Used together with --local allows setting the repository root") command.Flags().BoolVar(&serverSideGenerate, "server-side-generate", false, "Used with --local, this will send your manifests to the server for diffing") - command.Flags().StringArrayVar(&localIncludes, "local-include", []string{"*.yaml", "*.yml", "*.json"}, "Used with --server-side-generate, specify patterns of filenames to send. Matching is based on filename and not path.") + command.Flags().StringArrayVar(&localIncludes, "local-include", []string{"**/*.yaml", "**/*.yml", "**/*.json"}, "Used with --server-side-generate, specify patterns of filenames to send. It does support glob patterns") return command } diff --git a/cmpserver/plugin/plugin_test.go b/cmpserver/plugin/plugin_test.go index b253dc414cbdc8..1e55a06793119d 100644 --- a/cmpserver/plugin/plugin_test.go +++ b/cmpserver/plugin/plugin_test.go @@ -541,7 +541,7 @@ type MockGenerateManifestStream struct { } func NewMockGenerateManifestStream(repoPath, appPath string, env []string) (*MockGenerateManifestStream, error) { - tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil) + tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil, nil) if err != nil { return nil, err } @@ -615,7 +615,7 @@ type MockMatchRepositoryStream struct { } func NewMockMatchRepositoryStream(repoPath, appPath string, env []string) (*MockMatchRepositoryStream, error) { - tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil) + tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil, nil) if err != nil { return nil, err } @@ -688,7 +688,7 @@ type MockParametersAnnouncementStream struct { } func NewMockParametersAnnouncementStream(repoPath, appPath string, env []string) (*MockParametersAnnouncementStream, error) { - tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil) + tgz, mr, err := cmp.GetCompressedRepoAndMetadata(repoPath, appPath, env, nil, nil, nil) if err != nil { return nil, err } diff --git a/docs/operator-manual/config-management-plugins.md b/docs/operator-manual/config-management-plugins.md index 7c86075ff2f7fb..8c51b1ee011004 100644 --- a/docs/operator-manual/config-management-plugins.md +++ b/docs/operator-manual/config-management-plugins.md @@ -344,7 +344,7 @@ If you are actively developing a sidecar-installed CMP, keep a few things in min | -- | -- | | `no matches for kind "ConfigManagementPlugin" in version "argoproj.io/v1alpha1"` | The `ConfigManagementPlugin` CRD was deprecated in Argo CD 2.4 and removed in 2.8. This error means you've tried to put the configuration for your plugin directly into Kubernetes as a CRD. Refer to this [section of documentation](#write-the-plugin-configuration-file) for how to write the plugin configuration file and place it properly in the sidecar. | -## Plugin tar stream exclusions +## Plugin tar stream exclusions or inclusions In order to increase the speed of manifest generation, certain files and folders can be excluded from being sent to your plugin. We recommend excluding your `.git` folder if it isn't necessary. Use Go's @@ -356,6 +356,16 @@ You can set it one of three ways: 2. The `reposerver.plugin.tar.exclusions` key if you are using `argocd-cmd-params-cm` 3. Directly setting `ARGOCD_REPO_SERVER_PLUGIN_TAR_EXCLUSIONS` environment variable on the repo server. +In addition, if you're using a monorepo mixed with code and GitOps resources, you probably want to only include files +involved in manifests generation in the tar archive. Use Go's [filepatch.Match](https://pkg.go.dev/path/filepath#Match) syntax to include +desired files. For example, `*.y*ml` to include `.yaml` or `.yml` files. + +You can set it one of three ways: + +1. The `--plugin-tar-include` argument on the repo server. +2. The `reposerver.plugin.tar.inclusions` key if you are using `argocd-cmd-params-cm` +3. Directly setting `ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS` environment variable on the repo server. + For option 1, the flag can be repeated multiple times. For option 2 and 3, you can specify multiple globs by separating them with semicolons. diff --git a/docs/operator-manual/server-commands/argocd-repo-server.md b/docs/operator-manual/server-commands/argocd-repo-server.md index 7be45fe18d26ff..db6ada1bf7a94e 100644 --- a/docs/operator-manual/server-commands/argocd-repo-server.md +++ b/docs/operator-manual/server-commands/argocd-repo-server.md @@ -33,6 +33,7 @@ argocd-repo-server [flags] --otlp-insecure OpenTelemetry collector insecure mode (default true) --parallelismlimit int Limit on number of concurrent manifests generate requests. Any value less the 1 means no limit. --plugin-tar-exclude stringArray Globs to filter when sending tarballs to plugins. + --plugin-tar-include stringArray Globs to include files when sending tarballs to plugins. --port int Listen on given port for incoming connections (default 8081) --redis string Redis server hostname and port (e.g. argocd-redis:6379). --redis-ca-certificate string Path to Redis server CA certificate (e.g. /etc/certs/redis/ca.crt). If not specified, system trusted CAs will be used for server certificate validation. diff --git a/docs/user-guide/commands/argocd_app_diff.md b/docs/user-guide/commands/argocd_app_diff.md index 139584d4fead58..9e31b1dc559c0c 100644 --- a/docs/user-guide/commands/argocd_app_diff.md +++ b/docs/user-guide/commands/argocd_app_diff.md @@ -21,7 +21,7 @@ argocd app diff APPNAME [flags] --hard-refresh Refresh application data as well as target manifests cache -h, --help help for diff --local string Compare live app to a local manifests - --local-include stringArray Used with --server-side-generate, specify patterns of filenames to send. Matching is based on filename and not path. (default [*.yaml,*.yml,*.json]) + --local-include stringArray Used with --server-side-generate, specify patterns of filenames to send. It does support glob patterns (default [**/*.yaml,**/*.yml,**/*.json]) --local-repo-root string Path to the repository root. Used together with --local allows setting the repository root (default "/") --refresh Refresh application data when retrieving --revision string Compare live app to a particular revision diff --git a/go.sum b/go.sum index 2d33e5a248ccec..2514a1df65619d 100644 --- a/go.sum +++ b/go.sum @@ -753,6 +753,8 @@ github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6r github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= github.com/blang/semver/v4 v4.0.0 h1:1PFHFE6yCCTv8C1TeyNNarDzntLi7wMI5i/pzqYIsAM= github.com/blang/semver/v4 v4.0.0/go.mod h1:IbckMUScFkM3pff0VJDNKRiT6TG/YpiHIM2yvyW5YoQ= +github.com/bmatcuk/doublestar v1.3.4 h1:gPypJ5xD31uhX6Tf54sDPUOBXTqKH4c9aPY66CyQrS0= +github.com/bmatcuk/doublestar v1.3.4/go.mod h1:wiQtGV+rzVYxB7WIlirSN++5HPtPlXEo9MEoZQC/PmE= github.com/bmatcuk/doublestar/v4 v4.6.0 h1:HTuxyug8GyFbRkrffIpzNCSK4luc0TY3wzXvzIZhEXc= github.com/bmatcuk/doublestar/v4 v4.6.0/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc= github.com/bombsimon/logrusr/v2 v2.0.1 h1:1VgxVNQMCvjirZIYaT9JYn6sAVGVEcNtRE0y4mvaOAM= diff --git a/manifests/base/repo-server/argocd-repo-server-deployment.yaml b/manifests/base/repo-server/argocd-repo-server-deployment.yaml index 907bc80a34e560..633532cbf68b4d 100644 --- a/manifests/base/repo-server/argocd-repo-server-deployment.yaml +++ b/manifests/base/repo-server/argocd-repo-server-deployment.yaml @@ -144,6 +144,12 @@ spec: name: argocd-cmd-params-cm key: reposerver.plugin.tar.exclusions optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/manifests/core-install.yaml b/manifests/core-install.yaml index 254cd6e22044f7..591861200011c8 100644 --- a/manifests/core-install.yaml +++ b/manifests/core-install.yaml @@ -21309,6 +21309,12 @@ spec: key: reposerver.plugin.tar.exclusions name: argocd-cmd-params-cm optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/manifests/ha/install.yaml b/manifests/ha/install.yaml index 83fc7a0f1c864b..270534cb9da00b 100644 --- a/manifests/ha/install.yaml +++ b/manifests/ha/install.yaml @@ -22908,6 +22908,12 @@ spec: key: reposerver.plugin.tar.exclusions name: argocd-cmd-params-cm optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/manifests/ha/namespace-install.yaml b/manifests/ha/namespace-install.yaml index 044a061bf0cb1b..553e28e85ad953 100644 --- a/manifests/ha/namespace-install.yaml +++ b/manifests/ha/namespace-install.yaml @@ -2174,6 +2174,12 @@ spec: key: reposerver.plugin.tar.exclusions name: argocd-cmd-params-cm optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/manifests/install.yaml b/manifests/install.yaml index 6f9c88dbb9d57e..7f5b4dfb8234f6 100644 --- a/manifests/install.yaml +++ b/manifests/install.yaml @@ -21954,6 +21954,12 @@ spec: key: reposerver.plugin.tar.exclusions name: argocd-cmd-params-cm optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/manifests/namespace-install.yaml b/manifests/namespace-install.yaml index cb58228423c11f..19819295bb29c5 100644 --- a/manifests/namespace-install.yaml +++ b/manifests/namespace-install.yaml @@ -1220,6 +1220,12 @@ spec: key: reposerver.plugin.tar.exclusions name: argocd-cmd-params-cm optional: true + - name: ARGOCD_REPO_SERVER_PLUGIN_TAR_INCLUSIONS + valueFrom: + configMapKeyRef: + key: reposerver.plugin.tar.inclusions + name: argocd-cmd-params-cm + optional: true - name: ARGOCD_REPO_SERVER_ALLOW_OUT_OF_BOUNDS_SYMLINKS valueFrom: configMapKeyRef: diff --git a/reposerver/repository/repository.go b/reposerver/repository/repository.go index 5d11a6438272d6..4090165a742191 100644 --- a/reposerver/repository/repository.go +++ b/reposerver/repository/repository.go @@ -103,6 +103,7 @@ type RepoServerInitConstants struct { SubmoduleEnabled bool MaxCombinedDirectoryManifestsSize resource.Quantity CMPTarExcludedGlobs []string + CMPTarIncludedGlobs []string AllowOutOfBoundsSymlinks bool StreamedManifestMaxExtractedSize int64 StreamedManifestMaxTarSize int64 @@ -220,7 +221,7 @@ func (s *Service) ListApps(ctx context.Context, q *apiclient.ListAppsRequest) (* } defer io.Close(closer) - apps, err := discovery.Discover(ctx, gitClient.Root(), gitClient.Root(), q.EnabledSourceTypes, s.initConstants.CMPTarExcludedGlobs) + apps, err := discovery.Discover(ctx, gitClient.Root(), gitClient.Root(), q.EnabledSourceTypes, s.initConstants.CMPTarExcludedGlobs, s.initConstants.CMPTarIncludedGlobs) if err != nil { return nil, fmt.Errorf("error discovering applications: %w", err) } @@ -794,7 +795,7 @@ func (s *Service) runManifestGenAsync(ctx context.Context, repoRoot, commitSHA, } } - manifestGenResult, err = GenerateManifests(ctx, opContext.appPath, repoRoot, commitSHA, q, false, s.gitCredsStore, s.initConstants.MaxCombinedDirectoryManifestsSize, s.gitRepoPaths, WithCMPTarDoneChannel(ch.tarDoneCh), WithCMPTarExcludedGlobs(s.initConstants.CMPTarExcludedGlobs)) + manifestGenResult, err = GenerateManifests(ctx, opContext.appPath, repoRoot, commitSHA, q, false, s.gitCredsStore, s.initConstants.MaxCombinedDirectoryManifestsSize, s.gitRepoPaths, WithCMPTarDoneChannel(ch.tarDoneCh), WithCMPTarExcludedGlobs(s.initConstants.CMPTarExcludedGlobs), WithCMPTarIncludedGlobs(s.initConstants.CMPTarIncludedGlobs)) } refSourceCommitSHAs := make(map[string]string) if len(repoRefs) > 0 { @@ -1337,6 +1338,7 @@ type GenerateManifestOpt func(*generateManifestOpt) type generateManifestOpt struct { cmpTarDoneCh chan<- bool cmpTarExcludedGlobs []string + cmpTarIncludedGlobs []string } func newGenerateManifestOpt(opts ...GenerateManifestOpt) *generateManifestOpt { @@ -1364,6 +1366,14 @@ func WithCMPTarExcludedGlobs(excludedGlobs []string) GenerateManifestOpt { } } +// WithCMPTarIncludedGlobs defines globs for files to include when streaming the tarball +// to a CMP sidecar. +func WithCMPTarIncludedGlobs(includedGlobs []string) GenerateManifestOpt { + return func(o *generateManifestOpt) { + o.cmpTarIncludedGlobs = includedGlobs + } +} + // GenerateManifests generates manifests from a path. Overrides are applied as a side effect on the given ApplicationSource. func GenerateManifests(ctx context.Context, appPath, repoRoot, revision string, q *apiclient.ManifestRequest, isLocal bool, gitCredsStore git.CredsStore, maxCombinedManifestQuantity resource.Quantity, gitRepoPaths io.TempPaths, opts ...GenerateManifestOpt) (*apiclient.ManifestResponse, error) { opt := newGenerateManifestOpt(opts...) @@ -1371,7 +1381,7 @@ func GenerateManifests(ctx context.Context, appPath, repoRoot, revision string, resourceTracking := argo.NewResourceTracking() - appSourceType, err := GetAppSourceType(ctx, q.ApplicationSource, appPath, repoRoot, q.AppName, q.EnabledSourceTypes, opt.cmpTarExcludedGlobs) + appSourceType, err := GetAppSourceType(ctx, q.ApplicationSource, appPath, repoRoot, q.AppName, q.EnabledSourceTypes, opt.cmpTarExcludedGlobs, opt.cmpTarIncludedGlobs) if err != nil { return nil, fmt.Errorf("error getting app source type: %w", err) } @@ -1397,7 +1407,7 @@ func GenerateManifests(ctx context.Context, appPath, repoRoot, revision string, pluginName = q.ApplicationSource.Plugin.Name } // if pluginName is provided it has to be `-` or just `` if plugin version is empty - targetObjs, err = runConfigManagementPluginSidecars(ctx, appPath, repoRoot, pluginName, env, q, opt.cmpTarDoneCh, opt.cmpTarExcludedGlobs) + targetObjs, err = runConfigManagementPluginSidecars(ctx, appPath, repoRoot, pluginName, env, q, opt.cmpTarDoneCh, opt.cmpTarExcludedGlobs, opt.cmpTarIncludedGlobs) if err != nil { err = fmt.Errorf("plugin sidecar failed. %s", err.Error()) } @@ -1534,7 +1544,7 @@ func mergeSourceParameters(source *v1alpha1.ApplicationSource, path, appName str } // GetAppSourceType returns explicit application source type or examines a directory and determines its application source type -func GetAppSourceType(ctx context.Context, source *v1alpha1.ApplicationSource, appPath, repoPath, appName string, enableGenerateManifests map[string]bool, tarExcludedGlobs []string) (v1alpha1.ApplicationSourceType, error) { +func GetAppSourceType(ctx context.Context, source *v1alpha1.ApplicationSource, appPath, repoPath, appName string, enableGenerateManifests map[string]bool, tarExcludedGlobs, tarIncludedGlobs []string) (v1alpha1.ApplicationSourceType, error) { err := mergeSourceParameters(source, appPath, appName) if err != nil { return "", fmt.Errorf("error while parsing source parameters: %v", err) @@ -1551,7 +1561,7 @@ func GetAppSourceType(ctx context.Context, source *v1alpha1.ApplicationSource, a } return *appSourceType, nil } - appType, err := discovery.AppType(ctx, appPath, repoPath, enableGenerateManifests, tarExcludedGlobs) + appType, err := discovery.AppType(ctx, appPath, repoPath, enableGenerateManifests, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return "", fmt.Errorf("error getting app source type: %v", err) } @@ -1898,7 +1908,7 @@ func getPluginParamEnvs(envVars []string, plugin *v1alpha1.ApplicationSourcePlug return env, nil } -func runConfigManagementPluginSidecars(ctx context.Context, appPath, repoPath, pluginName string, envVars *v1alpha1.Env, q *apiclient.ManifestRequest, tarDoneCh chan<- bool, tarExcludedGlobs []string) ([]*unstructured.Unstructured, error) { +func runConfigManagementPluginSidecars(ctx context.Context, appPath, repoPath, pluginName string, envVars *v1alpha1.Env, q *apiclient.ManifestRequest, tarDoneCh chan<- bool, tarExcludedGlobs, tarIncludedGlobs []string) ([]*unstructured.Unstructured, error) { // compute variables. env, err := getPluginEnvs(envVars, q) if err != nil { @@ -1906,14 +1916,14 @@ func runConfigManagementPluginSidecars(ctx context.Context, appPath, repoPath, p } // detect config management plugin server - conn, cmpClient, err := discovery.DetectConfigManagementPlugin(ctx, appPath, repoPath, pluginName, env, tarExcludedGlobs) + conn, cmpClient, err := discovery.DetectConfigManagementPlugin(ctx, appPath, repoPath, pluginName, env, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return nil, err } defer io.Close(conn) // generate manifests using commands provided in plugin config file in detected cmp-server sidecar - cmpManifests, err := generateManifestsCMP(ctx, appPath, repoPath, env, cmpClient, tarDoneCh, tarExcludedGlobs) + cmpManifests, err := generateManifestsCMP(ctx, appPath, repoPath, env, cmpClient, tarDoneCh, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return nil, fmt.Errorf("error generating manifests in cmp: %s", err) } @@ -1936,7 +1946,7 @@ func runConfigManagementPluginSidecars(ctx context.Context, appPath, repoPath, p // generateManifestsCMP will send the appPath files to the cmp-server over a gRPC stream. // The cmp-server will generate the manifests. Returns a response object with the generated // manifests. -func generateManifestsCMP(ctx context.Context, appPath, repoPath string, env []string, cmpClient pluginclient.ConfigManagementPluginServiceClient, tarDoneCh chan<- bool, tarExcludedGlobs []string) (*pluginclient.ManifestResponse, error) { +func generateManifestsCMP(ctx context.Context, appPath, repoPath string, env []string, cmpClient pluginclient.ConfigManagementPluginServiceClient, tarDoneCh chan<- bool, tarExcludedGlobs, tarIncludedGlobs []string) (*pluginclient.ManifestResponse, error) { generateManifestStream, err := cmpClient.GenerateManifest(ctx, grpc_retry.Disable()) if err != nil { return nil, fmt.Errorf("error getting generateManifestStream: %w", err) @@ -1945,7 +1955,7 @@ func generateManifestsCMP(ctx context.Context, appPath, repoPath string, env []s cmp.WithTarDoneChan(tarDoneCh), } - err = cmp.SendRepoStream(generateManifestStream.Context(), appPath, repoPath, generateManifestStream, env, tarExcludedGlobs, opts...) + err = cmp.SendRepoStream(generateManifestStream.Context(), appPath, repoPath, generateManifestStream, env, tarExcludedGlobs, tarIncludedGlobs, opts...) if err != nil { return nil, fmt.Errorf("error sending file to cmp-server: %s", err) } @@ -1963,7 +1973,7 @@ func (s *Service) GetAppDetails(ctx context.Context, q *apiclient.RepoServerAppD return err } - appSourceType, err := GetAppSourceType(ctx, q.Source, opContext.appPath, repoRoot, q.AppName, q.EnabledSourceTypes, s.initConstants.CMPTarExcludedGlobs) + appSourceType, err := GetAppSourceType(ctx, q.Source, opContext.appPath, repoRoot, q.AppName, q.EnabledSourceTypes, s.initConstants.CMPTarExcludedGlobs, s.initConstants.CMPTarIncludedGlobs) if err != nil { return err } @@ -1980,7 +1990,7 @@ func (s *Service) GetAppDetails(ctx context.Context, q *apiclient.RepoServerAppD return err } case v1alpha1.ApplicationSourceTypePlugin: - if err := populatePluginAppDetails(ctx, res, opContext.appPath, repoRoot, q, s.gitCredsStore, s.initConstants.CMPTarExcludedGlobs); err != nil { + if err := populatePluginAppDetails(ctx, res, opContext.appPath, repoRoot, q, s.gitCredsStore, s.initConstants.CMPTarExcludedGlobs, s.initConstants.CMPTarIncludedGlobs); err != nil { return fmt.Errorf("failed to populate plugin app details: %w", err) } } @@ -2139,7 +2149,7 @@ func populateKustomizeAppDetails(res *apiclient.RepoAppDetailsResponse, q *apicl return nil } -func populatePluginAppDetails(ctx context.Context, res *apiclient.RepoAppDetailsResponse, appPath string, repoPath string, q *apiclient.RepoServerAppDetailsQuery, store git.CredsStore, tarExcludedGlobs []string) error { +func populatePluginAppDetails(ctx context.Context, res *apiclient.RepoAppDetailsResponse, appPath string, repoPath string, q *apiclient.RepoServerAppDetailsQuery, store git.CredsStore, tarExcludedGlobs, tarIncludedGlobs []string) error { res.Plugin = &apiclient.PluginAppSpec{} envVars := []string{ @@ -2159,7 +2169,7 @@ func populatePluginAppDetails(ctx context.Context, res *apiclient.RepoAppDetails pluginName = q.Source.Plugin.Name } // detect config management plugin server (sidecar) - conn, cmpClient, err := discovery.DetectConfigManagementPlugin(ctx, appPath, repoPath, pluginName, env, tarExcludedGlobs) + conn, cmpClient, err := discovery.DetectConfigManagementPlugin(ctx, appPath, repoPath, pluginName, env, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return fmt.Errorf("failed to detect CMP for app: %w", err) } @@ -2170,7 +2180,7 @@ func populatePluginAppDetails(ctx context.Context, res *apiclient.RepoAppDetails return fmt.Errorf("error getting parametersAnnouncementStream: %w", err) } - err = cmp.SendRepoStream(parametersAnnouncementStream.Context(), appPath, repoPath, parametersAnnouncementStream, env, tarExcludedGlobs) + err = cmp.SendRepoStream(parametersAnnouncementStream.Context(), appPath, repoPath, parametersAnnouncementStream, env, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return fmt.Errorf("error sending file to cmp-server: %s", err) } diff --git a/reposerver/repository/repository_test.go b/reposerver/repository/repository_test.go index 3f2f74c4e5ae02..7c4d51eb170545 100644 --- a/reposerver/repository/repository_test.go +++ b/reposerver/repository/repository_test.go @@ -1448,15 +1448,15 @@ func TestGenerateNullList(t *testing.T) { } func TestIdentifyAppSourceTypeByAppDirWithKustomizations(t *testing.T) { - sourceType, err := GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/kustomization_yaml", "./testdata", "testapp", map[string]bool{}, []string{}) + sourceType, err := GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/kustomization_yaml", "./testdata", "testapp", map[string]bool{}, []string{}, []string{}) assert.Nil(t, err) assert.Equal(t, argoappv1.ApplicationSourceTypeKustomize, sourceType) - sourceType, err = GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/kustomization_yml", "./testdata", "testapp", map[string]bool{}, []string{}) + sourceType, err = GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/kustomization_yml", "./testdata", "testapp", map[string]bool{}, []string{}, []string{}) assert.Nil(t, err) assert.Equal(t, argoappv1.ApplicationSourceTypeKustomize, sourceType) - sourceType, err = GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/Kustomization", "./testdata", "testapp", map[string]bool{}, []string{}) + sourceType, err = GetAppSourceType(context.Background(), &argoappv1.ApplicationSource{}, "./testdata/Kustomization", "./testdata", "testapp", map[string]bool{}, []string{}, []string{}) assert.Nil(t, err) assert.Equal(t, argoappv1.ApplicationSourceTypeKustomize, sourceType) } diff --git a/util/app/discovery/discovery.go b/util/app/discovery/discovery.go index 21fbe5fd4bf367..60552c42e1c48f 100644 --- a/util/app/discovery/discovery.go +++ b/util/app/discovery/discovery.go @@ -31,11 +31,11 @@ func IsManifestGenerationEnabled(sourceType v1alpha1.ApplicationSourceType, enab return enabled } -func Discover(ctx context.Context, appPath, repoPath string, enableGenerateManifests map[string]bool, tarExcludedGlobs []string) (map[string]string, error) { +func Discover(ctx context.Context, appPath, repoPath string, enableGenerateManifests map[string]bool, tarExcludedGlobs, tarIncludedGlobs []string) (map[string]string, error) { apps := make(map[string]string) // Check if it is CMP - conn, _, err := DetectConfigManagementPlugin(ctx, appPath, repoPath, "", []string{}, tarExcludedGlobs) + conn, _, err := DetectConfigManagementPlugin(ctx, appPath, repoPath, "", []string{}, tarExcludedGlobs, tarIncludedGlobs) if err == nil { // Found CMP io.Close(conn) @@ -67,8 +67,8 @@ func Discover(ctx context.Context, appPath, repoPath string, enableGenerateManif return apps, err } -func AppType(ctx context.Context, appPath, repoPath string, enableGenerateManifests map[string]bool, tarExcludedGlobs []string) (string, error) { - apps, err := Discover(ctx, appPath, repoPath, enableGenerateManifests, tarExcludedGlobs) +func AppType(ctx context.Context, appPath, repoPath string, enableGenerateManifests map[string]bool, tarExcludedGlobs, tarIncludedGlobs []string) (string, error) { + apps, err := Discover(ctx, appPath, repoPath, enableGenerateManifests, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return "", err } @@ -85,7 +85,7 @@ func AppType(ctx context.Context, appPath, repoPath string, enableGenerateManife // check cmpSupports() // if supported return conn for the cmp-server -func DetectConfigManagementPlugin(ctx context.Context, appPath, repoPath, pluginName string, env []string, tarExcludedGlobs []string) (io.Closer, pluginclient.ConfigManagementPluginServiceClient, error) { +func DetectConfigManagementPlugin(ctx context.Context, appPath, repoPath, pluginName string, env []string, tarExcludedGlobs, tarIncludedGlobs []string) (io.Closer, pluginclient.ConfigManagementPluginServiceClient, error) { var conn io.Closer var cmpClient pluginclient.ConfigManagementPluginServiceClient var connFound bool @@ -98,7 +98,7 @@ func DetectConfigManagementPlugin(ctx context.Context, appPath, repoPath, plugin if pluginName != "" { // check if the given plugin supports the repo - conn, cmpClient, connFound = cmpSupports(ctx, pluginSockFilePath, appPath, repoPath, fmt.Sprintf("%v.sock", pluginName), env, tarExcludedGlobs, true) + conn, cmpClient, connFound = cmpSupports(ctx, pluginSockFilePath, appPath, repoPath, fmt.Sprintf("%v.sock", pluginName), env, tarExcludedGlobs, tarIncludedGlobs, true) if !connFound { return nil, nil, fmt.Errorf("couldn't find cmp-server plugin with name %q supporting the given repository", pluginName) } @@ -109,7 +109,7 @@ func DetectConfigManagementPlugin(ctx context.Context, appPath, repoPath, plugin } for _, file := range fileList { if file.Type() == os.ModeSocket { - conn, cmpClient, connFound = cmpSupports(ctx, pluginSockFilePath, appPath, repoPath, file.Name(), env, tarExcludedGlobs, false) + conn, cmpClient, connFound = cmpSupports(ctx, pluginSockFilePath, appPath, repoPath, file.Name(), env, tarExcludedGlobs, tarIncludedGlobs, false) if connFound { break } @@ -125,13 +125,13 @@ func DetectConfigManagementPlugin(ctx context.Context, appPath, repoPath, plugin // matchRepositoryCMP will send the repoPath to the cmp-server. The cmp-server will // inspect the files and return true if the repo is supported for manifest generation. // Will return false otherwise. -func matchRepositoryCMP(ctx context.Context, appPath, repoPath string, client pluginclient.ConfigManagementPluginServiceClient, env []string, tarExcludedGlobs []string) (bool, bool, error) { +func matchRepositoryCMP(ctx context.Context, appPath, repoPath string, client pluginclient.ConfigManagementPluginServiceClient, env []string, tarExcludedGlobs, tarIncludedGlobs []string) (bool, bool, error) { matchRepoStream, err := client.MatchRepository(ctx, grpc_retry.Disable()) if err != nil { return false, false, fmt.Errorf("error getting stream client: %s", err) } - err = cmp.SendRepoStream(ctx, appPath, repoPath, matchRepoStream, env, tarExcludedGlobs) + err = cmp.SendRepoStream(ctx, appPath, repoPath, matchRepoStream, env, tarExcludedGlobs, tarIncludedGlobs) if err != nil { return false, false, fmt.Errorf("error sending stream: %s", err) } @@ -142,7 +142,7 @@ func matchRepositoryCMP(ctx context.Context, appPath, repoPath string, client pl return resp.GetIsSupported(), resp.GetIsDiscoveryEnabled(), nil } -func cmpSupports(ctx context.Context, pluginSockFilePath, appPath, repoPath, fileName string, env []string, tarExcludedGlobs []string, namedPlugin bool) (io.Closer, pluginclient.ConfigManagementPluginServiceClient, bool) { +func cmpSupports(ctx context.Context, pluginSockFilePath, appPath, repoPath, fileName string, env []string, tarExcludedGlobs, tarIncludedGlobs []string, namedPlugin bool) (io.Closer, pluginclient.ConfigManagementPluginServiceClient, bool) { absPluginSockFilePath, err := filepath.Abs(pluginSockFilePath) if err != nil { log.Errorf("error getting absolute path for plugin socket dir %v, %v", pluginSockFilePath, err) @@ -165,7 +165,7 @@ func cmpSupports(ctx context.Context, pluginSockFilePath, appPath, repoPath, fil return nil, nil, false } - isSupported, isDiscoveryEnabled, err := matchRepositoryCMP(ctx, appPath, repoPath, cmpClient, env, tarExcludedGlobs) + isSupported, isDiscoveryEnabled, err := matchRepositoryCMP(ctx, appPath, repoPath, cmpClient, env, tarExcludedGlobs, tarIncludedGlobs) if err != nil { log.WithFields(log.Fields{ common.SecurityField: common.SecurityMedium, diff --git a/util/app/discovery/discovery_test.go b/util/app/discovery/discovery_test.go index 54eb30aff4fd18..771a1942eb4675 100644 --- a/util/app/discovery/discovery_test.go +++ b/util/app/discovery/discovery_test.go @@ -10,7 +10,7 @@ import ( ) func TestDiscover(t *testing.T) { - apps, err := Discover(context.Background(), "./testdata", "./testdata", map[string]bool{}, []string{}) + apps, err := Discover(context.Background(), "./testdata", "./testdata", map[string]bool{}, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, map[string]string{ "foo": "Kustomize", @@ -19,15 +19,15 @@ func TestDiscover(t *testing.T) { } func TestAppType(t *testing.T) { - appType, err := AppType(context.Background(), "./testdata/foo", "./testdata", map[string]bool{}, []string{}) + appType, err := AppType(context.Background(), "./testdata/foo", "./testdata", map[string]bool{}, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Kustomize", appType) - appType, err = AppType(context.Background(), "./testdata/baz", "./testdata", map[string]bool{}, []string{}) + appType, err = AppType(context.Background(), "./testdata/baz", "./testdata", map[string]bool{}, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Helm", appType) - appType, err = AppType(context.Background(), "./testdata", "./testdata", map[string]bool{}, []string{}) + appType, err = AppType(context.Background(), "./testdata", "./testdata", map[string]bool{}, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Directory", appType) } @@ -37,15 +37,15 @@ func TestAppType_Disabled(t *testing.T) { string(v1alpha1.ApplicationSourceTypeKustomize): false, string(v1alpha1.ApplicationSourceTypeHelm): false, } - appType, err := AppType(context.Background(), "./testdata/foo", "./testdata", enableManifestGeneration, []string{}) + appType, err := AppType(context.Background(), "./testdata/foo", "./testdata", enableManifestGeneration, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Directory", appType) - appType, err = AppType(context.Background(), "./testdata/baz", "./testdata", enableManifestGeneration, []string{}) + appType, err = AppType(context.Background(), "./testdata/baz", "./testdata", enableManifestGeneration, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Directory", appType) - appType, err = AppType(context.Background(), "./testdata", "./testdata", enableManifestGeneration, []string{}) + appType, err = AppType(context.Background(), "./testdata", "./testdata", enableManifestGeneration, []string{}, []string{}) assert.NoError(t, err) assert.Equal(t, "Directory", appType) } diff --git a/util/cmp/stream.go b/util/cmp/stream.go index 8a0d1a97283d05..cf5f1a3060621a 100644 --- a/util/cmp/stream.go +++ b/util/cmp/stream.go @@ -85,10 +85,10 @@ func WithTarDoneChan(ch chan<- bool) SenderOption { // SendRepoStream will compress the files under the given repoPath and send // them using the plugin stream sender. -func SendRepoStream(ctx context.Context, appPath, repoPath string, sender StreamSender, env []string, excludedGlobs []string, opts ...SenderOption) error { +func SendRepoStream(ctx context.Context, appPath, repoPath string, sender StreamSender, env []string, excludedGlobs, includedGlobs []string, opts ...SenderOption) error { opt := newSenderOption(opts...) - tgz, mr, err := GetCompressedRepoAndMetadata(repoPath, appPath, env, excludedGlobs, opt) + tgz, mr, err := GetCompressedRepoAndMetadata(repoPath, appPath, env, excludedGlobs, includedGlobs, opt) if err != nil { return err } @@ -106,9 +106,9 @@ func SendRepoStream(ctx context.Context, appPath, repoPath string, sender Stream return nil } -func GetCompressedRepoAndMetadata(repoPath string, appPath string, env []string, excludedGlobs []string, opt *senderOption) (*os.File, *pluginclient.AppStreamRequest, error) { +func GetCompressedRepoAndMetadata(repoPath string, appPath string, env []string, excludedGlobs, includedGlobs []string, opt *senderOption) (*os.File, *pluginclient.AppStreamRequest, error) { // compress all files in repoPath in tgz - tgz, filesWritten, checksum, err := tgzstream.CompressFiles(repoPath, nil, excludedGlobs) + tgz, filesWritten, checksum, err := tgzstream.CompressFiles(repoPath, includedGlobs, excludedGlobs) if err != nil { return nil, nil, fmt.Errorf("error compressing repo files: %w", err) } diff --git a/util/cmp/stream_test.go b/util/cmp/stream_test.go index 93a76cf10e673d..ff350a97bd1088 100644 --- a/util/cmp/stream_test.go +++ b/util/cmp/stream_test.go @@ -59,7 +59,7 @@ func TestReceiveApplicationStream(t *testing.T) { close(streamMock.messages) os.RemoveAll(workdir) }() - go streamMock.sendFile(context.Background(), t, appDir, streamMock, []string{"env1", "env2"}, []string{"DUMMY.md", "dum*"}) + go streamMock.sendFile(context.Background(), t, appDir, streamMock, []string{"env1", "env2"}, []string{"DUMMY.md", "dum*"}, []string{}) // when env, err := cmp.ReceiveRepoStream(context.Background(), streamMock, workdir, false) @@ -82,12 +82,12 @@ func TestReceiveApplicationStream(t *testing.T) { }) } -func (m *streamMock) sendFile(ctx context.Context, t *testing.T, basedir string, sender cmp.StreamSender, env []string, excludedGlobs []string) { +func (m *streamMock) sendFile(ctx context.Context, t *testing.T, basedir string, sender cmp.StreamSender, env []string, excludedGlobs, includedBlobs []string) { t.Helper() defer func() { m.done <- true }() - err := cmp.SendRepoStream(ctx, basedir, basedir, sender, env, excludedGlobs) + err := cmp.SendRepoStream(ctx, basedir, basedir, sender, env, excludedGlobs, includedBlobs) require.NoError(t, err) } diff --git a/util/io/files/tar.go b/util/io/files/tar.go index 13973f732fe720..2b95dbd596e6bf 100644 --- a/util/io/files/tar.go +++ b/util/io/files/tar.go @@ -9,6 +9,7 @@ import ( "os" "path/filepath" + "github.com/bmatcuk/doublestar" log "github.com/sirupsen/logrus" ) @@ -158,10 +159,10 @@ func (t *tgz) tgzFile(path string, fi os.FileInfo, err error) error { return fmt.Errorf("relative path error: %s", err) } - if t.inclusions != nil && base != "." && !fi.IsDir() { + if t.inclusions != nil && len(t.inclusions) > 0 && base != "." && !fi.IsDir() { included := false for _, inclusionPattern := range t.inclusions { - found, err := filepath.Match(inclusionPattern, base) + found, err := doublestar.Match(inclusionPattern, relativePath) if err != nil { return fmt.Errorf("error verifying inclusion pattern %q: %w", inclusionPattern, err) } @@ -176,7 +177,7 @@ func (t *tgz) tgzFile(path string, fi os.FileInfo, err error) error { } if t.exclusions != nil { for _, exclusionPattern := range t.exclusions { - found, err := filepath.Match(exclusionPattern, relativePath) + found, err := doublestar.Match(exclusionPattern, relativePath) if err != nil { return fmt.Errorf("error verifying exclusion pattern %q: %w", exclusionPattern, err) } diff --git a/util/io/files/tar_test.go b/util/io/files/tar_test.go index 1817fa5ce9353e..6278cbfe088b95 100644 --- a/util/io/files/tar_test.go +++ b/util/io/files/tar_test.go @@ -51,12 +51,12 @@ func TestTgz(t *testing.T) { filesWritten, err := files.Tgz(getTestAppDir(t), nil, exclusions, f.file) // then - assert.Equal(t, 3, filesWritten) + assert.Equal(t, 6, filesWritten) assert.NoError(t, err) prepareRead(f) files, err := read(f.file) require.NoError(t, err) - assert.Equal(t, 8, len(files)) + assert.Equal(t, 16, len(files)) assert.Contains(t, files, "README.md") assert.Contains(t, files, "applicationset/latest/kustomization.yaml") assert.Contains(t, files, "applicationset/stable/kustomization.yaml") @@ -74,12 +74,12 @@ func TestTgz(t *testing.T) { filesWritten, err := files.Tgz(getTestAppDir(t), nil, exclusions, f.file) // then - assert.Equal(t, 2, filesWritten) + assert.Equal(t, 5, filesWritten) assert.NoError(t, err) prepareRead(f) files, err := read(f.file) require.NoError(t, err) - assert.Equal(t, 7, len(files)) + assert.Equal(t, 15, len(files)) assert.Contains(t, files, "applicationset/latest/kustomization.yaml") assert.Contains(t, files, "applicationset/stable/kustomization.yaml") }) @@ -94,14 +94,47 @@ func TestTgz(t *testing.T) { filesWritten, err := files.Tgz(getTestAppDir(t), nil, exclusions, f.file) // then - assert.Equal(t, 1, filesWritten) + assert.Equal(t, 4, filesWritten) assert.NoError(t, err) prepareRead(f) files, err := read(f.file) require.NoError(t, err) - assert.Equal(t, 5, len(files)) + assert.Equal(t, 13, len(files)) assert.Contains(t, files, "applicationset/stable/kustomization.yaml") }) + + t.Run("will include and exclude files from both inclusion and exclusion lists", func(t *testing.T) { + // given + t.Parallel() + inclusions := []string{ + "**/*.y*ml", + "**/deploy/**", + } + exclusions := []string{ + "git/**", + } + f := setup(t) + defer teardown(f) + + // when + filesWritten, err := files.Tgz(getTestAppDir(t), inclusions, exclusions, f.file) + + // then + assert.Equal(t, 4, filesWritten) + assert.NoError(t, err) + prepareRead(f) + + files, err := read(f.file) + require.NoError(t, err) + + assert.Equal(t, 13, len(files)) + assert.Contains(t, files, "applicationset/stable/kustomization.yaml") + assert.Contains(t, files, "applicationset/stable/kustomization.yaml") + assert.NotContains(t, files, "git/index") + assert.Contains(t, files, "src/domain/service/deploy/template.tpl") + assert.Contains(t, files, "src/domain/service/deploy/helmfile.yaml") + assert.NotContains(t, files, "README.md") + }) } func TestUntgz(t *testing.T) { @@ -174,7 +207,7 @@ func TestUntgz(t *testing.T) { // then require.NoError(t, err) names := readFiles(t, destDir) - assert.Equal(t, 8, len(names)) + assert.Equal(t, 16, len(names)) assert.Contains(t, names, "README.md") assert.Contains(t, names, "applicationset/latest/kustomization.yaml") assert.Contains(t, names, "applicationset/stable/kustomization.yaml") diff --git a/util/io/files/testdata/app/git/index b/util/io/files/testdata/app/git/index new file mode 100644 index 00000000000000..e69de29bb2d1d6 diff --git a/util/io/files/testdata/app/src/domain/service/deploy/helmfile.yaml b/util/io/files/testdata/app/src/domain/service/deploy/helmfile.yaml new file mode 100644 index 00000000000000..e69de29bb2d1d6 diff --git a/util/io/files/testdata/app/src/domain/service/deploy/template.tpl b/util/io/files/testdata/app/src/domain/service/deploy/template.tpl new file mode 100644 index 00000000000000..e69de29bb2d1d6