LEFT | RIGHT |
(no file at all) | |
1 // Copyright 2012, 2013 Canonical Ltd. | 1 // Copyright 2012, 2013 Canonical Ltd. |
2 // Licensed under the AGPLv3, see LICENCE file for details. | 2 // Licensed under the AGPLv3, see LICENCE file for details. |
3 | 3 |
4 package state | 4 package state |
5 | 5 |
6 import ( | 6 import ( |
7 stderrors "errors" | 7 stderrors "errors" |
8 "fmt" | 8 "fmt" |
9 "labix.org/v2/mgo" | 9 "labix.org/v2/mgo" |
10 "labix.org/v2/mgo/bson" | 10 "labix.org/v2/mgo/bson" |
(...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
52 st: st, | 52 st: st, |
53 } | 53 } |
54 return svc | 54 return svc |
55 } | 55 } |
56 | 56 |
57 // Name returns the service name. | 57 // Name returns the service name. |
58 func (s *Service) Name() string { | 58 func (s *Service) Name() string { |
59 return s.doc.Name | 59 return s.doc.Name |
60 } | 60 } |
61 | 61 |
62 // TAg returns a name identifying the service that is safe to use | 62 // Tag returns a name identifying the service that is safe to use |
63 // as a file name. The returned name will be different from other | 63 // as a file name. The returned name will be different from other |
64 // TAg values returned by any other entities from the same state. | 64 // Tag values returned by any other entities from the same state. |
65 func (s *Service) Tag() string { | 65 func (s *Service) Tag() string { |
66 return "service-" + s.Name() | 66 return "service-" + s.Name() |
67 } | 67 } |
68 | 68 |
69 // serviceGlobalKey returns the global database key for the service | 69 // serviceGlobalKey returns the global database key for the service |
70 // with the given name. | 70 // with the given name. |
71 func serviceGlobalKey(svcName string) string { | 71 func serviceGlobalKey(svcName string) string { |
72 return "s#" + svcName | 72 return "s#" + svcName |
73 } | 73 } |
74 | 74 |
(...skipping 25 matching lines...) Expand all Loading... |
100 func (s *Service) Destroy() (err error) { | 100 func (s *Service) Destroy() (err error) { |
101 defer utils.ErrorContextf(&err, "cannot destroy service %q", s) | 101 defer utils.ErrorContextf(&err, "cannot destroy service %q", s) |
102 defer func() { | 102 defer func() { |
103 if err == nil { | 103 if err == nil { |
104 // This is a white lie; the document might actually be r
emoved. | 104 // This is a white lie; the document might actually be r
emoved. |
105 s.doc.Life = Dying | 105 s.doc.Life = Dying |
106 } | 106 } |
107 }() | 107 }() |
108 svc := &Service{st: s.st, doc: s.doc} | 108 svc := &Service{st: s.st, doc: s.doc} |
109 for i := 0; i < 5; i++ { | 109 for i := 0; i < 5; i++ { |
110 » » ops, err := svc.destroyOps() | 110 » » switch ops, err := svc.destroyOps(); err { |
111 » » switch { | 111 » » case errRefresh: |
112 » » case err == errRefresh: | 112 » » case errAlreadyDying: |
113 » » case err == errAlreadyDying: | |
114 return nil | 113 return nil |
115 » » case err != nil: | 114 » » case nil: |
116 » » » return err | 115 » » » if err := svc.st.runTransaction(ops); err != txn.ErrAbor
ted { |
117 » » default: | |
118 » » » if err := svc.st.runner.Run(ops, "", nil); err != txn.Er
rAborted { | |
119 return err | 116 return err |
120 } | 117 } |
| 118 default: |
| 119 return err |
121 } | 120 } |
122 if err := svc.Refresh(); errors.IsNotFoundError(err) { | 121 if err := svc.Refresh(); errors.IsNotFoundError(err) { |
123 return nil | 122 return nil |
124 } else if err != nil { | 123 } else if err != nil { |
125 return err | 124 return err |
126 } | 125 } |
127 } | 126 } |
128 return ErrExcessiveContention | 127 return ErrExcessiveContention |
129 } | 128 } |
130 | 129 |
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
178 // will be caught by virtue of being a remove. | 177 // will be caught by virtue of being a remove. |
179 notLastRefs := D{ | 178 notLastRefs := D{ |
180 {"life", Alive}, | 179 {"life", Alive}, |
181 {"relationcount", s.doc.RelationCount}, | 180 {"relationcount", s.doc.RelationCount}, |
182 } | 181 } |
183 // With respect to unit count, a changing value doesn't matter, so long | 182 // With respect to unit count, a changing value doesn't matter, so long |
184 // as the count's equality with zero does not change, because all we car
e | 183 // as the count's equality with zero does not change, because all we car
e |
185 // about is that *some* unit is, or is not, keeping the service from | 184 // about is that *some* unit is, or is not, keeping the service from |
186 // being removed: the difference between 1 unit and 1000 is irrelevant. | 185 // being removed: the difference between 1 unit and 1000 is irrelevant. |
187 if s.doc.UnitCount > 0 { | 186 if s.doc.UnitCount > 0 { |
| 187 ops = append(ops, s.st.newCleanupOp("units", s.doc.Name+"/")) |
188 notLastRefs = append(notLastRefs, D{{"unitcount", D{{"$gt", 0}}}
}...) | 188 notLastRefs = append(notLastRefs, D{{"unitcount", D{{"$gt", 0}}}
}...) |
189 } else { | 189 } else { |
190 notLastRefs = append(notLastRefs, D{{"unitcount", 0}}...) | 190 notLastRefs = append(notLastRefs, D{{"unitcount", 0}}...) |
191 } | 191 } |
192 update := D{{"$set", D{{"life", Dying}}}} | 192 update := D{{"$set", D{{"life", Dying}}}} |
193 if removeCount != 0 { | 193 if removeCount != 0 { |
194 decref := D{{"$inc", D{{"relationcount", -removeCount}}}} | 194 decref := D{{"$inc", D{{"relationcount", -removeCount}}}} |
195 update = append(update, decref...) | 195 update = append(update, decref...) |
196 } | 196 } |
197 return append(ops, txn.Op{ | 197 return append(ops, txn.Op{ |
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
242 return s.setExposed(false) | 242 return s.setExposed(false) |
243 } | 243 } |
244 | 244 |
245 func (s *Service) setExposed(exposed bool) (err error) { | 245 func (s *Service) setExposed(exposed bool) (err error) { |
246 ops := []txn.Op{{ | 246 ops := []txn.Op{{ |
247 C: s.st.services.Name, | 247 C: s.st.services.Name, |
248 Id: s.doc.Name, | 248 Id: s.doc.Name, |
249 Assert: isAliveDoc, | 249 Assert: isAliveDoc, |
250 Update: D{{"$set", D{{"exposed", exposed}}}}, | 250 Update: D{{"$set", D{{"exposed", exposed}}}}, |
251 }} | 251 }} |
252 » if err := s.st.runner.Run(ops, "", nil); err != nil { | 252 » if err := s.st.runTransaction(ops); err != nil { |
253 return fmt.Errorf("cannot set exposed flag for service %q to %v:
%v", s, exposed, onAbort(err, errNotAlive)) | 253 return fmt.Errorf("cannot set exposed flag for service %q to %v:
%v", s, exposed, onAbort(err, errNotAlive)) |
254 } | 254 } |
255 s.doc.Exposed = exposed | 255 s.doc.Exposed = exposed |
256 return nil | 256 return nil |
257 } | 257 } |
258 | 258 |
259 // Charm returns the service's charm and whether units should upgrade to that | 259 // Charm returns the service's charm and whether units should upgrade to that |
260 // charm even if they are in an error state. | 260 // charm even if they are in an error state. |
261 func (s *Service) Charm() (ch *Charm, force bool, err error) { | 261 func (s *Service) Charm() (ch *Charm, force bool, err error) { |
262 ch, err = s.st.Charm(s.doc.CharmURL) | 262 ch, err = s.st.Charm(s.doc.CharmURL) |
(...skipping 214 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
477 Update: D{{"$set", D{{"forcecharm", force}}}}, | 477 Update: D{{"$set", D{{"forcecharm", force}}}}, |
478 }} | 478 }} |
479 } else { | 479 } else { |
480 // Change the charm URL. | 480 // Change the charm URL. |
481 ops, err = s.changeCharmOps(ch, force) | 481 ops, err = s.changeCharmOps(ch, force) |
482 if err != nil { | 482 if err != nil { |
483 return err | 483 return err |
484 } | 484 } |
485 } | 485 } |
486 | 486 |
487 » » if err := s.st.runner.Run(ops, "", nil); err == nil { | 487 » » if err := s.st.runTransaction(ops); err == nil { |
488 s.doc.CharmURL = ch.URL() | 488 s.doc.CharmURL = ch.URL() |
489 s.doc.ForceCharm = force | 489 s.doc.ForceCharm = force |
490 return nil | 490 return nil |
491 } else if err != txn.ErrAborted { | 491 } else if err != txn.ErrAborted { |
492 return err | 492 return err |
493 } | 493 } |
494 | 494 |
495 // If the service is not alive, fail out immediately; otherwise, | 495 // If the service is not alive, fail out immediately; otherwise, |
496 // data changed underneath us, so retry. | 496 // data changed underneath us, so retry. |
497 if alive, err := isAlive(s.st.services, s.doc.Name); err != nil
{ | 497 if alive, err := isAlive(s.st.services, s.doc.Name); err != nil
{ |
(...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
598 return name, ops, nil | 598 return name, ops, nil |
599 } | 599 } |
600 | 600 |
601 // AddUnit adds a new principal unit to the service. | 601 // AddUnit adds a new principal unit to the service. |
602 func (s *Service) AddUnit() (unit *Unit, err error) { | 602 func (s *Service) AddUnit() (unit *Unit, err error) { |
603 defer utils.ErrorContextf(&err, "cannot add unit to service %q", s) | 603 defer utils.ErrorContextf(&err, "cannot add unit to service %q", s) |
604 name, ops, err := s.addUnitOps("") | 604 name, ops, err := s.addUnitOps("") |
605 if err != nil { | 605 if err != nil { |
606 return nil, err | 606 return nil, err |
607 } | 607 } |
608 » if err := s.st.runner.Run(ops, "", nil); err == txn.ErrAborted { | 608 » if err := s.st.runTransaction(ops); err == txn.ErrAborted { |
609 if alive, err := isAlive(s.st.services, s.doc.Name); err != nil
{ | 609 if alive, err := isAlive(s.st.services, s.doc.Name); err != nil
{ |
610 return nil, err | 610 return nil, err |
611 } else if !alive { | 611 } else if !alive { |
612 return nil, fmt.Errorf("service is not alive") | 612 return nil, fmt.Errorf("service is not alive") |
613 } | 613 } |
614 return nil, fmt.Errorf("inconsistent state") | 614 return nil, fmt.Errorf("inconsistent state") |
615 } else if err != nil { | 615 } else if err != nil { |
616 return nil, err | 616 return nil, err |
617 } | 617 } |
618 return s.Unit(name) | 618 return s.Unit(name) |
(...skipping 13 matching lines...) Expand all Loading... |
632 Update: D{{"$pull", D{{"subordinates", u.doc.Name}}}}, | 632 Update: D{{"$pull", D{{"subordinates", u.doc.Name}}}}, |
633 }) | 633 }) |
634 } else if u.doc.MachineId != "" { | 634 } else if u.doc.MachineId != "" { |
635 ops = append(ops, txn.Op{ | 635 ops = append(ops, txn.Op{ |
636 C: s.st.machines.Name, | 636 C: s.st.machines.Name, |
637 Id: u.doc.MachineId, | 637 Id: u.doc.MachineId, |
638 Assert: txn.DocExists, | 638 Assert: txn.DocExists, |
639 Update: D{{"$pull", D{{"principals", u.doc.Name}}}}, | 639 Update: D{{"$pull", D{{"principals", u.doc.Name}}}}, |
640 }) | 640 }) |
641 } | 641 } |
| 642 observedFieldsMatch := D{ |
| 643 {"charmurl", u.doc.CharmURL}, |
| 644 {"machineid", u.doc.MachineId}, |
| 645 } |
642 ops = append(ops, txn.Op{ | 646 ops = append(ops, txn.Op{ |
643 C: s.st.units.Name, | 647 C: s.st.units.Name, |
644 Id: u.doc.Name, | 648 Id: u.doc.Name, |
645 » » Assert: asserts, | 649 » » Assert: append(observedFieldsMatch, asserts...), |
646 Remove: true, | 650 Remove: true, |
647 }, | 651 }, |
648 removeConstraintsOp(s.st, u.globalKey()), | 652 removeConstraintsOp(s.st, u.globalKey()), |
649 removeStatusOp(s.st, u.globalKey()), | 653 removeStatusOp(s.st, u.globalKey()), |
650 annotationRemoveOp(s.st, u.globalKey()), | 654 annotationRemoveOp(s.st, u.globalKey()), |
651 ) | 655 ) |
652 if u.doc.CharmURL != nil { | 656 if u.doc.CharmURL != nil { |
653 decOps, err := settingsDecRefOps(s.st, s.doc.Name, u.doc.CharmUR
L) | 657 decOps, err := settingsDecRefOps(s.st, s.doc.Name, u.doc.CharmUR
L) |
654 » » if err != nil { | 658 » » if errors.IsNotFoundError(err) { |
| 659 » » » return nil, errRefresh |
| 660 » » } else if err != nil { |
655 return nil, err | 661 return nil, err |
656 } | 662 } |
657 ops = append(ops, decOps...) | 663 ops = append(ops, decOps...) |
658 } | 664 } |
659 if s.doc.Life == Dying && s.doc.RelationCount == 0 && s.doc.UnitCount ==
1 { | 665 if s.doc.Life == Dying && s.doc.RelationCount == 0 && s.doc.UnitCount ==
1 { |
660 hasLastRef := D{{"life", Dying}, {"relationcount", 0}, {"unitcou
nt", 1}} | 666 hasLastRef := D{{"life", Dying}, {"relationcount", 0}, {"unitcou
nt", 1}} |
661 return append(ops, s.removeOps(hasLastRef)...), nil | 667 return append(ops, s.removeOps(hasLastRef)...), nil |
662 } | 668 } |
663 svcOp := txn.Op{ | 669 svcOp := txn.Op{ |
664 C: s.st.services.Name, | 670 C: s.st.services.Name, |
(...skipping 114 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
779 return errNotAlive | 785 return errNotAlive |
780 } | 786 } |
781 ops := []txn.Op{ | 787 ops := []txn.Op{ |
782 { | 788 { |
783 C: s.st.services.Name, | 789 C: s.st.services.Name, |
784 Id: s.doc.Name, | 790 Id: s.doc.Name, |
785 Assert: isAliveDoc, | 791 Assert: isAliveDoc, |
786 }, | 792 }, |
787 setConstraintsOp(s.st, s.globalKey(), cons), | 793 setConstraintsOp(s.st, s.globalKey(), cons), |
788 } | 794 } |
789 » return onAbort(s.st.runner.Run(ops, "", nil), errNotAlive) | 795 » return onAbort(s.st.runTransaction(ops), errNotAlive) |
790 } | 796 } |
791 | 797 |
792 // settingsIncRefOp returns an operation that increments the ref count | 798 // settingsIncRefOp returns an operation that increments the ref count |
793 // of the service settings identified by serviceName and curl. If | 799 // of the service settings identified by serviceName and curl. If |
794 // canCreate is false, a missing document will be treated as an error; | 800 // canCreate is false, a missing document will be treated as an error; |
795 // otherwise, it will be created with a ref count of 1. | 801 // otherwise, it will be created with a ref count of 1. |
796 func settingsIncRefOp(st *State, serviceName string, curl *charm.URL, canCreate
bool) (txn.Op, error) { | 802 func settingsIncRefOp(st *State, serviceName string, curl *charm.URL, canCreate
bool) (txn.Op, error) { |
797 key := serviceSettingsKey(serviceName, curl) | 803 key := serviceSettingsKey(serviceName, curl) |
798 if count, err := st.settingsrefs.FindId(key).Count(); err != nil { | 804 if count, err := st.settingsrefs.FindId(key).Count(); err != nil { |
799 return txn.Op{}, err | 805 return txn.Op{}, err |
(...skipping 16 matching lines...) Expand all Loading... |
816 }, nil | 822 }, nil |
817 } | 823 } |
818 | 824 |
819 // settingsDecRefOps returns a list of operations that decrement the | 825 // settingsDecRefOps returns a list of operations that decrement the |
820 // ref count of the service settings identified by serviceName and | 826 // ref count of the service settings identified by serviceName and |
821 // curl. If the ref count is set to zero, the appropriate setting and | 827 // curl. If the ref count is set to zero, the appropriate setting and |
822 // ref count documents will both be deleted. | 828 // ref count documents will both be deleted. |
823 func settingsDecRefOps(st *State, serviceName string, curl *charm.URL) ([]txn.Op
, error) { | 829 func settingsDecRefOps(st *State, serviceName string, curl *charm.URL) ([]txn.Op
, error) { |
824 key := serviceSettingsKey(serviceName, curl) | 830 key := serviceSettingsKey(serviceName, curl) |
825 var doc settingsRefsDoc | 831 var doc settingsRefsDoc |
826 » if err := st.settingsrefs.FindId(key).One(&doc); err != nil { | 832 » if err := st.settingsrefs.FindId(key).One(&doc); err == mgo.ErrNotFound
{ |
| 833 » » return nil, errors.NotFoundf("service %q settings for charm %q",
serviceName, curl) |
| 834 » } else if err != nil { |
827 return nil, err | 835 return nil, err |
828 } | 836 } |
829 if doc.RefCount == 1 { | 837 if doc.RefCount == 1 { |
830 return []txn.Op{{ | 838 return []txn.Op{{ |
831 C: st.settingsrefs.Name, | 839 C: st.settingsrefs.Name, |
832 Id: key, | 840 Id: key, |
833 Assert: D{{"refcount", 1}}, | 841 Assert: D{{"refcount", 1}}, |
834 Remove: true, | 842 Remove: true, |
835 }, { | 843 }, { |
836 C: st.settings.Name, | 844 C: st.settings.Name, |
(...skipping 20 matching lines...) Expand all Loading... |
857 // | 865 // |
858 // Note: We're not using the settingsDoc for this because changing | 866 // Note: We're not using the settingsDoc for this because changing |
859 // just the ref count is not considered a change worth reporting | 867 // just the ref count is not considered a change worth reporting |
860 // to watchers and firing config-changed hooks. | 868 // to watchers and firing config-changed hooks. |
861 // | 869 // |
862 // There is and implicit _id field here, which mongo creates, which is | 870 // There is and implicit _id field here, which mongo creates, which is |
863 // always the same as the settingsDoc's id. | 871 // always the same as the settingsDoc's id. |
864 type settingsRefsDoc struct { | 872 type settingsRefsDoc struct { |
865 RefCount int | 873 RefCount int |
866 } | 874 } |
LEFT | RIGHT |