mirror of https://github.com/fluxcd/flux2.git
118 lines
3.4 KiB
Go
118 lines
3.4 KiB
Go
/*
|
|
Copyright 2022 The Flux authors
|
|
|
|
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
|
|
|
|
http://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 (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
oci "github.com/fluxcd/pkg/oci/client"
|
|
"github.com/fluxcd/pkg/sourceignore"
|
|
)
|
|
|
|
var buildArtifactCmd = &cobra.Command{
|
|
Use: "artifact",
|
|
Short: "Build artifact",
|
|
Long: withPreviewNote(`The build artifact command creates a tgz file with the manifests
|
|
from the given directory or a single manifest file.`),
|
|
Example: ` # Build the given manifests directory into an artifact
|
|
flux build artifact --path ./path/to/local/manifests --output ./path/to/artifact.tgz
|
|
|
|
# Build the given single manifest file into an artifact
|
|
flux build artifact --path ./path/to/local/manifest.yaml --output ./path/to/artifact.tgz
|
|
|
|
# List the files bundled in the artifact
|
|
tar -ztvf ./path/to/artifact.tgz
|
|
`,
|
|
RunE: buildArtifactCmdRun,
|
|
}
|
|
|
|
type buildArtifactFlags struct {
|
|
output string
|
|
path string
|
|
ignorePaths []string
|
|
}
|
|
|
|
var excludeOCI = append(strings.Split(sourceignore.ExcludeVCS, ","), strings.Split(sourceignore.ExcludeExt, ",")...)
|
|
|
|
var buildArtifactArgs buildArtifactFlags
|
|
|
|
func init() {
|
|
buildArtifactCmd.Flags().StringVarP(&buildArtifactArgs.path, "path", "p", "", "Path to the directory where the Kubernetes manifests are located.")
|
|
buildArtifactCmd.Flags().StringVarP(&buildArtifactArgs.output, "output", "o", "artifact.tgz", "Path to where the artifact tgz file should be written.")
|
|
buildArtifactCmd.Flags().StringSliceVar(&buildArtifactArgs.ignorePaths, "ignore-paths", excludeOCI, "set paths to ignore in .gitignore format")
|
|
|
|
buildCmd.AddCommand(buildArtifactCmd)
|
|
}
|
|
|
|
func buildArtifactCmdRun(cmd *cobra.Command, args []string) error {
|
|
if buildArtifactArgs.path == "" {
|
|
return fmt.Errorf("invalid path %q", buildArtifactArgs.path)
|
|
}
|
|
|
|
path := buildArtifactArgs.path
|
|
var err error
|
|
if buildArtifactArgs.path == "-" {
|
|
path, err = saveReaderToFile(os.Stdin)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
defer os.Remove(path)
|
|
}
|
|
|
|
if _, err := os.Stat(path); err != nil {
|
|
return fmt.Errorf("invalid path '%s', must point to an existing directory or file", path)
|
|
}
|
|
|
|
logger.Actionf("building artifact from %s", path)
|
|
|
|
ociClient := oci.NewClient(oci.DefaultOptions())
|
|
if err := ociClient.Build(buildArtifactArgs.output, path, buildArtifactArgs.ignorePaths); err != nil {
|
|
return fmt.Errorf("building artifact failed, error: %w", err)
|
|
}
|
|
|
|
logger.Successf("artifact created at %s", buildArtifactArgs.output)
|
|
return nil
|
|
}
|
|
|
|
func saveReaderToFile(reader io.Reader) (string, error) {
|
|
b, err := io.ReadAll(bufio.NewReader(reader))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
b = bytes.TrimRight(b, "\r\n")
|
|
f, err := os.CreateTemp("", "*.yaml")
|
|
if err != nil {
|
|
return "", fmt.Errorf("unable to create temp dir for stdin")
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
if _, err := f.Write(b); err != nil {
|
|
return "", fmt.Errorf("error writing stdin to file: %w", err)
|
|
}
|
|
|
|
return f.Name(), nil
|
|
}
|