Left: | ||
Right: |
LEFT | RIGHT |
---|---|
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 := ¬ifyWorker{ | 47 nw := ¬ifyWorker{ |
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 } |
LEFT | RIGHT |