airshipctl/krm-functions/toolbox/main.go

177 lines
4.6 KiB
Go

/*
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
https://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package main
import (
"fmt"
"io"
"io/ioutil"
"os"
"os/exec"
"path/filepath"
v1 "k8s.io/api/core/v1"
kerror "k8s.io/apimachinery/pkg/util/errors"
"sigs.k8s.io/kustomize/kyaml/fn/framework"
"sigs.k8s.io/kustomize/kyaml/kio"
kyaml "sigs.k8s.io/kustomize/kyaml/yaml"
"opendev.org/airship/airshipctl/pkg/document/plugin/kyamlutils"
"opendev.org/airship/airshipctl/pkg/log"
)
const (
// EnvRenderedBundlePath will be passed to the script, it will contain path to the rendered bundle
EnvRenderedBundlePath = "RENDERED_BUNDLE_PATH"
// ResourceGroupFilter used for filtering input bundle by document group
ResourceGroupFilter = "RESOURCE_GROUP_FILTER"
// ResourceVersionFilter used for filtering input bundle by document version
ResourceVersionFilter = "RESOURCE_VERSION_FILTER"
// ResourceKindFilter used for filtering input bundle by document kind
ResourceKindFilter = "RESOURCE_KIND_FILTER"
scriptPath = "script.sh"
scriptKey = "script"
bundleFile = "bundle.yaml"
workdir = "/tmp"
)
func main() {
cfg := &v1.ConfigMap{}
resourceList := &framework.ResourceList{FunctionConfig: &cfg}
runner := ScriptRunner{
ScriptFile: scriptPath,
WorkDir: workdir,
RenderedBundleFile: bundleFile,
DataKey: scriptKey,
ResourceList: resourceList,
ConfigMap: cfg,
ErrStream: os.Stderr,
OutStream: os.Stdout,
}
cmd := framework.Command(resourceList, runner.Run)
if err := cmd.Execute(); err != nil {
log.Print(err)
os.Exit(1)
}
}
// ScriptRunner writes to file system and executes the script
type ScriptRunner struct {
ScriptFile, WorkDir, DataKey, RenderedBundleFile string
ErrStream io.Writer
OutStream io.Writer
ConfigMap *v1.ConfigMap
ResourceList *framework.ResourceList
}
// Run writes the script and bundle to the file system and executes it
func (c *ScriptRunner) Run() error {
bundlePath, scriptPath := c.getBundleAndScriptPath()
script, exist := c.ConfigMap.Data[c.DataKey]
if !exist {
return fmt.Errorf("ConfigMap '%s/%s' doesnt' have specified script key '%s'",
c.ConfigMap.Namespace, c.ConfigMap.Name, c.DataKey)
}
err := ioutil.WriteFile(scriptPath, []byte(script), 0555)
if err != nil {
return err
}
err = c.writeBundle(bundlePath, c.ResourceList.Items)
if err != nil {
return err
}
c.ResourceList.Items = nil
clicmd := exec.Command(scriptPath)
clicmd.Stdout = c.OutStream
clicmd.Stderr = c.ErrStream
clicmd.Env = os.Environ()
clicmd.Env = append(clicmd.Env, fmt.Sprintf("%s=%s", EnvRenderedBundlePath, bundlePath))
err = clicmd.Start()
if err != nil {
return err
}
return clicmd.Wait()
}
// Cleanup removes script and bundle files from filesystem
func (c *ScriptRunner) Cleanup() error {
bundlePath, scriptPath := c.getBundleAndScriptPath()
scriptErr := os.Remove(scriptPath)
if os.IsNotExist(scriptErr) {
// If file doesn't exist no error happened
scriptErr = nil
}
bundleErr := os.Remove(bundlePath)
if os.IsNotExist(bundleErr) {
// If file doesn't exist no error happened
bundleErr = nil
}
return kerror.NewAggregate([]error{scriptErr, bundleErr})
}
func (c *ScriptRunner) getBundleAndScriptPath() (string, string) {
return filepath.Join(c.WorkDir, c.RenderedBundleFile), filepath.Join(c.WorkDir, c.ScriptFile)
}
func (c *ScriptRunner) writeBundle(path string, items []*kyaml.RNode) error {
f, err := os.Create(path)
if err != nil {
return err
}
defer f.Close()
items, err = c.FilterBundle(items)
if err != nil {
return err
}
pipeline := kio.Pipeline{
Outputs: []kio.Writer{
kio.ByteWriter{
Writer: f,
},
},
Inputs: []kio.Reader{
kio.ResourceNodeSlice(items),
},
}
return pipeline.Execute()
}
// FilterBundle uses for filtering input documents
func (c *ScriptRunner) FilterBundle(items []*kyaml.RNode) ([]*kyaml.RNode, error) {
group := os.Getenv(ResourceGroupFilter)
version := os.Getenv(ResourceVersionFilter)
kind := os.Getenv(ResourceKindFilter)
log.Printf("Filtering input bundle by Group: %s, Version: %s, Kind: %s",
group, version, kind)
return kyamlutils.DocumentSelector{}.
ByGVK(group, version, kind).
Filter(items)
}