mirror of https://github.com/k3s-io/k3s
153 lines
4.1 KiB
Go
153 lines
4.1 KiB
Go
|
// Copyright 2019 The Kubernetes Authors.
|
||
|
// SPDX-License-Identifier: Apache-2.0
|
||
|
|
||
|
package kio
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
|
||
|
"sigs.k8s.io/kustomize/kyaml/errors"
|
||
|
"sigs.k8s.io/kustomize/kyaml/kio/kioutil"
|
||
|
"sigs.k8s.io/kustomize/kyaml/yaml"
|
||
|
)
|
||
|
|
||
|
// LocalPackageWriter writes ResourceNodes to a filesystem
|
||
|
type LocalPackageWriter struct {
|
||
|
Kind string `yaml:"kind,omitempty"`
|
||
|
|
||
|
// PackagePath is the path to the package directory.
|
||
|
PackagePath string `yaml:"path,omitempty"`
|
||
|
|
||
|
// KeepReaderAnnotations if set will retain the annotations set by LocalPackageReader
|
||
|
KeepReaderAnnotations bool `yaml:"keepReaderAnnotations,omitempty"`
|
||
|
|
||
|
// ClearAnnotations will clear annotations before writing the resources
|
||
|
ClearAnnotations []string `yaml:"clearAnnotations,omitempty"`
|
||
|
}
|
||
|
|
||
|
var _ Writer = LocalPackageWriter{}
|
||
|
|
||
|
func (r LocalPackageWriter) Write(nodes []*yaml.RNode) error {
|
||
|
// set the path and index annotations if they are missing
|
||
|
if err := kioutil.DefaultPathAndIndexAnnotation("", nodes); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if s, err := os.Stat(r.PackagePath); err != nil {
|
||
|
return err
|
||
|
} else if !s.IsDir() {
|
||
|
// if the user specified input isn't a directory, the package is the directory of the
|
||
|
// target
|
||
|
r.PackagePath = filepath.Dir(r.PackagePath)
|
||
|
}
|
||
|
|
||
|
// setup indexes for writing Resources back to files
|
||
|
if err := r.errorIfMissingRequiredAnnotation(nodes); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
outputFiles, err := r.indexByFilePath(nodes)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
for k := range outputFiles {
|
||
|
if err = kioutil.SortNodes(outputFiles[k]); err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if !r.KeepReaderAnnotations {
|
||
|
r.ClearAnnotations = append(r.ClearAnnotations, kioutil.PathAnnotation)
|
||
|
}
|
||
|
|
||
|
// validate outputs before writing any
|
||
|
for path := range outputFiles {
|
||
|
outputPath := filepath.Join(r.PackagePath, path)
|
||
|
if st, err := os.Stat(outputPath); !os.IsNotExist(err) {
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
if st.IsDir() {
|
||
|
return fmt.Errorf("config.kubernetes.io/path cannot be a directory: %s", path)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
err = os.MkdirAll(filepath.Dir(outputPath), 0700)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// write files
|
||
|
for path := range outputFiles {
|
||
|
outputPath := filepath.Join(r.PackagePath, path)
|
||
|
err = os.MkdirAll(filepath.Dir(filepath.Join(r.PackagePath, path)), 0700)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
|
||
|
f, err := os.OpenFile(outputPath, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, os.FileMode(0600))
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
if err := func() error {
|
||
|
defer f.Close()
|
||
|
w := ByteWriter{
|
||
|
Writer: f,
|
||
|
KeepReaderAnnotations: r.KeepReaderAnnotations,
|
||
|
ClearAnnotations: r.ClearAnnotations,
|
||
|
}
|
||
|
if err = w.Write(outputFiles[path]); err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
return nil
|
||
|
}(); err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (r LocalPackageWriter) errorIfMissingRequiredAnnotation(nodes []*yaml.RNode) error {
|
||
|
for i := range nodes {
|
||
|
for _, s := range requiredResourcePackageAnnotations {
|
||
|
key, err := nodes[i].Pipe(yaml.GetAnnotation(s))
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err)
|
||
|
}
|
||
|
if key == nil || key.YNode() == nil || key.YNode().Value == "" {
|
||
|
return errors.Errorf(
|
||
|
"resources must be annotated with %s to be written to files", s)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (r LocalPackageWriter) indexByFilePath(nodes []*yaml.RNode) (map[string][]*yaml.RNode, error) {
|
||
|
outputFiles := map[string][]*yaml.RNode{}
|
||
|
for i := range nodes {
|
||
|
// parse the file write path
|
||
|
node := nodes[i]
|
||
|
value, err := node.Pipe(yaml.GetAnnotation(kioutil.PathAnnotation))
|
||
|
if err != nil {
|
||
|
// this should never happen if errorIfMissingRequiredAnnotation was run
|
||
|
return nil, errors.Wrap(err)
|
||
|
}
|
||
|
path := value.YNode().Value
|
||
|
outputFiles[path] = append(outputFiles[path], node)
|
||
|
|
||
|
if filepath.IsAbs(path) {
|
||
|
return nil, errors.Errorf("package paths may not be absolute paths")
|
||
|
}
|
||
|
if strings.Contains(filepath.Clean(path), "..") {
|
||
|
return nil, fmt.Errorf("resource must be written under package %s: %s",
|
||
|
r.PackagePath, filepath.Clean(path))
|
||
|
}
|
||
|
}
|
||
|
return outputFiles, nil
|
||
|
}
|