Rietveld Code Review Tool
Help | Bug tracker | Discussion group | Source code | Sign in
(240)

Delta Between Two Patch Sets: state/service.go

Issue 9738043: cmd/jujud: do not change password
Left Patch Set: cmd/jujud: do not change password Created 11 years, 10 months ago
Right Patch Set: cmd/jujud: do not change password Created 11 years, 9 months ago
Left:
Right:
Use n/p to move between diff chunks; N/P to move between comments. Please Sign in to add in-line comments.
Jump to:
Right: Side by side diff | Download
« no previous file with change/comment | « state/relationunit_test.go ('k') | state/service_test.go » ('j') | no next file with change/comment »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
LEFTRIGHT
(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
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
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
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
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
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
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
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
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
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
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 }
LEFTRIGHT

Powered by Google App Engine
RSS Feeds Recent Issues | This issue
This is Rietveld f62528b