-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrelation.go
288 lines (268 loc) · 8.59 KB
/
relation.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
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
// Copyright 2012, 2013 Canonical Ltd.
// Licensed under the AGPLv3, see LICENCE file for details.
package state
import (
stderrors "errors"
"fmt"
"sort"
"strconv"
"strings"
"github.com/juju/charm"
"github.com/juju/errors"
"github.com/juju/names"
jujutxn "github.com/juju/txn"
"labix.org/v2/mgo"
"labix.org/v2/mgo/bson"
"labix.org/v2/mgo/txn"
)
// relationKey returns a string describing the relation defined by
// endpoints, for use in various contexts (including error messages).
func relationKey(endpoints []Endpoint) string {
eps := epSlice{}
for _, ep := range endpoints {
eps = append(eps, ep)
}
sort.Sort(eps)
names := []string{}
for _, ep := range eps {
names = append(names, ep.String())
}
return strings.Join(names, " ")
}
// relationDoc is the internal representation of a Relation in MongoDB.
// Note the correspondence with RelationInfo in state/api/params.
type relationDoc struct {
Key string `bson:"_id"`
Id int
Endpoints []Endpoint
Life Life
UnitCount int
}
// Relation represents a relation between one or two service endpoints.
type Relation struct {
st *State
doc relationDoc
}
func newRelation(st *State, doc *relationDoc) *Relation {
return &Relation{
st: st,
doc: *doc,
}
}
func (r *Relation) String() string {
return r.doc.Key
}
// Tag returns a name identifying the relation.
func (r *Relation) Tag() names.Tag {
return names.NewRelationTag(r.doc.Key)
}
// Refresh refreshes the contents of the relation from the underlying
// state. It returns an error that satisfies errors.IsNotFound if the
// relation has been removed.
func (r *Relation) Refresh() error {
doc := relationDoc{}
err := r.st.relations.FindId(r.doc.Key).One(&doc)
if err == mgo.ErrNotFound {
return errors.NotFoundf("relation %v", r)
}
if err != nil {
return fmt.Errorf("cannot refresh relation %v: %v", r, err)
}
if r.doc.Id != doc.Id {
// The relation has been destroyed and recreated. This is *not* the
// same relation; if we pretend it is, we run the risk of violating
// the lifecycle-only-advances guarantee.
return errors.NotFoundf("relation %v", r)
}
r.doc = doc
return nil
}
// Life returns the relation's current life state.
func (r *Relation) Life() Life {
return r.doc.Life
}
// Destroy ensures that the relation will be removed at some point; if no units
// are currently in scope, it will be removed immediately.
func (r *Relation) Destroy() (err error) {
defer errors.Maskf(&err, "cannot destroy relation %q", r)
if len(r.doc.Endpoints) == 1 && r.doc.Endpoints[0].Role == charm.RolePeer {
return fmt.Errorf("is a peer relation")
}
defer func() {
if err == nil {
// This is a white lie; the document might actually be removed.
r.doc.Life = Dying
}
}()
rel := &Relation{r.st, r.doc}
// In this context, aborted transactions indicate that the number of units
// in scope have changed between 0 and not-0. The chances of 5 successive
// attempts each hitting this change -- which is itself an unlikely one --
// are considered to be extremely small.
buildTxn := func(attempt int) ([]txn.Op, error) {
if attempt > 0 {
if err := rel.Refresh(); errors.IsNotFound(err) {
return []txn.Op{}, nil
} else if err != nil {
return nil, err
}
}
ops, _, err := rel.destroyOps("")
if err == errAlreadyDying {
return nil, jujutxn.ErrNoOperations
} else if err != nil {
return nil, err
}
return ops, nil
}
return rel.st.run(buildTxn)
}
var errAlreadyDying = stderrors.New("entity is already dying and cannot be destroyed")
// destroyOps returns the operations necessary to destroy the relation, and
// whether those operations will lead to the relation's removal. These
// operations may include changes to the relation's services; however, if
// ignoreService is not empty, no operations modifying that service will
// be generated.
func (r *Relation) destroyOps(ignoreService string) (ops []txn.Op, isRemove bool, err error) {
if r.doc.Life != Alive {
return nil, false, errAlreadyDying
}
if r.doc.UnitCount == 0 {
removeOps, err := r.removeOps(ignoreService, nil)
if err != nil {
return nil, false, err
}
return removeOps, true, nil
}
return []txn.Op{{
C: r.st.relations.Name,
Id: r.doc.Key,
Assert: bson.D{{"life", Alive}, {"unitcount", bson.D{{"$gt", 0}}}},
Update: bson.D{{"$set", bson.D{{"life", Dying}}}},
}}, false, nil
}
// removeOps returns the operations necessary to remove the relation. If
// ignoreService is not empty, no operations affecting that service will be
// included; if departingUnit is not nil, this implies that the relation's
// services may be Dying and otherwise unreferenced, and may thus require
// removal themselves.
func (r *Relation) removeOps(ignoreService string, departingUnit *Unit) ([]txn.Op, error) {
relOp := txn.Op{
C: r.st.relations.Name,
Id: r.doc.Key,
Remove: true,
}
if departingUnit != nil {
relOp.Assert = bson.D{{"life", Dying}, {"unitcount", 1}}
} else {
relOp.Assert = bson.D{{"life", Alive}, {"unitcount", 0}}
}
ops := []txn.Op{relOp}
for _, ep := range r.doc.Endpoints {
if ep.ServiceName == ignoreService {
continue
}
var asserts bson.D
hasRelation := bson.D{{"relationcount", bson.D{{"$gt", 0}}}}
if departingUnit == nil {
// We're constructing a destroy operation, either of the relation
// or one of its services, and can therefore be assured that both
// services are Alive.
asserts = append(hasRelation, isAliveDoc...)
} else if ep.ServiceName == departingUnit.ServiceName() {
// This service must have at least one unit -- the one that's
// departing the relation -- so it cannot be ready for removal.
cannotDieYet := bson.D{{"unitcount", bson.D{{"$gt", 0}}}}
asserts = append(hasRelation, cannotDieYet...)
} else {
// This service may require immediate removal.
svc := &Service{st: r.st}
hasLastRef := bson.D{{"life", Dying}, {"unitcount", 0}, {"relationcount", 1}}
removable := append(bson.D{{"_id", ep.ServiceName}}, hasLastRef...)
if err := r.st.services.Find(removable).One(&svc.doc); err == nil {
ops = append(ops, svc.removeOps(hasLastRef)...)
continue
} else if err != mgo.ErrNotFound {
return nil, err
}
// If not, we must check that this is still the case when the
// transaction is applied.
asserts = bson.D{{"$or", []bson.D{
{{"life", Alive}},
{{"unitcount", bson.D{{"$gt", 0}}}},
{{"relationcount", bson.D{{"$gt", 1}}}},
}}}
}
ops = append(ops, txn.Op{
C: r.st.services.Name,
Id: ep.ServiceName,
Assert: asserts,
Update: bson.D{{"$inc", bson.D{{"relationcount", -1}}}},
})
}
cleanupOp := r.st.newCleanupOp(cleanupRelationSettings, fmt.Sprintf("r#%d#", r.Id()))
return append(ops, cleanupOp), nil
}
// Id returns the integer internal relation key. This is exposed
// because the unit agent needs to expose a value derived from this
// (as JUJU_RELATION_ID) to allow relation hooks to differentiate
// between relations with different services.
func (r *Relation) Id() int {
return r.doc.Id
}
// Endpoint returns the endpoint of the relation for the named service.
// If the service is not part of the relation, an error will be returned.
func (r *Relation) Endpoint(serviceName string) (Endpoint, error) {
for _, ep := range r.doc.Endpoints {
if ep.ServiceName == serviceName {
return ep, nil
}
}
return Endpoint{}, fmt.Errorf("service %q is not a member of %q", serviceName, r)
}
// Endpoints returns the endpoints for the relation.
func (r *Relation) Endpoints() []Endpoint {
return r.doc.Endpoints
}
// RelatedEndpoints returns the endpoints of the relation r with which
// units of the named service will establish relations. If the service
// is not part of the relation r, an error will be returned.
func (r *Relation) RelatedEndpoints(serviceName string) ([]Endpoint, error) {
local, err := r.Endpoint(serviceName)
if err != nil {
return nil, err
}
role := counterpartRole(local.Role)
var eps []Endpoint
for _, ep := range r.doc.Endpoints {
if ep.Role == role {
eps = append(eps, ep)
}
}
if eps == nil {
return nil, fmt.Errorf("no endpoints of %q relate to service %q", r, serviceName)
}
return eps, nil
}
// Unit returns a RelationUnit for the supplied unit.
func (r *Relation) Unit(u *Unit) (*RelationUnit, error) {
ep, err := r.Endpoint(u.doc.Service)
if err != nil {
return nil, err
}
scope := []string{"r", strconv.Itoa(r.doc.Id)}
if ep.Scope == charm.ScopeContainer {
container := u.doc.Principal
if container == "" {
container = u.doc.Name
}
scope = append(scope, container)
}
return &RelationUnit{
st: r.st,
relation: r,
unit: u,
endpoint: ep,
scope: strings.Join(scope, "#"),
}, nil
}