airshipctl/pkg/phase/command.go
Ruslan Aliev bc9f97ff2e Embed validation cfg into phase and plan definition
* validation config is now part of airshipctl api
 * additional CRD locations can be only kustomize entrypoints
 * changed mechanism to call document-validation executor to allow
   to pass validation config from phase or plan
 * kubeval version pinned to the latest 0.16.1
 * default k8s version to validate against uplifted to 1.18.6
 * default URL with k8s schemas changed to more updated and reliable

Change-Id: Ifb24be224d5f0860d323a671b94e28a86debc65b
Signed-off-by: Ruslan Aliev <raliev@mirantis.com>
Closes: #563
2021-06-11 15:37:38 +00:00

366 lines
7.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 phase
import (
"io"
"os"
"path/filepath"
"strings"
"time"
"opendev.org/airship/airshipctl/pkg/cluster/clustermap"
"opendev.org/airship/airshipctl/pkg/config"
"opendev.org/airship/airshipctl/pkg/document"
phaseerrors "opendev.org/airship/airshipctl/pkg/phase/errors"
"opendev.org/airship/airshipctl/pkg/phase/ifc"
"opendev.org/airship/airshipctl/pkg/util/yaml"
)
const (
// TableOutputFormat table
TableOutputFormat = "table"
// YamlOutputFormat yaml
YamlOutputFormat = "yaml"
)
// GenericRunFlags generic options for run command
type GenericRunFlags struct {
DryRun bool
Timeout time.Duration
}
// RunFlags options for phase run command
type RunFlags struct {
GenericRunFlags
PhaseID ifc.ID
}
// RunCommand phase run command
type RunCommand struct {
Options RunFlags
Factory config.Factory
}
// RunE runs the phase
func (c *RunCommand) RunE() error {
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
client := NewClient(helper)
phase, err := client.PhaseByID(c.Options.PhaseID)
if err != nil {
return err
}
return phase.Run(ifc.RunOptions{DryRun: c.Options.DryRun, Timeout: c.Options.Timeout})
}
// ListCommand phase list command
type ListCommand struct {
Factory config.Factory
Writer io.Writer
ClusterName string
PlanID ifc.ID
OutputFormat string
}
// RunE runs a phase list command
func (c *ListCommand) RunE() error {
if c.OutputFormat != TableOutputFormat && c.OutputFormat != YamlOutputFormat {
return phaseerrors.ErrInvalidFormat{RequestedFormat: c.OutputFormat}
}
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
o := ifc.ListPhaseOptions{ClusterName: c.ClusterName, PlanID: c.PlanID}
phaseList, err := helper.ListPhases(o)
if err != nil {
return err
}
if c.OutputFormat == TableOutputFormat {
return PrintPhaseListTable(c.Writer, phaseList)
}
return yaml.WriteOut(c.Writer, phaseList)
}
// TreeCommand plan command
type TreeCommand struct {
Factory config.Factory
PhaseID ifc.ID
Writer io.Writer
Argument string
}
// RunE runs the phase tree command
func (c *TreeCommand) RunE() error {
var entrypoint string
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
client := NewClient(helper)
var manifestsDir string
// check if its a relative path
if _, err = os.Stat(c.Argument); err == nil {
// capture manifests directory from phase relative path
manifestsDir = strings.SplitAfter(c.Argument, "/manifests")[0]
entrypoint = filepath.Join(c.Argument, document.KustomizationFile)
} else {
c.PhaseID.Name = c.Argument
manifestsDir = filepath.Join(helper.TargetPath(), helper.PhaseRepoDir())
var phase ifc.Phase
phase, err = client.PhaseByID(c.PhaseID)
if err != nil {
return err
}
var rootPath string
rootPath, err = phase.DocumentRoot()
if err != nil {
return err
}
entrypoint = filepath.Join(rootPath, document.KustomizationFile)
}
t, err := document.BuildKustomTree(entrypoint, c.Writer, manifestsDir)
if err != nil {
return err
}
t.PrintTree("")
return nil
}
// PlanListFlags flags given for plan list command
type PlanListFlags struct {
FormatType string
}
// PlanListCommand phase list command
type PlanListCommand struct {
Options PlanListFlags
Factory config.Factory
Writer io.Writer
}
// RunE runs a plan list command
func (c *PlanListCommand) RunE() error {
if c.Options.FormatType != TableOutputFormat && c.Options.FormatType != YamlOutputFormat {
return phaseerrors.ErrInvalidFormat{RequestedFormat: c.Options.FormatType}
}
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
phasePlans, err := helper.ListPlans()
if err != nil {
return err
}
switch {
case c.Options.FormatType == YamlOutputFormat:
return yaml.WriteOut(c.Writer, phasePlans)
case c.Options.FormatType == TableOutputFormat:
return PrintPlanListTable(c.Writer, phasePlans)
default:
return PrintPlanListTable(c.Writer, phasePlans)
}
}
// PlanRunFlags options for phase run command
type PlanRunFlags struct {
GenericRunFlags
PlanID ifc.ID
}
// PlanRunCommand phase run command
type PlanRunCommand struct {
Options PlanRunFlags
Factory config.Factory
}
// RunE executes phase plan
func (c *PlanRunCommand) RunE() error {
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
client := NewClient(helper)
plan, err := client.PlanByID(c.Options.PlanID)
if err != nil {
return err
}
return plan.Run(ifc.RunOptions{DryRun: c.Options.DryRun, Timeout: c.Options.Timeout})
}
// ClusterListCommand options for cluster list command
type ClusterListCommand struct {
Factory config.Factory
Writer io.Writer
Format string
}
// RunE executes cluster list command
func (c *ClusterListCommand) RunE() error {
if c.Format != TableOutputFormat && c.Format != "name" {
return phaseerrors.ErrInvalidOutputFormat{RequestedFormat: c.Format}
}
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
clusterMap, err := helper.ClusterMap()
if err != nil {
return err
}
err = clusterMap.Write(c.Writer, clustermap.WriteOptions{Format: c.Format})
if err != nil {
return err
}
return nil
}
// ValidateFlags options for phase validate command
type ValidateFlags struct {
PhaseID ifc.ID
}
// ValidateCommand phase validate command
type ValidateCommand struct {
Options ValidateFlags
Factory config.Factory
}
// RunE runs the phase validate command
func (c *ValidateCommand) RunE() error {
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
client := NewClient(helper)
phase, err := client.PhaseByID(c.Options.PhaseID)
if err != nil {
return err
}
return phase.Validate()
}
// StatusFlags is a struct to define status type
type StatusFlags struct {
Timeout time.Duration
PhaseID ifc.ID
Progress bool
}
// StatusCommand is a struct which defines status
type StatusCommand struct {
Options StatusFlags
Factory config.Factory
}
// RunE returns the status of the given phase
func (s *StatusCommand) RunE() error {
cfg, err := s.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
ph, err := NewClient(helper).PhaseByID(s.Options.PhaseID)
if err != nil {
return err
}
_, err = ph.Status()
return err
}
// PlanValidateFlags options for plan validate command
type PlanValidateFlags struct {
PlanID ifc.ID
}
// PlanValidateCommand plan validate command
type PlanValidateCommand struct {
Options PlanValidateFlags
Factory config.Factory
}
// RunE runs the plan validate command
func (c *PlanValidateCommand) RunE() error {
cfg, err := c.Factory()
if err != nil {
return err
}
helper, err := NewHelper(cfg)
if err != nil {
return err
}
client := NewClient(helper)
plan, err := client.PlanByID(c.Options.PlanID)
if err != nil {
return err
}
return plan.Validate()
}