bc9f97ff2e
* 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
366 lines
7.6 KiB
Go
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()
|
|
}
|