-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpolicy.go
207 lines (187 loc) · 6.99 KB
/
policy.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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
// Copyright 2014 Canonical Ltd.
// Licensed under the AGPLv3, see LICENCE file for details.
package state
import (
"fmt"
"github.com/juju/errors"
"github.com/juju/juju/constraints"
"github.com/juju/juju/environs/config"
"github.com/juju/juju/instance"
)
// Policy is an interface provided to State that may
// be consulted by State to validate or modify the
// behaviour of certain operations.
//
// If a Policy implementation does not implement one
// of the methods, it must return an error that
// satisfies errors.IsNotImplemented, and will thus
// be ignored. Any other error will cause an error
// in the use of the policy.
type Policy interface {
// Prechecker takes a *config.Config and returns a Prechecker or an error.
Prechecker(*config.Config) (Prechecker, error)
// ConfigValidator takes a provider type name and returns a ConfigValidator
// or an error.
ConfigValidator(providerType string) (ConfigValidator, error)
// EnvironCapability takes a *config.Config and returns an EnvironCapability
// or an error.
EnvironCapability(*config.Config) (EnvironCapability, error)
// ConstraintsValidator takes a *config.Config and returns a
// constraints.Validator or an error.
ConstraintsValidator(*config.Config) (constraints.Validator, error)
// InstanceDistributor takes a *config.Config and returns an
// InstanceDistributor or an error.
InstanceDistributor(*config.Config) (InstanceDistributor, error)
}
// Prechecker is a policy interface that is provided to State
// to perform pre-flight checking of instance creation.
type Prechecker interface {
// PrecheckInstance performs a preflight check on the specified
// series and constraints, ensuring that they are possibly valid for
// creating an instance in this environment.
//
// PrecheckInstance is best effort, and not guaranteed to eliminate
// all invalid parameters. If PrecheckInstance returns nil, it is not
// guaranteed that the constraints are valid; if a non-nil error is
// returned, then the constraints are definitely invalid.
PrecheckInstance(series string, cons constraints.Value, placement string) error
}
// ConfigValidator is a policy interface that is provided to State
// to check validity of new configuration attributes before applying them to state.
type ConfigValidator interface {
Validate(cfg, old *config.Config) (valid *config.Config, err error)
}
// EnvironCapability implements access to metadata about the capabilities
// of an environment.
type EnvironCapability interface {
// SupportedArchitectures returns the image architectures which can
// be hosted by this environment.
SupportedArchitectures() ([]string, error)
// SupportsUnitAssignment returns an error which, if non-nil, indicates
// that the environment does not support unit placement. If the environment
// does not support unit placement, then machines may not be created
// without units, and units cannot be placed explcitly.
SupportsUnitPlacement() error
}
// precheckInstance calls the state's assigned policy, if non-nil, to obtain
// a Prechecker, and calls PrecheckInstance if a non-nil Prechecker is returned.
func (st *State) precheckInstance(series string, cons constraints.Value, placement string) error {
if st.policy == nil {
return nil
}
cfg, err := st.EnvironConfig()
if err != nil {
return err
}
prechecker, err := st.policy.Prechecker(cfg)
if errors.IsNotImplemented(err) {
return nil
} else if err != nil {
return err
}
if prechecker == nil {
return fmt.Errorf("policy returned nil prechecker without an error")
}
return prechecker.PrecheckInstance(series, cons, placement)
}
func (st *State) constraintsValidator() (constraints.Validator, error) {
// Default behaviour is to simply use a standard validator with
// no environment specific behaviour built in.
defaultValidator := constraints.NewValidator()
if st.policy == nil {
return defaultValidator, nil
}
cfg, err := st.EnvironConfig()
if err != nil {
return nil, err
}
validator, err := st.policy.ConstraintsValidator(cfg)
if errors.IsNotImplemented(err) {
return defaultValidator, nil
} else if err != nil {
return nil, err
}
if validator == nil {
return nil, fmt.Errorf("policy returned nil constraints validator without an error")
}
return validator, nil
}
// resolveConstraints combines the given constraints with the environ constraints to get
// a constraints which will be used to create a new instance.
func (st *State) resolveConstraints(cons constraints.Value) (constraints.Value, error) {
validator, err := st.constraintsValidator()
if err != nil {
return constraints.Value{}, err
}
envCons, err := st.EnvironConstraints()
if err != nil {
return constraints.Value{}, err
}
return validator.Merge(envCons, cons)
}
// validateConstraints returns an error if the given constraints are not valid for the
// current environment, and also any unsupported attributes.
func (st *State) validateConstraints(cons constraints.Value) ([]string, error) {
validator, err := st.constraintsValidator()
if err != nil {
return nil, err
}
return validator.Validate(cons)
}
// validate calls the state's assigned policy, if non-nil, to obtain
// a ConfigValidator, and calls Validate if a non-nil ConfigValidator is
// returned.
func (st *State) validate(cfg, old *config.Config) (valid *config.Config, err error) {
if st.policy == nil {
return cfg, nil
}
configValidator, err := st.policy.ConfigValidator(cfg.Type())
if errors.IsNotImplemented(err) {
return cfg, nil
} else if err != nil {
return nil, err
}
if configValidator == nil {
return nil, fmt.Errorf("policy returned nil configValidator without an error")
}
return configValidator.Validate(cfg, old)
}
// supportsUnitPlacement calls the state's assigned policy, if non-nil,
// to obtain an EnvironCapability, and calls SupportsUnitPlacement if a
// non-nil EnvironCapability is returned.
func (st *State) supportsUnitPlacement() error {
if st.policy == nil {
return nil
}
cfg, err := st.EnvironConfig()
if err != nil {
return errors.Trace(err)
}
capability, err := st.policy.EnvironCapability(cfg)
if errors.IsNotImplemented(err) {
return nil
} else if err != nil {
return errors.Trace(err)
}
if capability == nil {
return fmt.Errorf("policy returned nil EnvironCapability without an error")
}
return capability.SupportsUnitPlacement()
}
// InstanceDistributor is a policy interface that is provided
// to State to perform distribution of units across instances
// for high availability.
type InstanceDistributor interface {
// DistributeInstance takes a set of clean, empty
// instances, and a distribution group, and returns
// the subset of candidates which the policy will
// allow entry into the distribution group.
//
// The AssignClean and AssignCleanEmpty unit
// assignment policies will attempt to assign a
// unit to each of the resulting instances until
// one is successful. If no instances can be assigned
// to (e.g. because of concurrent deployments), then
// a new machine will be allocated.
DistributeInstances(candidates, distributionGroup []instance.Id) ([]instance.Id, error)
}