mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-05-17 18:25:26 +00:00
277 lines
9.0 KiB
Go
277 lines
9.0 KiB
Go
// Copyright 2022 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package localizer
|
|
|
|
import (
|
|
"log"
|
|
"path/filepath"
|
|
|
|
"sigs.k8s.io/kustomize/api/ifc"
|
|
pLdr "sigs.k8s.io/kustomize/api/internal/plugins/loader"
|
|
"sigs.k8s.io/kustomize/api/internal/target"
|
|
"sigs.k8s.io/kustomize/api/konfig"
|
|
"sigs.k8s.io/kustomize/api/loader"
|
|
"sigs.k8s.io/kustomize/api/provider"
|
|
"sigs.k8s.io/kustomize/api/resmap"
|
|
"sigs.k8s.io/kustomize/api/types"
|
|
"sigs.k8s.io/kustomize/kyaml/errors"
|
|
"sigs.k8s.io/kustomize/kyaml/filesys"
|
|
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
"sigs.k8s.io/yaml"
|
|
)
|
|
|
|
// localizer encapsulates all state needed to localize the root at ldr.
|
|
type localizer struct {
|
|
fSys filesys.FileSystem
|
|
|
|
// kusttarget fields
|
|
validator ifc.Validator
|
|
rFactory *resmap.Factory
|
|
pLdr *pLdr.Loader
|
|
|
|
// underlying type is Loader
|
|
ldr ifc.Loader
|
|
|
|
// root is at ldr.Root()
|
|
root filesys.ConfirmedDir
|
|
|
|
// destination directory in newDir that mirrors root
|
|
dst string
|
|
}
|
|
|
|
// Run attempts to localize the kustomization root at target with the given localize arguments
|
|
func Run(target string, scope string, newDir string, fSys filesys.FileSystem) error {
|
|
ldr, args, err := NewLoader(target, scope, newDir, fSys)
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
defer func() { _ = ldr.Cleanup() }()
|
|
|
|
toDst, err := filepath.Rel(args.Scope.String(), args.Target.String())
|
|
if err != nil {
|
|
log.Panicf("cannot find path from %q to child directory %q: %s", args.Scope, args.Target, err)
|
|
}
|
|
dst := args.NewDir.Join(toDst)
|
|
if err = fSys.MkdirAll(dst); err != nil {
|
|
return errors.WrapPrefixf(err, "unable to create directory in localize destination")
|
|
}
|
|
|
|
depProvider := provider.NewDepProvider()
|
|
rFactory := resmap.NewFactory(depProvider.GetResourceFactory())
|
|
// As of alpha, only built-in plugins, using kustomize's built-in definitions of them,
|
|
// are potentially localized.
|
|
plgnsLdr := pLdr.NewLoader(types.DisabledPluginConfig(), rFactory, filesys.MakeFsOnDisk())
|
|
err = (&localizer{
|
|
fSys: fSys,
|
|
validator: depProvider.GetFieldValidator(),
|
|
rFactory: rFactory,
|
|
pLdr: plgnsLdr,
|
|
ldr: ldr,
|
|
root: args.Target,
|
|
dst: dst,
|
|
}).localize()
|
|
if err != nil {
|
|
errCleanup := fSys.RemoveAll(args.NewDir.String())
|
|
if errCleanup != nil {
|
|
log.Printf("unable to clean localize destination: %s", errCleanup)
|
|
}
|
|
return errors.WrapPrefixf(err, "unable to localize target %q", target)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// localize localizes the root that lc is at
|
|
func (lc *localizer) localize() error {
|
|
kt := target.NewKustTarget(lc.ldr, lc.validator, lc.rFactory, lc.pLdr)
|
|
err := kt.Load()
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
|
|
kustomization := kt.Kustomization()
|
|
err = lc.localizeNativeFields(&kustomization)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = lc.localizeBuiltinPlugins(&kustomization)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
content, err := yaml.Marshal(&kustomization)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, "unable to serialize localized kustomization file")
|
|
}
|
|
if err = lc.fSys.WriteFile(filepath.Join(lc.dst, konfig.DefaultKustomizationFileName()), content); err != nil {
|
|
return errors.WrapPrefixf(err, "unable to write localized kustomization file")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// localizeNativeFields localizes paths on kustomize-native fields, like configMapGenerator, that kustomize has a
|
|
// built-in understanding of. This excludes helm-related fields, such as `helmGlobals` and `helmCharts`.
|
|
func (lc *localizer) localizeNativeFields(kust *types.Kustomization) error {
|
|
for i, path := range kust.Components {
|
|
newPath, err := lc.localizeDir(path)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, "unable to localize components field")
|
|
}
|
|
kust.Components[i] = newPath
|
|
}
|
|
for i := range kust.Patches {
|
|
if kust.Patches[i].Path != "" {
|
|
newPath, err := lc.localizeFile(kust.Patches[i].Path)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, "unable to localize patches path %q", kust.Patches[i].Path)
|
|
}
|
|
kust.Patches[i].Path = newPath
|
|
}
|
|
}
|
|
// TODO(annasong): localize all other kustomization fields: resources, bases, crds, configurations,
|
|
// openapi, patchesJson6902, patchesStrategicMerge, replacements, configMapGenerators, secretGenerators
|
|
return nil
|
|
}
|
|
|
|
// localizeFile localizes file path and returns the localized path
|
|
func (lc *localizer) localizeFile(path string) (string, error) {
|
|
content, err := lc.ldr.Load(path)
|
|
if err != nil {
|
|
return "", errors.Wrap(err)
|
|
}
|
|
|
|
var locPath string
|
|
if loader.IsRemoteFile(path) {
|
|
// TODO(annasong): You need to check if you can add a localize directory here to store
|
|
// the remote file content. There may be a directory that shares the localize directory name.
|
|
locPath = locFilePath(path)
|
|
} else {
|
|
// ldr has checked that path must be relative; this is subject to change in beta.
|
|
|
|
// We must clean path to:
|
|
// 1. avoid symlinks. A `kustomize build` run will fail if we write files to
|
|
// symlink paths outside the current root, given that we don't want to recreate
|
|
// the symlinks. Even worse, we could be writing files outside the localize destination.
|
|
// 2. avoid paths that temporarily traverse outside the current root,
|
|
// i.e. ../../../scope/target/current-root. The localized file will be surrounded by
|
|
// different directories than its source, and so an uncleaned path may no longer be valid.
|
|
locPath = cleanFilePath(lc.fSys, lc.root, path)
|
|
}
|
|
absPath := filepath.Join(lc.dst, locPath)
|
|
if err = lc.fSys.MkdirAll(filepath.Dir(absPath)); err != nil {
|
|
return "", errors.WrapPrefixf(err, "unable to create directories to localize file %q", path)
|
|
}
|
|
if err = lc.fSys.WriteFile(absPath, content); err != nil {
|
|
return "", errors.WrapPrefixf(err, "unable to localize file %q", path)
|
|
}
|
|
return locPath, nil
|
|
}
|
|
|
|
// localizeDir localizes root path and returns the localized path
|
|
func (lc *localizer) localizeDir(path string) (string, error) {
|
|
ldr, err := lc.ldr.New(path)
|
|
if err != nil {
|
|
return "", errors.Wrap(err)
|
|
}
|
|
defer func() { _ = ldr.Cleanup() }()
|
|
|
|
root, err := filesys.ConfirmDir(lc.fSys, ldr.Root())
|
|
if err != nil {
|
|
log.Panicf("unable to establish validated root reference %q: %s", path, err)
|
|
}
|
|
var locPath string
|
|
if repo := ldr.Repo(); repo != "" {
|
|
// TODO(annasong): You need to check if you can add a localize directory here to store
|
|
// the remote file content. There may be a directory that shares the localize directory name.
|
|
locPath = locRootPath(path, repo, root)
|
|
} else {
|
|
locPath, err = filepath.Rel(lc.root.String(), root.String())
|
|
if err != nil {
|
|
log.Panicf("cannot find relative path between scoped localize roots %q and %q: %s", lc.root, root, err)
|
|
}
|
|
}
|
|
newDst := filepath.Join(lc.dst, locPath)
|
|
if err = lc.fSys.MkdirAll(newDst); err != nil {
|
|
return "", errors.WrapPrefixf(err, "unable to create root %q in localize destination", path)
|
|
}
|
|
err = (&localizer{
|
|
fSys: lc.fSys,
|
|
validator: lc.validator,
|
|
rFactory: lc.rFactory,
|
|
pLdr: lc.pLdr,
|
|
ldr: ldr,
|
|
root: root,
|
|
dst: newDst,
|
|
}).localize()
|
|
if err != nil {
|
|
return "", errors.WrapPrefixf(err, "unable to localize root %q", path)
|
|
}
|
|
return locPath, nil
|
|
}
|
|
|
|
// localizeBuiltinPlugins localizes built-in plugins on kust that can contain file paths. The built-in plugins
|
|
// can be inline or in a file. This excludes the HelmChartInflationGenerator.
|
|
//
|
|
// Note that the localization in this function has not been implemented yet.
|
|
func (lc *localizer) localizeBuiltinPlugins(kust *types.Kustomization) error {
|
|
for fieldName, plugins := range map[string]struct {
|
|
entries []string
|
|
localizer kio.Filter
|
|
}{
|
|
"generators": {
|
|
kust.Generators,
|
|
&localizeBuiltinGenerators{},
|
|
},
|
|
"transformers": {
|
|
kust.Transformers,
|
|
&localizeBuiltinTransformers{},
|
|
},
|
|
"validators": {
|
|
kust.Validators,
|
|
&localizeBuiltinTransformers{},
|
|
},
|
|
} {
|
|
for i, entry := range plugins.entries {
|
|
rm, isPath, err := lc.loadResource(entry)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, "unable to load %s entry", fieldName)
|
|
}
|
|
err = rm.ApplyFilter(plugins.localizer)
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
localizedPlugin, err := rm.AsYaml()
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, "unable to serialize localized %s entry %q", fieldName, entry)
|
|
}
|
|
var newEntry string
|
|
if isPath {
|
|
// TODO(annasong): write localizedPlugin to dst
|
|
newEntry = entry
|
|
} else {
|
|
newEntry = string(localizedPlugin)
|
|
}
|
|
plugins.entries[i] = newEntry
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// loadResource tries to load resourceEntry as a file path or inline.
|
|
// On success, loadResource returns the loaded resource map and whether resourceEntry is a file path.
|
|
func (lc *localizer) loadResource(resourceEntry string) (resmap.ResMap, bool, error) {
|
|
var fileErr error
|
|
rm, inlineErr := lc.rFactory.NewResMapFromBytes([]byte(resourceEntry))
|
|
if inlineErr != nil {
|
|
rm, fileErr = lc.rFactory.FromFile(lc.ldr, resourceEntry)
|
|
if fileErr != nil {
|
|
err := ResourceLoadError{
|
|
InlineError: inlineErr,
|
|
FileError: fileErr,
|
|
}
|
|
return nil, false, errors.WrapPrefixf(err, "unable to load resource entry %q", resourceEntry)
|
|
}
|
|
}
|
|
return rm, fileErr == nil, nil
|
|
}
|