0a7661ab7c
This commit adds cluster map api object, cluster map interface and it's implementation built on top of cluster map api object. Important note: ClusterMap interface needs a method to identify namespace of the cluster, it can't be a part of api object, because real source for cluster namespace is cluster object from cluster-api upstream lib. This will need further design discussion on how we will find cluster-api kind: Cluster object in our manifests. For now, there is dummy "default" namespace being used Change-Id: I8175f54abbe77331f0c87c0bde50857ee5c0eb1d
259 lines
7.1 KiB
Go
259 lines
7.1 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 (
|
|
"path/filepath"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
airshipv1 "opendev.org/airship/airshipctl/pkg/api/v1alpha1"
|
|
"opendev.org/airship/airshipctl/pkg/cluster/clustermap"
|
|
clusterctl "opendev.org/airship/airshipctl/pkg/clusterctl/client"
|
|
"opendev.org/airship/airshipctl/pkg/config"
|
|
"opendev.org/airship/airshipctl/pkg/document"
|
|
"opendev.org/airship/airshipctl/pkg/events"
|
|
"opendev.org/airship/airshipctl/pkg/k8s/applier"
|
|
"opendev.org/airship/airshipctl/pkg/k8s/kubeconfig"
|
|
k8sutils "opendev.org/airship/airshipctl/pkg/k8s/utils"
|
|
"opendev.org/airship/airshipctl/pkg/log"
|
|
"opendev.org/airship/airshipctl/pkg/phase/ifc"
|
|
)
|
|
|
|
// ExecutorRegistry returns map with executor factories
|
|
type ExecutorRegistry func() map[schema.GroupVersionKind]ifc.ExecutorFactory
|
|
|
|
// DefaultExecutorRegistry returns map with executor factories
|
|
func DefaultExecutorRegistry() map[schema.GroupVersionKind]ifc.ExecutorFactory {
|
|
execMap := make(map[schema.GroupVersionKind]ifc.ExecutorFactory)
|
|
|
|
if err := clusterctl.RegisterExecutor(execMap); err != nil {
|
|
log.Fatal(ErrExecutorRegistration{ExecutorName: "clusterctl", Err: err})
|
|
}
|
|
if err := applier.RegisterExecutor(execMap); err != nil {
|
|
log.Fatal(ErrExecutorRegistration{ExecutorName: "kubernetes-apply", Err: err})
|
|
}
|
|
return execMap
|
|
}
|
|
|
|
// Cmd object to work with phase api
|
|
type Cmd struct {
|
|
DryRun bool
|
|
|
|
Registry ExecutorRegistry
|
|
// Will be used to get processor based on executor action
|
|
Processor events.EventProcessor
|
|
*config.Config
|
|
}
|
|
|
|
func (p *Cmd) getBundle() (document.Bundle, error) {
|
|
tp, err := p.CurrentContextTargetPath()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
meta, err := p.Config.CurrentContextManifestMetadata()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
log.Debugf("Building phase bundle from path %s", tp)
|
|
return document.NewBundleByPath(filepath.Join(tp, meta.PhaseMeta.Path))
|
|
}
|
|
|
|
func (p *Cmd) getPhaseExecutor(name string) (ifc.Executor, error) {
|
|
phaseConfig, err := p.GetPhase(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return p.GetExecutor(phaseConfig)
|
|
}
|
|
|
|
// GetPhase returns particular phase object identified by name
|
|
func (p *Cmd) GetPhase(name string) (*airshipv1.Phase, error) {
|
|
bundle, err := p.getBundle()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
phaseConfig := &airshipv1.Phase{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: name,
|
|
},
|
|
}
|
|
selector, err := document.NewSelector().ByObject(phaseConfig, airshipv1.Scheme)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
doc, err := bundle.SelectOne(selector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err = doc.ToAPIObject(phaseConfig, airshipv1.Scheme); err != nil {
|
|
return nil, err
|
|
}
|
|
return phaseConfig, nil
|
|
}
|
|
|
|
// GetClusterMap returns cluster map object
|
|
func (p *Cmd) GetClusterMap() (clustermap.ClusterMap, error) {
|
|
bundle, err := p.getBundle()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
clusterMap := &airshipv1.ClusterMap{}
|
|
selector, err := document.NewSelector().ByObject(clusterMap, airshipv1.Scheme)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
doc, err := bundle.SelectOne(selector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err = doc.ToAPIObject(clusterMap, airshipv1.Scheme); err != nil {
|
|
return nil, err
|
|
}
|
|
return clustermap.NewClusterMap(clusterMap), nil
|
|
}
|
|
|
|
// GetExecutor referenced in a phase configuration
|
|
func (p *Cmd) GetExecutor(phase *airshipv1.Phase) (ifc.Executor, error) {
|
|
bundle, err := p.getBundle()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
phaseConfig := phase.Config
|
|
// Searching executor configuration document referenced in
|
|
// phase configuration
|
|
refGVK := phaseConfig.ExecutorRef.GroupVersionKind()
|
|
selector := document.NewSelector().
|
|
ByGvk(refGVK.Group, refGVK.Version, refGVK.Kind).
|
|
ByName(phaseConfig.ExecutorRef.Name).
|
|
ByNamespace(phaseConfig.ExecutorRef.Namespace)
|
|
executorDoc, err := bundle.SelectOne(selector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Define executor configuration options
|
|
targetPath, err := p.Config.CurrentContextTargetPath()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var executorDocBundle document.Bundle
|
|
// if entrypoint is defined use it, if not, just pass nil bundle, executors should be ready for that
|
|
if phaseConfig.DocumentEntryPoint != "" {
|
|
bundlePath := filepath.Join(targetPath, phaseConfig.DocumentEntryPoint)
|
|
executorDocBundle, err = document.NewBundleByPath(bundlePath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if p.Registry == nil {
|
|
p.Registry = DefaultExecutorRegistry
|
|
}
|
|
// Look for executor factory defined in registry
|
|
executorFactory, found := p.Registry()[refGVK]
|
|
if !found {
|
|
return nil, ErrExecutorNotFound{GVK: refGVK}
|
|
}
|
|
meta, err := p.Config.CurrentContextManifestMetadata()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cMap, err := p.GetClusterMap()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
kubeConfig := kubeconfig.NewBuilder().
|
|
// TODO add kubeconfig flags path here, when kubeconfig flag is not controlled
|
|
// by config module during config loading.
|
|
WithBundle(meta.PhaseMeta.Path).
|
|
WithClusterMap(cMap).
|
|
WithClusterName(phase.ClusterName).
|
|
WithTempRoot(filepath.Dir(p.Config.LoadedConfigPath())).
|
|
Build()
|
|
return executorFactory(
|
|
ifc.ExecutorConfig{
|
|
ExecutorBundle: executorDocBundle,
|
|
PhaseName: phase.Name,
|
|
ExecutorDocument: executorDoc,
|
|
AirshipConfig: p.Config,
|
|
KubeConfig: kubeConfig,
|
|
ClusterName: phase.ClusterName,
|
|
ClusterMap: cMap,
|
|
})
|
|
}
|
|
|
|
// Exec starts executor goroutine and processes the events
|
|
func (p *Cmd) Exec(name string) error {
|
|
runCh := make(chan events.Event)
|
|
processor := events.NewDefaultProcessor(k8sutils.Streams())
|
|
go func() {
|
|
executor, err := p.getPhaseExecutor(name)
|
|
if err != nil {
|
|
handleError(err, runCh)
|
|
return
|
|
}
|
|
executor.Run(runCh, ifc.RunOptions{
|
|
Debug: log.DebugEnabled(),
|
|
DryRun: p.DryRun,
|
|
})
|
|
}()
|
|
return processor.Process(runCh)
|
|
}
|
|
|
|
// Plan shows available phase names
|
|
func (p *Cmd) Plan() (map[string][]string, error) {
|
|
bundle, err := p.getBundle()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
plan := &airshipv1.PhasePlan{}
|
|
selector, err := document.NewSelector().ByObject(plan, airshipv1.Scheme)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
doc, err := bundle.SelectOne(selector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := doc.ToAPIObject(plan, airshipv1.Scheme); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result := make(map[string][]string)
|
|
for _, phaseGroup := range plan.PhaseGroups {
|
|
phases := make([]string, len(phaseGroup.Phases))
|
|
for i, phase := range phaseGroup.Phases {
|
|
phases[i] = phase.Name
|
|
}
|
|
result[phaseGroup.Name] = phases
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
func handleError(err error, ch chan events.Event) {
|
|
ch <- events.Event{
|
|
Type: events.ErrorType,
|
|
ErrorEvent: events.ErrorEvent{
|
|
Error: err,
|
|
},
|
|
}
|
|
close(ch)
|
|
}
|