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

Delta Between Two Patch Sets: worker/notifyworker.go

Issue 12198044: worker: Introducing StringsWorker (Closed)
Left Patch Set: Created 11 years, 8 months ago
Right Patch Set: worker: Introducing StringsWorker Created 11 years, 8 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:
Left: Side by side diff | Download
Right: Side by side diff | Download
« no previous file with change/comment | « worker/machiner/machiner_test.go ('k') | worker/notifyworker_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
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 worker 4 package worker
5 5
6 import ( 6 import (
7 "launchpad.net/tomb" 7 "launchpad.net/tomb"
8 8
9 "launchpad.net/juju-core/state/api" 9 "launchpad.net/juju-core/state/api"
10 "launchpad.net/juju-core/state/watcher" 10 "launchpad.net/juju-core/state/watcher"
11 ) 11 )
12 12
13 // notifyWorker is the internal implementation of the NotifyWorker 13 // notifyWorker is the internal implementation of the Worker
14 // interface 14 // interface, using a NotifyWatcher for handling changes.
15 type notifyWorker struct { 15 type notifyWorker struct {
16
17 // Internal structure
18 tomb tomb.Tomb 16 tomb tomb.Tomb
19 17
20 // handler is what will handle when events are triggered 18 // handler is what will handle when events are triggered
21 handler NotifyWatchHandler 19 handler NotifyWatchHandler
22 20
23 » // closedHandler is set to watcher.MustErr, but that panic()s, so 21 » // mustErr is set to watcher.MustErr, but that panic()s, so
24 // we let the test suite override it. 22 // we let the test suite override it.
25 » closedHandler func(watcher.Errer) error 23 » mustErr func(watcher.Errer) error
rog 2013/08/01 11:18:47 can we just call this mustErr, as that represents
dimitern 2013/08/01 12:53:54 Done.
26 } 24 }
27
28 // NotifyWorker encapsulates the logic for a worker which is based on
29 // a NotifyWatcher. We do a bit of setup, and then spin waiting for
30 // the watcher to trigger or for us to be killed, and then teardown
31 // cleanly.
32 type NotifyWorker CommonWorker
rog 2013/08/01 11:18:47 I don't think we need this type. Let's just make N
dimitern 2013/08/01 12:53:54 Done.
33 25
34 // NotifyWatchHandler implements the business logic that is triggered 26 // NotifyWatchHandler implements the business logic that is triggered
35 // as part of watching a NotifyWatcher. 27 // as part of watching a NotifyWatcher.
36 type NotifyWatchHandler interface { 28 type NotifyWatchHandler interface {
37
38 // SetUp starts the handler, this should create the watcher we 29 // SetUp starts the handler, this should create the watcher we
39 // will be waiting on for more events. SetUp can return a Watcher 30 // will be waiting on for more events. SetUp can return a Watcher
40 » // even if there is an error, and NotifyWorker will make sure to 31 » // even if there is an error, and the notify Worker will make sure
41 » // stop the Watcher. 32 » // to stop the watcher.
42 SetUp() (api.NotifyWatcher, error) 33 SetUp() (api.NotifyWatcher, error)
43 34
44 // TearDown should cleanup any resources that are left around 35 // TearDown should cleanup any resources that are left around
45 TearDown() error 36 TearDown() error
46 37
47 // Handle is called when the Watcher has indicated there are 38 // Handle is called when the Watcher has indicated there are
48 // changes, do whatever work is necessary to process it 39 // changes, do whatever work is necessary to process it
49 Handle() error 40 Handle() error
50
51 // String is used when reporting. It is required because
52 // NotifyWatcher is wrapping the NotifyWatchHandler, but the
53 // NotifyWatchHandler is the interesting (identifying) logic.
54 String() string
rog 2013/08/01 11:18:47 I don't think we need a String method at all. It w
dimitern 2013/08/01 12:53:54 Done.
55 } 41 }
56 42
57 // NewNotifyWorker starts a new worker running the business logic from the 43 // NewNotifyWorker starts a new worker running the business logic from
58 // handler. The worker loop is started in another goroutine as a side effect of 44 // the handler. The worker loop is started in another goroutine as a
59 // calling this. 45 // side effect of calling this.
60 func NewNotifyWorker(handler NotifyWatchHandler) NotifyWorker { 46 func NewNotifyWorker(handler NotifyWatchHandler) Worker {
rog 2013/08/01 11:18:47 I think this should return a Worker.
dimitern 2013/08/01 12:53:54 Done.
61 nw := &notifyWorker{ 47 nw := &notifyWorker{
62 » » handler: handler, 48 » » handler: handler,
63 » » closedHandler: watcher.MustErr, 49 » » mustErr: watcher.MustErr,
64 } 50 }
65 go func() { 51 go func() {
66 defer nw.tomb.Done() 52 defer nw.tomb.Done()
67 nw.tomb.Kill(nw.loop()) 53 nw.tomb.Kill(nw.loop())
68 }() 54 }()
69 return nw 55 return nw
70 } 56 }
71 57
72 // Kill the loop with no-error 58 // Kill the loop with no-error
73 func (nw *notifyWorker) Kill() { 59 func (nw *notifyWorker) Kill() {
74 nw.tomb.Kill(nil) 60 nw.tomb.Kill(nil)
75 } 61 }
76 62
77 // Stop kils the worker and waits for it to exit
78 func (nw *notifyWorker) Stop() error {
rog 2013/08/01 11:18:47 I don't think we need to bother implementing this
dimitern 2013/08/01 12:53:54 Done. As discussed, worker.Stop(x) can be used ins
79 nw.tomb.Kill(nil)
80 return nw.tomb.Wait()
81 }
82
83 // Wait for the looping to finish 63 // Wait for the looping to finish
84 func (nw *notifyWorker) Wait() error { 64 func (nw *notifyWorker) Wait() error {
85 return nw.tomb.Wait() 65 return nw.tomb.Wait()
86 } 66 }
87 67
88 // String returns a nice description of this worker, taken from the underlying W atchHandler 68 type tearDowner interface {
89 func (nw *notifyWorker) String() string { 69 » TearDown() error
90 » return nw.handler.String()
91 } 70 }
92 71
93 func notifyHandlerTearDown(handler NotifyWatchHandler, t *tomb.Tomb) { 72 // propagateTearDown tears down the handler, but ensures any error is
rog 2013/08/01 11:18:47 how about this? func propagateTearDown(handler in
dimitern 2013/08/01 12:53:54 Done.
94 » // Tear down the handler, but ensure any error is propagated 73 // propagated through the tomb's Kill method.
mue 2013/08/01 10:35:24 Why the comment in the method?
dimitern 2013/08/01 12:53:54 Because it was describing what's being done, rathe
74 func propagateTearDown(handler tearDowner, t *tomb.Tomb) {
95 if err := handler.TearDown(); err != nil { 75 if err := handler.TearDown(); err != nil {
96 t.Kill(err) 76 t.Kill(err)
97 } 77 }
98 } 78 }
99 79
100 func (nw *notifyWorker) loop() error { 80 func (nw *notifyWorker) loop() error {
101 » var w api.NotifyWatcher 81 » w, err := nw.handler.SetUp()
102 » var err error 82 » if err != nil {
103 » defer notifyHandlerTearDown(nw.handler, &nw.tomb)
rog 2013/08/01 11:18:47 I'm not convinced that it's right to call TearDown
dimitern 2013/08/01 12:53:54 As I understand it, you propose to call this after
104 » if w, err = nw.handler.SetUp(); err != nil {
105 if w != nil { 83 if w != nil {
106 » » » // We don't bother to propogate an error, because we 84 » » » // We don't bother to propagate an error, because we
rog 2013/08/01 11:18:47 s/propo/propa/
dimitern 2013/08/01 12:53:54 Done.
107 // already have an error 85 // already have an error
108 w.Stop() 86 w.Stop()
109 } 87 }
110 return err 88 return err
111 } 89 }
90 defer propagateTearDown(nw.handler, &nw.tomb)
112 defer watcher.Stop(w, &nw.tomb) 91 defer watcher.Stop(w, &nw.tomb)
113 for { 92 for {
114 select { 93 select {
115 case <-nw.tomb.Dying(): 94 case <-nw.tomb.Dying():
116 return tomb.ErrDying 95 return tomb.ErrDying
117 case _, ok := <-w.Changes(): 96 case _, ok := <-w.Changes():
118 if !ok { 97 if !ok {
119 » » » » return nw.closedHandler(w) 98 » » » » return nw.mustErr(w)
120 } 99 }
121 if err := nw.handler.Handle(); err != nil { 100 if err := nw.handler.Handle(); err != nil {
122 return err 101 return err
123 } 102 }
124 } 103 }
125 } 104 }
126 } 105 }
LEFTRIGHT

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