Documentation
¶
Overview ¶
Package layout provides utilities for generating cluster directory layouts and for writing Kubernetes and Flux manifests to disk.
Flux Kustomizations and ArgoCD Applications reference directories in a Git repository using different fields. Flux uses `spec.path`, which must start with `./` and is always interpreted relative to the repository root. ArgoCD uses `spec.source.path` without the `./` prefix but with the same relative semantics.
When nodes or bundles live in nested subfolders, the path must point directly to the folder containing the manifests unless the directory tree only contains files for a single node or bundle. Flux will recursively auto-generate a `kustomization.yaml` when one is missing and include every manifest under the specified path. ArgoCD does not auto-generate a `kustomization.yaml` and therefore ignores nested directories unless they are referenced from a `kustomization.yaml` at the target path.
For example, consider the layout:
repo/
clusters/
prod/
nodes/
cp/
kustomization.yaml
bundles/
monitoring/
kustomization.yaml
The Flux Kustomization for the control-plane node uses:
spec.path: ./clusters/prod/nodes/cp
The equivalent ArgoCD Application uses:
spec.source.path: clusters/prod/nodes/cp
With this layout, each node or bundle is targeted individually. Pointing a Flux Kustomization at `./clusters/prod` would combine the `cp` and `monitoring` manifests into a single deployment because it would auto-generate a `kustomization.yaml` for the entire tree. ArgoCD will only process the manifests under `clusters/prod` itself unless a `kustomization.yaml` aggregates the subdirectories, so each subfolder must be referenced separately.
Package api defines configuration structures used to generate Kubernetes manifests and Flux resources.
Index ¶
- func ApplyFlattenPathRewrites(root *ManifestLayout)
- func DefaultKustomizationFileName(name string) string
- func DefaultManifestFileName(namespace, kind, name string, mode FileExportMode) string
- func KindNameManifestFileName(_, kind, name string, mode FileExportMode) string
- func WalkClusterByPackage(c *stack.Cluster, rules LayoutRules) (map[string]*ManifestLayout, error)
- func WriteManifest(basePath string, cfg Config, ml *ManifestLayout) error
- func WritePackagesToDisk(packages map[string]*ManifestLayout, basePath string) error
- type ApplicationFileMode
- type Config
- type ConfigMapGeneratorSpec
- type ExtraFile
- type FileExportMode
- type FileNamingMode
- type FluxPlacement
- type GroupingMode
- type KustomizationFileNameFunc
- type KustomizationMode
- type LayoutAugmenter
- type LayoutPreset
- type LayoutRules
- type ManifestFileNameFunc
- type ManifestLayout
- func (ml *ManifestLayout) FlattenInfoNodeAlias(nodePath string) *ManifestLayout
- func (ml *ManifestLayout) FlattenInfoPathRewrites() map[string]string
- func (ml *ManifestLayout) FullRepoPath() string
- func (ml *ManifestLayout) FullRepoPathWithPackage() string
- func (ml *ManifestLayout) WriteToDisk(basePath string) error
- func (ml *ManifestLayout) WriteToTar(w io.Writer) error
- type Profile
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func ApplyFlattenPathRewrites ¶
func ApplyFlattenPathRewrites(root *ManifestLayout)
ApplyFlattenPathRewrites walks the layout tree, gathers all path rewrites recorded by flattenSingleTier collapses, and rewrites Spec.Path on every Flux Kustomization CR found in the tree's Resources. Idempotent: if no rewrites were recorded, returns immediately; on repeated invocations already-rewritten paths no longer match the rewrite keys, so subsequent passes are no-ops.
flattenInfo is intentionally left in place after a rewrite pass so that IntegrateWithLayout can be called multiple times on the same flattened layout — the integrator's findLayoutNode fallback depends on the nodeAliases remaining populated for the lifetime of the layout.
Called by the Flux integrator's IntegrateWithLayout before returning. Lives in the layout package because the flattenInfo field is unexported here.
func DefaultKustomizationFileName ¶
DefaultKustomizationFileName returns the standard Flux Kustomization file name.
func DefaultManifestFileName ¶
func DefaultManifestFileName(namespace, kind, name string, mode FileExportMode) string
DefaultManifestFileName implements the standard file naming convention used by Kure. It writes either one file per resource or groups by kind depending on the FileExportMode.
func KindNameManifestFileName ¶
func KindNameManifestFileName(_, kind, name string, mode FileExportMode) string
KindNameManifestFileName returns file names using {kind}-{name}.yaml format, without the namespace prefix. This is the default for Pattern A (CentralizedControlPlane) where per-app artifact directories make the namespace prefix redundant.
func WalkClusterByPackage ¶
func WalkClusterByPackage(c *stack.Cluster, rules LayoutRules) (map[string]*ManifestLayout, error)
WalkClusterByPackage traverses a stack.Cluster and builds separate ManifestLayout trees for each unique PackageRef (OCI artifact). Returns a map where keys are PackageRef GVKs and values are the corresponding ManifestLayout trees. Nodes without PackageRef inherit from their parent, with nil representing the default package.
func WriteManifest ¶
func WriteManifest(basePath string, cfg Config, ml *ManifestLayout) error
WriteManifest writes a ManifestLayout to disk using the provided configuration.
func WritePackagesToDisk ¶
func WritePackagesToDisk(packages map[string]*ManifestLayout, basePath string) error
WritePackagesToDisk writes multiple package layouts to separate directory structures
Types ¶
type ApplicationFileMode ¶
type ApplicationFileMode string
ApplicationFileMode specifies how resources within an application are written.
The default is AppFilePerResource which mirrors the behaviour of FilePerResource and writes each generated resource to its own file. AppFileSingle groups all resources belonging to an application into a single manifest file.
const ( // AppFilePerResource writes each application resource to its own file. AppFilePerResource ApplicationFileMode = "resource" // AppFileSingle writes all resources for an application into one file. AppFileSingle ApplicationFileMode = "single" // AppFileUnset indicates that no application file mode was specified. AppFileUnset ApplicationFileMode = "" )
type Config ¶
type Config struct {
// ManifestsDir is the directory under which Kubernetes manifests are written.
ManifestsDir string
// FluxDir is the directory under which Flux manifests are written.
FluxDir string
// FilePer determines how resources are grouped into files when writing manifests.
FilePer FileExportMode
// ApplicationFileMode controls whether application resources are written
// to a single file or split per resource. Defaults to AppFilePerResource.
ApplicationFileMode ApplicationFileMode
// KustomizationMode controls how kustomization.yaml files are generated.
// Defaults to KustomizationExplicit.
KustomizationMode KustomizationMode
// FluxKustomizationMode overrides KustomizationMode based on FluxPlacement.
// When a ManifestLayout's FluxPlacement matches a key in this map, the
// corresponding KustomizationMode is used instead of the global
// KustomizationMode. This allows different kustomization.yaml reference
// styles per flux placement strategy.
FluxKustomizationMode map[FluxPlacement]KustomizationMode
// FileNaming controls the file naming pattern. When set, it determines
// the ManifestFileNameFunc to use. If ManifestFileName is also set, it
// takes precedence over FileNaming.
FileNaming FileNamingMode
// ManifestFileName formats the file name for a resource manifest.
// Takes precedence over FileNaming when set.
ManifestFileName ManifestFileNameFunc
// KustomizationFileName formats the file name for a Flux Kustomization.
KustomizationFileName KustomizationFileNameFunc
}
Config LayoutConfig defines rules for generating a cluster layout.
func ConfigForPreset ¶
func ConfigForPreset(p LayoutPreset) (Config, error)
ConfigForPreset returns a Config configured for the given preset. Unknown presets return an error.
func DefaultConfigForProfile ¶
DefaultConfigForProfile returns a Config initialised with defaults for the given profile. Unknown profiles fall back to FluxProfile.
func DefaultLayoutConfig ¶
func DefaultLayoutConfig() Config
DefaultLayoutConfig returns a configuration that matches the directory layout expected by FluxCD when writing manifests and Kustomizations.
func (Config) ResolveKustomizationMode ¶
func (c Config) ResolveKustomizationMode(fp FluxPlacement) KustomizationMode
ResolveKustomizationMode returns the effective KustomizationMode for the given FluxPlacement. If FluxKustomizationMode contains an override for the placement, that value is used. Otherwise, the global KustomizationMode (or KustomizationExplicit when unset) is returned.
func (Config) ResolveManifestFileName ¶
func (c Config) ResolveManifestFileName() ManifestFileNameFunc
ResolveManifestFileName returns the effective ManifestFileNameFunc for this Config. If ManifestFileName is set it is returned directly. Otherwise, FileNaming is used to select the function. If neither is set, DefaultManifestFileName is returned.
type ConfigMapGeneratorSpec ¶
ConfigMapGeneratorSpec describes a single kustomize configMapGenerator entry. Files are paths (relative to the layout directory) of files included in the generated ConfigMap.
type ExtraFile ¶
ExtraFile is an arbitrary file written into a ManifestLayout's directory alongside the resource YAMLs.
type FileExportMode ¶
type FileExportMode string
FileExportMode determines how resources are written to disk.
const ( // FilePerResource writes each resource to its own file. FilePerResource FileExportMode = "resource" // FilePerKind groups resources by kind into a single file. FilePerKind FileExportMode = "kind" // FilePerUnset indicates that no export mode is specified. FilePerUnset FileExportMode = "" )
type FileNamingMode ¶
type FileNamingMode string
FileNamingMode controls the file naming pattern for manifest files.
const ( // FileNamingDefault uses the standard {namespace}-{kind}-{name}.yaml format. FileNamingDefault FileNamingMode = "default" // FileNamingKindName uses the {kind}-{name}.yaml format, omitting the namespace prefix. FileNamingKindName FileNamingMode = "kind-name" // FileNamingUnset indicates no file naming preference. FileNamingUnset FileNamingMode = "" )
type FluxPlacement ¶
type FluxPlacement string
FluxPlacement determines how Flux Kustomizations are placed in the layout.
const ( // FluxSeparate places all Flux Kustomizations in a separate directory. FluxSeparate FluxPlacement = "separate" // FluxIntegratedPerLayout places a Flux Kustomization CR inline for every // layout node, including augmenter-added child layouts. Children are // referenced from the parent kustomization.yaml as kustomization-<child>.yaml // CR files. Finest granularity; use when each child should be reconciled by // its own Flux Kustomization (e.g. hook-group dependsOn). FluxIntegratedPerLayout FluxPlacement = "integrated" // FluxIntegratedPerBundle places Flux Kustomization CRs inline at bundle/node // boundaries only; a bundle's interior (including augmenter-added child // layouts) is a single kustomize build, with children referenced as // directories. Coarser than PerLayout: Flux reconciles per bundle, kustomize // handles the interior. Use when the unit of Flux reconciliation is the // bundle, not each layout node. FluxIntegratedPerBundle FluxPlacement = "integrated-per-bundle" // FluxUnset indicates no flux placement preference. FluxUnset FluxPlacement = "" )
type GroupingMode ¶
type GroupingMode string
GroupingMode controls how nodes, bundles and applications are laid out on disk.
The default for all grouping modes is GroupByName which creates a directory per entity. GroupFlat places all entities in the same directory.
const ( // GroupByName creates a directory for each item in the hierarchy. GroupByName GroupingMode = "name" // GroupFlat flattens the hierarchy placing all items in the same directory. GroupFlat GroupingMode = "flat" // GroupUnset indicates that no grouping preference was specified. GroupUnset GroupingMode = "" )
type KustomizationFileNameFunc ¶
KustomizationFileNameFunc returns the file name for a Flux Kustomization manifest.
type KustomizationMode ¶
type KustomizationMode string
KustomizationMode determines how kustomization.yaml files reference manifests.
const ( // KustomizationExplicit lists each manifest file in kustomization.yaml. KustomizationExplicit KustomizationMode = "explicit" // KustomizationRecursive references only subdirectories in kustomization.yaml. KustomizationRecursive KustomizationMode = "recursive" // KustomizationUnset indicates no kustomization mode preference. KustomizationUnset KustomizationMode = "" )
type LayoutAugmenter ¶
type LayoutAugmenter interface {
AugmentLayout(layout *ManifestLayout) error
}
LayoutAugmenter is an optional interface that ApplicationConfig implementations can implement to attach extra files or configMapGenerator entries to their per-app ManifestLayout after resource generation. The walker invokes AugmentLayout when app.Config satisfies this interface.
The interface lives in the layout package (rather than pkg/stack alongside Validator) because ApplicationConfig — defined in pkg/stack — cannot reference *ManifestLayout without creating an import cycle: the layout package already imports pkg/stack.
type LayoutPreset ¶
type LayoutPreset string
LayoutPreset identifies a named layout pattern that configures all layout dimensions into a known-valid combination. Presets are based on the layout patterns defined in the Wharf FluxCD layout research.
const ( // PresetCentralizedControlPlane implements Pattern A: all Flux KS CRs in // dedicated aggregator directories, completely separate from payload. Best // suited for fleet management with many applications. Uses flat directory // grouping and {kind}-{name}.yaml file naming. PresetCentralizedControlPlane LayoutPreset = "CentralizedControlPlane" // PresetSiblingControlPlane implements Pattern B: Flux KS CRs in a // flux-system/ sibling directory within each artifact. Designed for // single-artifact or per-app artifact scenarios. PresetSiblingControlPlane LayoutPreset = "SiblingControlPlane" // PresetParentDeployedControl implements Pattern C: KS CRs live in the // payload of their parent Kustomization. Best suited for simple, // single-app deployments. PresetParentDeployedControl LayoutPreset = "ParentDeployedControl" )
type LayoutRules ¶
type LayoutRules struct {
// NodeGrouping controls how nodes are written to disk. Defaults to
// GroupByName.
NodeGrouping GroupingMode
// BundleGrouping controls how bundles are written to disk. Defaults to
// GroupByName.
BundleGrouping GroupingMode
// ApplicationGrouping controls how applications are written to disk.
// Defaults to GroupByName.
ApplicationGrouping GroupingMode
// ApplicationFileMode controls whether application resources are
// combined into a single file or split per resource. Defaults to
// AppFilePerResource.
ApplicationFileMode ApplicationFileMode
// FilePer sets the default file export mode for resources. Defaults to
// FilePerResource.
FilePer FileExportMode
// ClusterName specifies a cluster name to prepend to all paths.
// When set, creates clusters/{ClusterName}/... structure.
ClusterName string
// FluxPlacement determines how Flux Kustomizations are placed.
// Defaults to FluxSeparate.
FluxPlacement FluxPlacement
// FileNaming controls the file naming pattern for manifest files.
// Defaults to FileNamingDefault ({namespace}-{kind}-{name}.yaml).
FileNaming FileNamingMode
// FlattenSingleTier collapses a vestigial intermediate directory layer
// produced by the walker when it adds no semantic value: a parent layout
// with exactly one named child whose own children are empty and which is
// not an UmbrellaChild, where the parent itself is a top-level layout
// (Namespace has no path separator) with no own Resources.
//
// Typical case: flat single-bundle apps where the caller wraps the bundle
// in an extra Node (e.g. crane's "apps" Node). Multi-tier apps with sub-
// Kustomizations are unaffected — the collapse rules require the
// intermediate to be terminal.
//
// Only effective for WalkCluster (not WalkClusterByPackage, which uses
// synthetic unnamed wrappers to express package boundaries).
//
// When the layout participates in Flux integration, the flatten helper
// records pathRewrites/nodeAliases on the absorbing layout.
// IntegrateWithLayout consults aliases via findLayoutNode and calls
// ApplyFlattenPathRewrites before returning, so generated Flux
// Kustomization CRs resolve to the post-collapse directory.
FlattenSingleTier bool
}
LayoutRules control how layouts are generated.
Zero values are interpreted as the defaults described in the field documentation.
func DefaultLayoutRules ¶
func DefaultLayoutRules() LayoutRules
DefaultLayoutRules returns a LayoutRules instance populated with the documented default values.
func LayoutRulesForPreset ¶
func LayoutRulesForPreset(p LayoutPreset) (LayoutRules, error)
LayoutRulesForPreset returns LayoutRules configured for the given preset. Unknown presets return an error.
func (LayoutRules) Validate ¶
func (lr LayoutRules) Validate() error
Validate ensures the LayoutRules contain known option values.
type ManifestFileNameFunc ¶
type ManifestFileNameFunc func(namespace, kind, name string, mode FileExportMode) string
ManifestFileNameFunc returns a file name for the given namespace, kind and resource name.
type ManifestLayout ¶
type ManifestLayout struct {
Name string
Namespace string
PackageRef *schema.GroupVersionKind
FilePer FileExportMode
ApplicationFileMode ApplicationFileMode
Mode KustomizationMode
FluxPlacement FluxPlacement // Track flux placement mode for kustomization generation
FileNaming FileNamingMode // Controls resource file naming pattern
Resources []client.Object
Children []*ManifestLayout
// ExtraFiles are arbitrary files written alongside resource YAMLs in this
// layout's directory. Typical use: a values.yaml referenced by a
// configMapGenerator entry. Augmenters (LayoutAugmenter) attach these.
ExtraFiles []ExtraFile
// ConfigMapGenerators emit a kustomize configMapGenerator: section in
// kustomization.yaml. kustomize appends a content-hash suffix to each
// generated ConfigMap name; resources referencing it (e.g.
// HelmRelease.spec.valuesFrom) are rewritten to the suffixed name on
// build, so any change to the source file forces re-reconciliation.
ConfigMapGenerators []ConfigMapGeneratorSpec
// UmbrellaChild marks this layout as rendered from a Bundle.Children
// entry. When true, kustomization.yaml writers emit a
// flux-system-kustomization-{Name}.yaml reference in the parent directory
// (regardless of FluxPlacement), and the layout integrator places the
// child's Flux Kustomization CR at the parent layout node rather than in
// the child's own directory.
UmbrellaChild bool
// DependsOn lists sibling layout names whose Kustomization CRs must reconcile
// before this layout's CR. In FluxIntegratedPerLayout mode the layout integrator
// translates these into spec.dependsOn on the emitted Kustomization CR.
// Augmenters (LayoutAugmenter) set this field; the integrator reads it.
DependsOn []string
// contains filtered or unexported fields
}
func FindByNodeAlias ¶
func FindByNodeAlias(ml *ManifestLayout, nodePath string) *ManifestLayout
FindByNodeAlias walks the layout tree looking for a flattenInfo nodeAlias matching nodePath. Returns the absorbing layout, or nil if no alias matches. Used by the Flux integrator's findLayoutNode as a fallback when regular path-based search fails.
func WalkCluster ¶
func WalkCluster(c *stack.Cluster, rules LayoutRules) (*ManifestLayout, error)
WalkCluster traverses a stack.Cluster and builds a ManifestLayout tree that mirrors the node and bundle hierarchy. Behaviour is controlled via LayoutRules. When BundleGrouping and ApplicationGrouping are set to GroupFlat, all application resources are written directly to their parent node's directory.
func (*ManifestLayout) FlattenInfoNodeAlias ¶
func (ml *ManifestLayout) FlattenInfoNodeAlias(nodePath string) *ManifestLayout
FlattenInfoNodeAlias returns the absorbing layout for the given node path recorded on this layout's flattenInfo, or nil if no alias matches. Exposed for the Flux integrator's findLayoutNode fallback.
func (*ManifestLayout) FlattenInfoPathRewrites ¶
func (ml *ManifestLayout) FlattenInfoPathRewrites() map[string]string
FlattenInfoPathRewrites returns the path-rewrite map recorded on this layout's flattenInfo, or nil. Exposed for ApplyFlattenPathRewrites.
func (*ManifestLayout) FullRepoPath ¶
func (ml *ManifestLayout) FullRepoPath() string
func (*ManifestLayout) FullRepoPathWithPackage ¶
func (ml *ManifestLayout) FullRepoPathWithPackage() string
FullRepoPathWithPackage returns the repository path including package-specific prefix
func (*ManifestLayout) WriteToDisk ¶
func (ml *ManifestLayout) WriteToDisk(basePath string) error
func (*ManifestLayout) WriteToTar ¶
func (ml *ManifestLayout) WriteToTar(w io.Writer) error
WriteToTar writes the ManifestLayout to a tar archive, mirroring the directory structure that WriteToDisk would produce. File paths use forward slashes and output is deterministic (sorted file names).