/*
 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"
	"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
}

// RunCommand phase run command
type RunCommand struct {
	PhaseID ifc.ID
	Options ifc.RunOptions
	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.PhaseID)
	if err != nil {
		return err
	}
	return phase.Run(c.Options)
}

// 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 util.PrintObjects(phaseList, util.PhaseListFormat, c.Writer, false)
	}
	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
	}
	if c.Options.FormatType == YamlOutputFormat {
		return yaml.WriteOut(c.Writer, phasePlans)
	}
	return util.PrintObjects(phasePlans, util.PlanListFormat, c.Writer, false)
}

// PlanRunFlags options for phase run command
type PlanRunFlags struct {
	GenericRunFlags
	ResumeFromPhase string
}

// PlanRunCommand phase run command
type PlanRunCommand struct {
	PlanID  ifc.ID
	Options ifc.PlanRunOptions
	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.PlanID)
	if err != nil {
		return err
	}
	return plan.Run(c.Options)
}

// 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()
}