mirror of https://github.com/k3s-io/k3s
327 lines
9.7 KiB
Go
327 lines
9.7 KiB
Go
// Copyright 2019 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package kio
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"sigs.k8s.io/kustomize/kyaml/errors"
|
|
"sigs.k8s.io/kustomize/kyaml/kio/kioutil"
|
|
"sigs.k8s.io/kustomize/kyaml/sets"
|
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
)
|
|
|
|
// requiredResourcePackageAnnotations are annotations that are required to write resources back to
|
|
// files.
|
|
var requiredResourcePackageAnnotations = []string{kioutil.IndexAnnotation, kioutil.PathAnnotation}
|
|
|
|
// PackageBuffer implements Reader and Writer, storing Resources in a local field.
|
|
type PackageBuffer struct {
|
|
Nodes []*yaml.RNode
|
|
}
|
|
|
|
func (r *PackageBuffer) Read() ([]*yaml.RNode, error) {
|
|
return r.Nodes, nil
|
|
}
|
|
|
|
func (r *PackageBuffer) Write(nodes []*yaml.RNode) error {
|
|
r.Nodes = nodes
|
|
return nil
|
|
}
|
|
|
|
// LocalPackageReadWriter reads and writes Resources from / to a local directory.
|
|
// When writing, LocalPackageReadWriter will delete files if all of the Resources from
|
|
// that file have been removed from the output.
|
|
type LocalPackageReadWriter struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
KeepReaderAnnotations bool `yaml:"keepReaderAnnotations,omitempty"`
|
|
|
|
// PackagePath is the path to the package directory.
|
|
PackagePath string `yaml:"path,omitempty"`
|
|
|
|
// PackageFileName is the name of file containing package metadata.
|
|
// It will be used to identify package.
|
|
PackageFileName string `yaml:"packageFileName,omitempty"`
|
|
|
|
// MatchFilesGlob configures Read to only read Resources from files matching any of the
|
|
// provided patterns.
|
|
// Defaults to ["*.yaml", "*.yml"] if empty. To match all files specify ["*"].
|
|
MatchFilesGlob []string `yaml:"matchFilesGlob,omitempty"`
|
|
|
|
// IncludeSubpackages will configure Read to read Resources from subpackages.
|
|
// Subpackages are identified by presence of PackageFileName.
|
|
IncludeSubpackages bool `yaml:"includeSubpackages,omitempty"`
|
|
|
|
// ErrorIfNonResources will configure Read to throw an error if yaml missing missing
|
|
// apiVersion or kind is read.
|
|
ErrorIfNonResources bool `yaml:"errorIfNonResources,omitempty"`
|
|
|
|
// OmitReaderAnnotations will cause the reader to skip annotating Resources with the file
|
|
// path and mode.
|
|
OmitReaderAnnotations bool `yaml:"omitReaderAnnotations,omitempty"`
|
|
|
|
// SetAnnotations are annotations to set on the Resources as they are read.
|
|
SetAnnotations map[string]string `yaml:"setAnnotations,omitempty"`
|
|
|
|
// NoDeleteFiles if set to true, LocalPackageReadWriter won't delete any files
|
|
NoDeleteFiles bool `yaml:"noDeleteFiles,omitempty"`
|
|
|
|
files sets.String
|
|
|
|
// FileSkipFunc is a function which returns true if reader should ignore
|
|
// the file
|
|
FileSkipFunc LocalPackageSkipFileFunc
|
|
}
|
|
|
|
func (r *LocalPackageReadWriter) Read() ([]*yaml.RNode, error) {
|
|
nodes, err := LocalPackageReader{
|
|
PackagePath: r.PackagePath,
|
|
MatchFilesGlob: r.MatchFilesGlob,
|
|
IncludeSubpackages: r.IncludeSubpackages,
|
|
ErrorIfNonResources: r.ErrorIfNonResources,
|
|
SetAnnotations: r.SetAnnotations,
|
|
PackageFileName: r.PackageFileName,
|
|
FileSkipFunc: r.FileSkipFunc,
|
|
}.Read()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err)
|
|
}
|
|
// keep track of all the files
|
|
if !r.NoDeleteFiles {
|
|
r.files, err = r.getFiles(nodes)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err)
|
|
}
|
|
}
|
|
return nodes, nil
|
|
}
|
|
|
|
func (r *LocalPackageReadWriter) Write(nodes []*yaml.RNode) error {
|
|
newFiles, err := r.getFiles(nodes)
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
var clear []string
|
|
for k := range r.SetAnnotations {
|
|
clear = append(clear, k)
|
|
}
|
|
err = LocalPackageWriter{
|
|
PackagePath: r.PackagePath,
|
|
ClearAnnotations: clear,
|
|
KeepReaderAnnotations: r.KeepReaderAnnotations,
|
|
}.Write(nodes)
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
deleteFiles := r.files.Difference(newFiles)
|
|
for f := range deleteFiles {
|
|
if err = os.Remove(filepath.Join(r.PackagePath, f)); err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *LocalPackageReadWriter) getFiles(nodes []*yaml.RNode) (sets.String, error) {
|
|
val := sets.String{}
|
|
for _, n := range nodes {
|
|
path, _, err := kioutil.GetFileAnnotations(n)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err)
|
|
}
|
|
val.Insert(path)
|
|
}
|
|
return val, nil
|
|
}
|
|
|
|
// LocalPackageSkipFileFunc is a function which returns true if the file
|
|
// in the package should be ignored by reader.
|
|
// relPath is an OS specific relative path
|
|
type LocalPackageSkipFileFunc func(relPath string) bool
|
|
|
|
// LocalPackageReader reads ResourceNodes from a local package.
|
|
type LocalPackageReader struct {
|
|
Kind string `yaml:"kind,omitempty"`
|
|
|
|
// PackagePath is the path to the package directory.
|
|
PackagePath string `yaml:"path,omitempty"`
|
|
|
|
// PackageFileName is the name of file containing package metadata.
|
|
// It will be used to identify package.
|
|
PackageFileName string `yaml:"packageFileName,omitempty"`
|
|
|
|
// MatchFilesGlob configures Read to only read Resources from files matching any of the
|
|
// provided patterns.
|
|
// Defaults to ["*.yaml", "*.yml"] if empty. To match all files specify ["*"].
|
|
MatchFilesGlob []string `yaml:"matchFilesGlob,omitempty"`
|
|
|
|
// IncludeSubpackages will configure Read to read Resources from subpackages.
|
|
// Subpackages are identified by presence of PackageFileName.
|
|
IncludeSubpackages bool `yaml:"includeSubpackages,omitempty"`
|
|
|
|
// ErrorIfNonResources will configure Read to throw an error if yaml missing missing
|
|
// apiVersion or kind is read.
|
|
ErrorIfNonResources bool `yaml:"errorIfNonResources,omitempty"`
|
|
|
|
// OmitReaderAnnotations will cause the reader to skip annotating Resources with the file
|
|
// path and mode.
|
|
OmitReaderAnnotations bool `yaml:"omitReaderAnnotations,omitempty"`
|
|
|
|
// SetAnnotations are annotations to set on the Resources as they are read.
|
|
SetAnnotations map[string]string `yaml:"setAnnotations,omitempty"`
|
|
|
|
// FileSkipFunc is a function which returns true if reader should ignore
|
|
// the file
|
|
FileSkipFunc LocalPackageSkipFileFunc
|
|
}
|
|
|
|
var _ Reader = LocalPackageReader{}
|
|
|
|
var DefaultMatch = []string{"*.yaml", "*.yml"}
|
|
var JSONMatch = []string{"*.json"}
|
|
var MatchAll = append(DefaultMatch, JSONMatch...)
|
|
|
|
// Read reads the Resources.
|
|
func (r LocalPackageReader) Read() ([]*yaml.RNode, error) {
|
|
if r.PackagePath == "" {
|
|
return nil, fmt.Errorf("must specify package path")
|
|
}
|
|
|
|
// use slash for path
|
|
r.PackagePath = filepath.ToSlash(r.PackagePath)
|
|
if len(r.MatchFilesGlob) == 0 {
|
|
r.MatchFilesGlob = DefaultMatch
|
|
}
|
|
|
|
var operand ResourceNodeSlice
|
|
var pathRelativeTo string
|
|
var err error
|
|
ignoreFilesMatcher := &ignoreFilesMatcher{}
|
|
r.PackagePath, err = filepath.Abs(r.PackagePath)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err)
|
|
}
|
|
err = filepath.Walk(r.PackagePath, func(
|
|
path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
|
|
// is this the user specified path?
|
|
if path == r.PackagePath {
|
|
if info.IsDir() {
|
|
// skip the root package directory, but check for a
|
|
// .krmignore file first.
|
|
pathRelativeTo = r.PackagePath
|
|
return ignoreFilesMatcher.readIgnoreFile(path)
|
|
}
|
|
|
|
// user specified path is a file rather than a directory.
|
|
// make its path relative to its parent so it can be written to another file.
|
|
pathRelativeTo = filepath.Dir(r.PackagePath)
|
|
}
|
|
|
|
// check if we should skip the directory or file
|
|
if info.IsDir() {
|
|
return r.shouldSkipDir(path, ignoreFilesMatcher)
|
|
}
|
|
|
|
// get the relative path to file within the package so we can write the files back out
|
|
// to another location.
|
|
relPath, err := filepath.Rel(pathRelativeTo, path)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, pathRelativeTo)
|
|
}
|
|
if match, err := r.shouldSkipFile(path, relPath, ignoreFilesMatcher); err != nil {
|
|
return err
|
|
} else if match {
|
|
// skip this file
|
|
return nil
|
|
}
|
|
|
|
r.initReaderAnnotations(relPath, info)
|
|
nodes, err := r.readFile(path, info)
|
|
if err != nil {
|
|
return errors.WrapPrefixf(err, path)
|
|
}
|
|
operand = append(operand, nodes...)
|
|
return nil
|
|
})
|
|
return operand, err
|
|
}
|
|
|
|
// readFile reads the ResourceNodes from a file
|
|
func (r *LocalPackageReader) readFile(path string, _ os.FileInfo) ([]*yaml.RNode, error) {
|
|
f, err := os.Open(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer f.Close()
|
|
|
|
rr := &ByteReader{
|
|
DisableUnwrapping: true,
|
|
Reader: f,
|
|
OmitReaderAnnotations: r.OmitReaderAnnotations,
|
|
SetAnnotations: r.SetAnnotations,
|
|
}
|
|
return rr.Read()
|
|
}
|
|
|
|
// shouldSkipFile returns true if the file should be skipped
|
|
func (r *LocalPackageReader) shouldSkipFile(path, relPath string, matcher *ignoreFilesMatcher) (bool, error) {
|
|
// check if the file is covered by a .krmignore file.
|
|
if matcher.matchFile(path) {
|
|
return true, nil
|
|
}
|
|
|
|
if r.FileSkipFunc != nil && r.FileSkipFunc(relPath) {
|
|
return true, nil
|
|
}
|
|
|
|
// check if the files are in scope
|
|
for _, g := range r.MatchFilesGlob {
|
|
if match, err := filepath.Match(g, filepath.Base(path)); err != nil {
|
|
return true, errors.Wrap(err)
|
|
} else if match {
|
|
return false, nil
|
|
}
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// initReaderAnnotations adds the LocalPackageReader Annotations to r.SetAnnotations
|
|
func (r *LocalPackageReader) initReaderAnnotations(path string, _ os.FileInfo) {
|
|
if r.SetAnnotations == nil {
|
|
r.SetAnnotations = map[string]string{}
|
|
}
|
|
if !r.OmitReaderAnnotations {
|
|
r.SetAnnotations[kioutil.PathAnnotation] = path
|
|
}
|
|
}
|
|
|
|
// shouldSkipDir returns a filepath.SkipDir if the directory should be skipped
|
|
func (r *LocalPackageReader) shouldSkipDir(path string, matcher *ignoreFilesMatcher) error {
|
|
if matcher.matchDir(path) {
|
|
return filepath.SkipDir
|
|
}
|
|
|
|
if r.PackageFileName == "" {
|
|
return nil
|
|
}
|
|
// check if this is a subpackage
|
|
_, err := os.Stat(filepath.Join(path, r.PackageFileName))
|
|
if os.IsNotExist(err) {
|
|
return nil
|
|
} else if err != nil {
|
|
return errors.Wrap(err)
|
|
}
|
|
if !r.IncludeSubpackages {
|
|
return filepath.SkipDir
|
|
}
|
|
return matcher.readIgnoreFile(path)
|
|
}
|