This repository has been archived by the owner on Apr 25, 2021. It is now read-only.
forked from themuvarov/kops
/
task.go
109 lines (87 loc) · 2.87 KB
/
task.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
/*
Copyright 2016 The Kubernetes 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 fi
import (
"fmt"
"reflect"
"strings"
"github.com/golang/glog"
)
type Task interface {
Run(*Context) error
}
// TaskAsString renders the task for debug output
// TODO: Use reflection to make this cleaner: don't recurse into tasks - print their names instead
// also print resources in a cleaner way (use the resource source information?)
func TaskAsString(t Task) string {
return fmt.Sprintf("%T %s", t, DebugAsJsonString(t))
}
type HasCheckExisting interface {
CheckExisting(c *Context) bool
}
// ModelBuilder allows for plugins that configure an aspect of the model, based on the configuration
type ModelBuilder interface {
Build(context *ModelBuilderContext) error
}
// ModelBuilderContext is a context object that holds state we want to pass to ModelBuilder
type ModelBuilderContext struct {
Tasks map[string]Task
}
func (c *ModelBuilderContext) AddTask(task Task) {
key := buildTaskKey(task)
existing, found := c.Tasks[key]
if found {
glog.Fatalf("found duplicate tasks with name %q: %v and %v", key, task, existing)
}
c.Tasks[key] = task
}
// EnsureTask ensures that the specified task is configured.
// It adds the task if it does not already exist.
// If it does exist, it verifies that the existing task reflect.DeepEqual the new task,
// if they are different an error is returned.
func (c *ModelBuilderContext) EnsureTask(task Task) error {
key := buildTaskKey(task)
existing, found := c.Tasks[key]
if found {
if reflect.DeepEqual(task, existing) {
glog.V(8).Infof("EnsureTask ignoring identical ")
return nil
} else {
glog.Warningf("EnsureTask found task mismatch for %q", key)
glog.Warningf("\tExisting: %v", existing)
glog.Warningf("\tNew: %v", task)
return fmt.Errorf("cannot add different task with same key %q", key)
}
}
c.Tasks[key] = task
return nil
}
func buildTaskKey(task Task) string {
hasName, ok := task.(HasName)
if !ok {
glog.Fatalf("task %T does not implement HasName", task)
}
name := StringValue(hasName.GetName())
if name == "" {
glog.Fatalf("task %T (%v) did not have a Name", task, task)
}
typeName := TypeNameForTask(task)
key := typeName + "/" + name
return key
}
func TypeNameForTask(task interface{}) string {
typeName := fmt.Sprintf("%T", task)
lastDot := strings.LastIndex(typeName, ".")
typeName = typeName[lastDot+1:]
return typeName
}