forked from juju/juju
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconstraints.go
234 lines (205 loc) · 6.17 KB
/
constraints.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
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
// Copyright 2013 Canonical Ltd.
// Licensed under the AGPLv3, see LICENCE file for details.
package state
import (
"fmt"
"github.com/juju/errors"
"github.com/juju/mgo/v2"
"github.com/juju/mgo/v2/bson"
"github.com/juju/mgo/v2/txn"
"github.com/juju/juju/core/constraints"
"github.com/juju/juju/core/instance"
)
// constraintsDoc is the Mongo DB representation of a constraints.Value.
type constraintsDoc struct {
DocID string `bson:"_id,omitempty"`
ModelUUID string `bson:"model-uuid"`
Arch *string
CpuCores *uint64
CpuPower *uint64
Mem *uint64
RootDisk *uint64
RootDiskSource *string
InstanceRole *string
InstanceType *string
Container *instance.ContainerType
Tags *[]string
Spaces *[]string
VirtType *string
Zones *[]string
AllocatePublicIP *bool
}
func newConstraintsDoc(cons constraints.Value, id string) constraintsDoc {
result := constraintsDoc{
DocID: id,
Arch: cons.Arch,
CpuCores: cons.CpuCores,
CpuPower: cons.CpuPower,
Mem: cons.Mem,
RootDisk: cons.RootDisk,
RootDiskSource: cons.RootDiskSource,
InstanceRole: cons.InstanceRole,
InstanceType: cons.InstanceType,
Container: cons.Container,
Tags: cons.Tags,
Spaces: cons.Spaces,
VirtType: cons.VirtType,
Zones: cons.Zones,
AllocatePublicIP: cons.AllocatePublicIP,
}
return result
}
func (doc constraintsDoc) value() constraints.Value {
result := constraints.Value{
Arch: doc.Arch,
CpuCores: doc.CpuCores,
CpuPower: doc.CpuPower,
Mem: doc.Mem,
RootDisk: doc.RootDisk,
RootDiskSource: doc.RootDiskSource,
InstanceRole: doc.InstanceRole,
InstanceType: doc.InstanceType,
Container: doc.Container,
Tags: doc.Tags,
Spaces: doc.Spaces,
VirtType: doc.VirtType,
Zones: doc.Zones,
AllocatePublicIP: doc.AllocatePublicIP,
}
return result
}
// Constraints represents the state of a constraints with an ID.
type Constraints struct {
doc constraintsDoc
}
// ID returns the Mongo document ID for the constraints instance.
func (c *Constraints) ID() string {
return c.doc.DocID
}
// Value returns the constraints.Value represented by the Constraints'
// Mongo document.
func (c *Constraints) Value() constraints.Value {
return c.doc.value()
}
// ChangeSpaceNameOps returns the transaction operations required to rename
// a space used in these constraints.
func (c *Constraints) ChangeSpaceNameOps(from, to string) []txn.Op {
val := c.doc.value()
spaces := val.Spaces
if spaces == nil {
return nil
}
for i, space := range *spaces {
if space == from {
(*spaces)[i] = to
break
}
if space == "^"+from {
(*spaces)[i] = "^" + to
}
}
return []txn.Op{setConstraintsOp(c.ID(), val)}
}
// AllConstraints returns all constraints in the collection.
func (st *State) AllConstraints() ([]*Constraints, error) {
constraintsCollection, closer := st.db().GetCollection(constraintsC)
defer closer()
var docs []constraintsDoc
err := constraintsCollection.Find(nil).All(&docs)
cons := make([]*Constraints, len(docs))
for i, doc := range docs {
cons[i] = &Constraints{doc: doc}
}
return cons, err
}
// ConstraintsBySpaceName returns all Constraints that include a positive
// or negative space constraint for the input space name.
func (st *State) ConstraintsBySpaceName(spaceName string) ([]*Constraints, error) {
constraintsCollection, closer := st.db().GetCollection(constraintsC)
defer closer()
var docs []constraintsDoc
query := bson.D{{"$or", []bson.D{
{{"spaces", spaceName}},
{{"spaces", "^" + spaceName}},
}}}
err := constraintsCollection.Find(query).All(&docs)
cons := make([]*Constraints, len(docs))
for i, doc := range docs {
cons[i] = &Constraints{doc: doc}
}
return cons, err
}
func createConstraintsOp(id string, cons constraints.Value) txn.Op {
return txn.Op{
C: constraintsC,
Id: id,
Assert: txn.DocMissing,
Insert: newConstraintsDoc(cons, id),
}
}
func setConstraintsOp(id string, cons constraints.Value) txn.Op {
return txn.Op{
C: constraintsC,
Id: id,
Assert: txn.DocExists,
Update: bson.D{{"$set", newConstraintsDoc(cons, id)}},
}
}
func removeConstraintsOp(id string) txn.Op {
return txn.Op{
C: constraintsC,
Id: id,
Remove: true,
}
}
func readConstraints(mb modelBackend, id string) (constraints.Value, error) {
constraintsCollection, closer := mb.db().GetCollection(constraintsC)
defer closer()
var doc constraintsDoc
if err := constraintsCollection.FindId(id).One(&doc); err == mgo.ErrNotFound {
return constraints.Value{}, errors.NotFoundf("constraints")
} else if err != nil {
return constraints.Value{}, err
}
return doc.value(), nil
}
func writeConstraints(mb modelBackend, id string, cons constraints.Value) error {
ops := []txn.Op{setConstraintsOp(id, cons)}
if err := mb.db().RunTransaction(ops); err != nil {
return fmt.Errorf("cannot set constraints: %v", err)
}
return nil
}
// AllConstraints retrieves all the constraints in the model
// and provides a way to query based on machine or application.
func (m *Model) AllConstraints() (*ModelConstraints, error) {
coll, closer := m.st.db().GetCollection(constraintsC)
defer closer()
var docs []constraintsDoc
err := coll.Find(nil).All(&docs)
if err != nil {
return nil, errors.Annotate(err, "cannot get all constraints for model")
}
all := &ModelConstraints{
data: make(map[string]constraintsDoc),
}
for _, doc := range docs {
all.data[m.localID(doc.DocID)] = doc
}
return all, nil
}
// ModelConstraints represents all the constraints in a model
// keyed on global key.
type ModelConstraints struct {
data map[string]constraintsDoc
}
// Machine returns the constraints for the specified machine.
func (c *ModelConstraints) Machine(machineID string) constraints.Value {
doc, found := c.data[machineGlobalKey(machineID)]
if !found {
return constraints.Value{}
}
return doc.value()
}
// TODO: add methods for Model and Application constraints checks
// if desired. Not currently used in status calls.