resque-pool-dynamic 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
data/doc/css/style.css ADDED
@@ -0,0 +1,322 @@
1
+ body {
2
+ padding: 0 20px;
3
+ font-family: "Lucida Sans", "Lucida Grande", Verdana, Arial, sans-serif;
4
+ font-size: 13px;
5
+ }
6
+ body.frames { padding: 0 5px; }
7
+ h1 { font-size: 25px; margin: 1em 0 0.5em; padding-top: 4px; border-top: 1px dotted #d5d5d5; }
8
+ h1.noborder { border-top: 0px; margin-top: 0; padding-top: 4px; }
9
+ h1.title { margin-bottom: 10px; }
10
+ h1.alphaindex { margin-top: 0; font-size: 22px; }
11
+ h2 {
12
+ padding: 0;
13
+ padding-bottom: 3px;
14
+ border-bottom: 1px #aaa solid;
15
+ font-size: 1.4em;
16
+ margin: 1.8em 0 0.5em;
17
+ }
18
+ h2 small { font-weight: normal; font-size: 0.7em; display: block; float: right; }
19
+ .clear { clear: both; }
20
+ .inline { display: inline; }
21
+ .inline p:first-child { display: inline; }
22
+ .docstring h1, .docstring h2, .docstring h3, .docstring h4 { padding: 0; border: 0; border-bottom: 1px dotted #bbb; }
23
+ .docstring h1 { font-size: 1.2em; }
24
+ .docstring h2 { font-size: 1.1em; }
25
+ .docstring h3, .docstring h4 { font-size: 1em; border-bottom: 0; padding-top: 10px; }
26
+ .summary_desc .object_link, .docstring .object_link { font-family: monospace; }
27
+ .rdoc-term { padding-right: 25px; font-weight: bold; }
28
+ .rdoc-list p { margin: 0; padding: 0; margin-bottom: 4px; }
29
+
30
+ /* style for <ul> */
31
+ #filecontents li > p, .docstring li > p { margin: 0px; }
32
+ #filecontents ul, .docstring ul { padding-left: 20px; }
33
+ /* style for <dl> */
34
+ #filecontents dl, .docstring dl { border: 1px solid #ccc; }
35
+ #filecontents dt, .docstring dt { background: #ddd; font-weight: bold; padding: 3px 5px; }
36
+ #filecontents dd, .docstring dd { padding: 5px 0px; margin-left: 18px; }
37
+ #filecontents dd > p, .docstring dd > p { margin: 0px; }
38
+
39
+ .note {
40
+ color: #222;
41
+ -moz-border-radius: 3px; -webkit-border-radius: 3px;
42
+ background: #e3e4e3; border: 1px solid #d5d5d5; padding: 7px 10px;
43
+ display: block;
44
+ }
45
+ .note.todo { background: #ffffc5; border-color: #ececaa; }
46
+ .note.returns_void { background: #efefef; }
47
+ .note.deprecated { background: #ffe5e5; border-color: #e9dada; }
48
+ .note.private { background: #ffffc5; border-color: #ececaa; }
49
+ .note.title { text-transform: lowercase; padding: 1px 5px; font-size: 0.9em; font-family: "Lucida Sans", "Lucida Grande", Verdana, Arial, sans-serif; display: inline; }
50
+ .summary_signature + .note.title { margin-left: 7px; }
51
+ h1 .note.title { font-size: 0.5em; font-weight: normal; padding: 3px 5px; position: relative; top: -3px; text-transform: capitalize; }
52
+ .note.title.constructor { color: #fff; background: #6a98d6; border-color: #6689d6; }
53
+ .note.title.writeonly { color: #fff; background: #45a638; border-color: #2da31d; }
54
+ .note.title.readonly { color: #fff; background: #6a98d6; border-color: #6689d6; }
55
+ .note.title.private { background: #d5d5d5; border-color: #c5c5c5; }
56
+ .discussion .note { margin-top: 6px; }
57
+ .discussion .note:first-child { margin-top: 0; }
58
+
59
+ h3.inherited {
60
+ font-style: italic;
61
+ font-family: "Lucida Sans", "Lucida Grande", Verdana, Arial, sans-serif;
62
+ font-weight: normal;
63
+ padding: 0;
64
+ margin: 0;
65
+ margin-top: 12px;
66
+ margin-bottom: 3px;
67
+ font-size: 13px;
68
+ }
69
+ p.inherited {
70
+ padding: 0;
71
+ margin: 0;
72
+ margin-left: 25px;
73
+ }
74
+
75
+ #filecontents dl.box, dl.box {
76
+ border: 0;
77
+ width: 520px;
78
+ font-size: 1em;
79
+ }
80
+ #filecontents dl.box dt, dl.box dt {
81
+ float: left;
82
+ display: block;
83
+ width: 100px;
84
+ margin: 0;
85
+ text-align: right;
86
+ font-weight: bold;
87
+ background: transparent;
88
+ border: 1px solid #aaa;
89
+ border-width: 1px 0px 0px 1px;
90
+ padding: 6px 0;
91
+ padding-right: 10px;
92
+ }
93
+ #filecontents dl.box dd, dl.box dd {
94
+ float: left;
95
+ display: block;
96
+ width: 380px;
97
+ margin: 0;
98
+ padding: 6px 0;
99
+ padding-right: 20px;
100
+ border: 1px solid #aaa;
101
+ border-width: 1px 1px 0 0;
102
+ }
103
+ #filecontents dl.box .last, dl.box .last {
104
+ border-bottom: 1px solid #aaa;
105
+ }
106
+ #filecontents dl.box .r1, dl.box .r1 { background: #eee; }
107
+
108
+ ul.toplevel { list-style: none; padding-left: 0; font-size: 1.1em; }
109
+ #files { padding-left: 15px; font-size: 1.1em; }
110
+
111
+ #files { padding: 0; }
112
+ #files li { list-style: none; display: inline; padding: 7px 12px; line-height: 35px; }
113
+
114
+ dl.constants { margin-left: 40px; }
115
+ dl.constants dt { font-weight: bold; font-size: 1.1em; margin-bottom: 5px; }
116
+ dl.constants dd { width: 75%; white-space: pre; font-family: monospace; margin-bottom: 18px; }
117
+
118
+ .summary_desc { margin-left: 32px; display: block; font-family: sans-serif; }
119
+ .summary_desc tt { font-size: 0.9em; }
120
+ dl.constants .note { padding: 2px 6px; padding-right: 12px; margin-top: 6px; }
121
+ dl.constants .docstring { margin-left: 32px; font-size: 0.9em; font-weight: normal; }
122
+ dl.constants .tags { padding-left: 32px; font-size: 0.9em; line-height: 0.8em; }
123
+ dl.constants .discussion *:first-child { margin-top: 0; }
124
+ dl.constants .discussion *:last-child { margin-bottom: 0; }
125
+
126
+ .method_details { border-top: 1px dotted #aaa; margin-top: 15px; padding-top: 0; }
127
+ .method_details.first { border: 0; }
128
+ p.signature {
129
+ font-size: 1.1em; font-weight: normal; font-family: Monaco, Consolas, Courier, monospace;
130
+ padding: 6px 10px; margin-top: 18px;
131
+ background: #e5e8ff; border: 1px solid #d8d8e5; -moz-border-radius: 3px; -webkit-border-radius: 3px;
132
+ }
133
+ p.signature tt { font-family: Monaco, Consolas, Courier, monospace; }
134
+ p.signature .overload { display: block; }
135
+ p.signature .extras { font-weight: normal; font-family: sans-serif; color: #444; font-size: 1em; }
136
+ p.signature .aliases { display: block; font-weight: normal; font-size: 0.9em; font-family: sans-serif; margin-top: 0px; color: #555; }
137
+ p.signature .aliases .names { font-family: Monaco, Consolas, Courier, monospace; font-weight: bold; color: #000; font-size: 1.2em; }
138
+
139
+ .tags h3 { font-size: 1em; margin-bottom: 0; }
140
+ .tags ul { margin-top: 5px; padding-left: 30px; list-style: square; }
141
+ .tags ul li { margin-bottom: 3px; }
142
+ .tags ul .name { font-family: monospace; font-weight: bold; }
143
+ .tags ul .note { padding: 3px 6px; }
144
+ .tags { margin-bottom: 12px; }
145
+
146
+ .tags .examples h3 { margin-bottom: 10px; }
147
+ .tags .examples h4 { padding: 0; margin: 0; margin-left: 15px; font-weight: bold; font-size: 0.9em; }
148
+
149
+ .tags .overload .overload_item { list-style: none; margin-bottom: 25px; }
150
+ .tags .overload .overload_item .signature {
151
+ padding: 2px 8px;
152
+ background: #e5e8ff; border: 1px solid #d8d8e5; -moz-border-radius: 3px; -webkit-border-radius: 3px;
153
+ }
154
+ .tags .overload .signature { margin-left: -15px; font-family: monospace; display: block; font-size: 1.1em; }
155
+ .tags .overload .docstring { margin-top: 15px; }
156
+
157
+ .defines { display: none; }
158
+
159
+ #method_missing_details .notice.this { position: relative; top: -8px; color: #888; padding: 0; margin: 0; }
160
+
161
+ .showSource { font-size: 0.9em; }
162
+ .showSource a:link, .showSource a:visited { text-decoration: none; color: #666; }
163
+
164
+ #content a:link, #content a:visited { text-decoration: none; color: #05a; }
165
+ #content a:hover { background: #ffffa5; }
166
+ div.docstring, p.docstring { margin-right: 6em; }
167
+
168
+ ul.summary {
169
+ list-style: none;
170
+ font-family: monospace;
171
+ font-size: 1em;
172
+ line-height: 1.5em;
173
+ }
174
+ ul.summary a:link, ul.summary a:visited {
175
+ text-decoration: none; font-size: 1.1em;
176
+ }
177
+ ul.summary li { margin-bottom: 5px; }
178
+ .summary .summary_signature {
179
+ padding: 1px 10px;
180
+ background: #eaeaff; border: 1px solid #dfdfe5;
181
+ -moz-border-radius: 3px; -webkit-border-radius: 3px;
182
+ }
183
+ .summary_signature:hover { background: #eeeeff; cursor: pointer; }
184
+ ul.summary.compact li { display: inline-block; margin: 0px 5px 0px 0px; line-height: 2.6em;}
185
+ ul.summary.compact .summary_signature { padding: 5px 7px; padding-right: 4px; }
186
+ #content .summary_signature:hover a:link,
187
+ #content .summary_signature:hover a:visited {
188
+ background: transparent;
189
+ color: #48f;
190
+ }
191
+
192
+ p.inherited a { font-family: monospace; font-size: 0.9em; }
193
+ p.inherited { word-spacing: 5px; font-size: 1.2em; }
194
+
195
+ p.children { font-size: 1.2em; }
196
+ p.children a { font-size: 0.9em; }
197
+ p.children strong { font-size: 0.8em; }
198
+ p.children strong.modules { padding-left: 5px; }
199
+
200
+ ul.fullTree { display: none; padding-left: 0; list-style: none; margin-left: 0; margin-bottom: 10px; }
201
+ ul.fullTree ul { margin-left: 0; padding-left: 0; list-style: none; }
202
+ ul.fullTree li { text-align: center; padding-top: 18px; padding-bottom: 12px; background: url() no-repeat top center; }
203
+ ul.fullTree li:first-child { padding-top: 0; background: transparent; }
204
+ ul.fullTree li:last-child { padding-bottom: 0; }
205
+ .showAll ul.fullTree { display: block; }
206
+ .showAll .inheritName { display: none; }
207
+
208
+ #search { position: absolute; right: 14px; top: 0px; }
209
+ #search a:link, #search a:visited {
210
+ display: block; float: left; margin-right: 4px;
211
+ padding: 8px 10px; text-decoration: none; color: #05a;
212
+ border: 1px solid #d8d8e5;
213
+ -moz-border-radius-bottomleft: 3px; -moz-border-radius-bottomright: 3px;
214
+ -webkit-border-bottom-left-radius: 3px; -webkit-border-bottom-right-radius: 3px;
215
+ background: #eaf0ff;
216
+ -webkit-box-shadow: -1px 1px 3px #ddd;
217
+ }
218
+ #search a:hover { background: #f5faff; color: #06b; }
219
+ #search a.active {
220
+ background: #568; padding-bottom: 20px; color: #fff; border: 1px solid #457;
221
+ -moz-border-radius-topleft: 5px; -moz-border-radius-topright: 5px;
222
+ -webkit-border-top-left-radius: 5px; -webkit-border-top-right-radius: 5px;
223
+ }
224
+ #search a.inactive { color: #999; }
225
+ .frames #search { display: none; }
226
+ .inheritanceTree, .toggleDefines { float: right; }
227
+
228
+ #menu { font-size: 1.3em; color: #bbb; top: -5px; position: relative; }
229
+ #menu .title, #menu a { font-size: 0.7em; }
230
+ #menu .title a { font-size: 1em; }
231
+ #menu .title { color: #555; }
232
+ #menu a:link, #menu a:visited { color: #333; text-decoration: none; border-bottom: 1px dotted #bbd; }
233
+ #menu a:hover { color: #05a; }
234
+ #menu .noframes { display: none; }
235
+ .frames #menu .noframes { display: inline; float: right; }
236
+
237
+ #footer { margin-top: 15px; border-top: 1px solid #ccc; text-align: center; padding: 7px 0; color: #999; }
238
+ #footer a:link, #footer a:visited { color: #444; text-decoration: none; border-bottom: 1px dotted #bbd; }
239
+ #footer a:hover { color: #05a; }
240
+
241
+ #listing ul.alpha { font-size: 1.1em; }
242
+ #listing ul.alpha { margin: 0; padding: 0; padding-bottom: 10px; list-style: none; }
243
+ #listing ul.alpha li.letter { font-size: 1.4em; padding-bottom: 10px; }
244
+ #listing ul.alpha ul { margin: 0; padding-left: 15px; }
245
+ #listing ul small { color: #666; font-size: 0.7em; }
246
+
247
+ li.r1 { background: #f0f0f0; }
248
+ li.r2 { background: #fafafa; }
249
+
250
+ #search_frame {
251
+ z-index: 9999;
252
+ background: #fff;
253
+ display: none;
254
+ position: absolute;
255
+ top: 36px;
256
+ right: 18px;
257
+ width: 500px;
258
+ height: 80%;
259
+ overflow-y: scroll;
260
+ border: 1px solid #999;
261
+ border-collapse: collapse;
262
+ -webkit-box-shadow: -7px 5px 25px #aaa;
263
+ -moz-box-shadow: -7px 5px 25px #aaa;
264
+ -moz-border-radius: 2px;
265
+ -webkit-border-radius: 2px;
266
+ }
267
+
268
+ #content ul.summary li.deprecated .summary_signature a:link,
269
+ #content ul.summary li.deprecated .summary_signature a:visited { text-decoration: line-through; font-style: italic; }
270
+
271
+ #toc {
272
+ padding: 20px; padding-right: 30px; border: 1px solid #ddd; float: right; background: #fff; margin-left: 20px; margin-bottom: 20px;
273
+ max-width: 300px;
274
+ -webkit-box-shadow: -2px 2px 6px #bbb;
275
+ -moz-box-shadow: -2px 2px 6px #bbb;
276
+ z-index: 5000;
277
+ position: relative;
278
+ }
279
+ #toc.nofloat { float: none; max-width: none; border: none; padding: 0; margin: 20px 0; -webkit-box-shadow: none; -moz-box-shadow: none; }
280
+ #toc.nofloat.hidden { padding: 0; background: 0; margin-bottom: 5px; }
281
+ #toc .title { margin: 0; }
282
+ #toc ol { padding-left: 1.8em; }
283
+ #toc li { font-size: 1.1em; line-height: 1.7em; }
284
+ #toc > ol > li { font-size: 1.1em; font-weight: bold; }
285
+ #toc ol > ol { font-size: 0.9em; }
286
+ #toc ol ol > ol { padding-left: 2.3em; }
287
+ #toc ol + li { margin-top: 0.3em; }
288
+ #toc.hidden { padding: 10px; background: #f6f6f6; -webkit-box-shadow: none; -moz-box-shadow: none; }
289
+ #filecontents h1 + #toc.nofloat { margin-top: 0; }
290
+
291
+ /* syntax highlighting */
292
+ .source_code { display: none; padding: 3px 8px; border-left: 8px solid #ddd; margin-top: 5px; }
293
+ #filecontents pre.code, .docstring pre.code, .source_code pre { font-family: monospace; }
294
+ #filecontents pre.code, .docstring pre.code { display: block; }
295
+ .source_code .lines { padding-right: 12px; color: #555; text-align: right; }
296
+ #filecontents pre.code, .docstring pre.code,
297
+ .tags pre.example { padding: 5px 12px; margin-top: 4px; border: 1px solid #eef; background: #f5f5ff; }
298
+ pre.code { color: #000; }
299
+ pre.code .info.file { color: #555; }
300
+ pre.code .val { color: #036A07; }
301
+ pre.code .tstring_content,
302
+ pre.code .heredoc_beg, pre.code .heredoc_end,
303
+ pre.code .qwords_beg, pre.code .qwords_end,
304
+ pre.code .tstring, pre.code .dstring { color: #036A07; }
305
+ pre.code .fid, pre.code .rubyid_new, pre.code .rubyid_to_s,
306
+ pre.code .rubyid_to_sym, pre.code .rubyid_to_f,
307
+ pre.code .dot + pre.code .id,
308
+ pre.code .rubyid_to_i pre.code .rubyid_each { color: #0085FF; }
309
+ pre.code .comment { color: #0066FF; }
310
+ pre.code .const, pre.code .constant { color: #585CF6; }
311
+ pre.code .symbol { color: #C5060B; }
312
+ pre.code .kw,
313
+ pre.code .label,
314
+ pre.code .rubyid_require,
315
+ pre.code .rubyid_extend,
316
+ pre.code .rubyid_include { color: #0000FF; }
317
+ pre.code .ivar { color: #318495; }
318
+ pre.code .gvar,
319
+ pre.code .rubyid_backref,
320
+ pre.code .rubyid_nth_ref { color: #6D79DE; }
321
+ pre.code .regexp, .dregexp { color: #036A07; }
322
+ pre.code a { border-bottom: 1px dotted #bbf; }
@@ -0,0 +1,316 @@
1
+ <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
2
+ "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
3
+ <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
4
+ <head>
5
+ <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
6
+ <title>
7
+ File: README
8
+
9
+ &mdash; Documentation by YARD 0.7.5
10
+
11
+ </title>
12
+
13
+ <link rel="stylesheet" href="css/style.css" type="text/css" media="screen" charset="utf-8" />
14
+
15
+ <link rel="stylesheet" href="css/common.css" type="text/css" media="screen" charset="utf-8" />
16
+
17
+ <script type="text/javascript" charset="utf-8">
18
+ relpath = '';
19
+ if (relpath != '') relpath += '/';
20
+ </script>
21
+
22
+ <script type="text/javascript" charset="utf-8" src="js/jquery.js"></script>
23
+
24
+ <script type="text/javascript" charset="utf-8" src="js/app.js"></script>
25
+
26
+
27
+ </head>
28
+ <body>
29
+ <script type="text/javascript" charset="utf-8">
30
+ if (window.top.frames.main) document.body.className = 'frames';
31
+ </script>
32
+
33
+ <div id="header">
34
+ <div id="menu">
35
+
36
+ <a href="_index.html" title="Index">Index</a> &raquo;
37
+ <span class="title">File: README</span>
38
+
39
+
40
+ <div class="noframes"><span class="title">(</span><a href="." target="_top">no frames</a><span class="title">)</span></div>
41
+ </div>
42
+
43
+ <div id="search">
44
+
45
+ <a id="class_list_link" href="#">Class List</a>
46
+
47
+ <a id="method_list_link" href="#">Method List</a>
48
+
49
+ <a id="file_list_link" href="#">File List</a>
50
+
51
+ </div>
52
+ <div class="clear"></div>
53
+ </div>
54
+
55
+ <iframe id="search_frame"></iframe>
56
+
57
+ <div id="content"><div id='filecontents'><h1 id="dynamic-resque-pool">Dynamic Resque Pool</h1>
58
+
59
+ <p><a href="https://github.com/nevans/resque-pool">Resque pool</a> by Nicholas Evans
60
+ is a library for managing a pool of
61
+ <a href="http://github.com/defunkt/resque">resque</a> workers. Given a a config
62
+ file, it manages your workers for you, starting up the appropriate
63
+ number of workers for each worker type.</p>
64
+
65
+ <p>While the resque pool is convenient for a permanent process
66
+ overlooking a fairly constant set of workers, it is less than
67
+ convenient to use for more dynamic tasks - one-time long running batch
68
+ processing worker families that need to be started and supervised
69
+ manually, and often require adjusting number of worker processes for
70
+ maximum performance.</p>
71
+
72
+ <p>Workflow for this kind of task would be:</p>
73
+
74
+ <ul>
75
+ <li>prepare a yaml file with number of workers</li>
76
+ <li>start resque-pool</li>
77
+ <li><code>tail -f</code> the log file on the other console</li>
78
+ <li>look into <code>pstree</code> (or <code>ps faxuw</code> on Linux)</li>
79
+ <li>babysit the process, check the system load whether number of workers
80
+ needs to be adjusted</li>
81
+ <li>edit the yaml file</li>
82
+ <li><code>kill -HUP</code> the resque pool master (hope you remembered the PID from
83
+ pstree!)</li>
84
+ <li>rinse, repeat</li>
85
+ </ul>
86
+
87
+ <p>Or you can add <code>gem "resque-pool-dynamic"</code> to your Gemfile, <code>require
88
+ 'resque/pool/dynamic/tasks'</code> to your Rakefile, and start an
89
+ interactive CLI interface that controls the resque-pool-master.</p>
90
+
91
+ <h2 id="example-resque-pool-manager-session">Example resque-pool-manager session</h2>
92
+
93
+ <pre class="code ruby"><code>$ bundle exec rake resque:pool:dynamic WORKERS=foo=2:bar=2:\*=1
94
+ resque-pool-manager[21134]: started manager
95
+
96
+ Status: running, pid: 21134
97
+ Configuration:
98
+ bar: 2
99
+ &quot;*&quot;: 1
100
+ foo: 2
101
+ Process tree:
102
+ -+= 21134 japhy resque-pool-master: managing [21136, 21135, 21137, 21138, 21141]
103
+ |--- 21135 japhy resque-1.20.0: Waiting for bar
104
+ |--- 21136 japhy resque-1.20.0: Waiting for bar
105
+ |--- 21137 japhy resque-1.20.0: Waiting for *
106
+ |--- 21138 japhy resque-1.20.0: Waiting for foo
107
+ \--- 21141 japhy resque-1.20.0: Waiting for foo
108
+
109
+ &gt;&gt; log.tail
110
+ resque-pool-manager[21134]: Pool contains worker PIDs: [21136, 21135, 21137, 21138, 21141]
111
+ resque-pool-worker[21135]: Starting worker portinari.local:21135:bar
112
+ resque-pool-worker[21137]: Starting worker portinari.local:21137:*
113
+
114
+ resque-pool-worker[21136]: Starting worker portinari.local:21136:bar
115
+ resque-pool-worker[21141]: Starting worker portinari.local:21141:foo
116
+ resque-pool-worker[21138]: Starting worker portinari.local:21138:foo
117
+ </code></pre>
118
+
119
+ <p>(log.tail is following the logfile like <code>tail -f</code> until you break with ^C)</p>
120
+
121
+ <pre class="code ruby"><code>^C&gt;&gt; config :foo =&gt; 1, :bar =&gt; 4
122
+ Reloading resque-pool-master 21134 configuration
123
+ =&gt; {&quot;bar&quot;=&gt;4, &quot;*&quot;=&gt;1, &quot;foo&quot;=&gt;1}
124
+ &gt;&gt; status
125
+
126
+ Status: running, pid: 21134
127
+ Configuration:
128
+ bar: 4
129
+ &quot;*&quot;: 1
130
+ foo: 1
131
+ Process tree:
132
+ -+= 21134 japhy resque-pool-master: managing [21176, 21178, 21162, 21163, 21180, 21181]
133
+ |--- 21162 japhy resque-1.20.0: Waiting for bar
134
+ |--- 21163 japhy resque-1.20.0: Waiting for bar
135
+ |--- 21176 japhy resque-1.20.0: Waiting for bar
136
+ |--- 21178 japhy resque-1.20.0: Waiting for bar
137
+ |--- 21180 japhy resque-1.20.0: Waiting for *
138
+ \--- 21181 japhy resque-1.20.0: Waiting for foo
139
+
140
+ &gt;&gt; log.tail
141
+ resque-pool-manager[21134]: HUP: reload config file and reload logfiles
142
+ resque-pool-manager[21134]: Flushing logs
143
+ resque-pool-manager[21134]: HUP: gracefully shutdown old children (which have old logfiles open)
144
+ resque-pool-manager[21134]: HUP: new children will inherit new logfiles
145
+ resque-pool-worker[21163]: Starting worker portinari.local:21163:bar
146
+ resque-pool-worker[21162]: Starting worker portinari.local:21162:bar
147
+ resque-pool-manager[21134]: Reaped resque worker[21136] (status: 0) queues: bar
148
+ resque-pool-manager[21134]: Reaped resque worker[21135] (status: 0) queues: bar
149
+ resque-pool-worker[21176]: Starting worker portinari.local:21176:bar
150
+ resque-pool-worker[21178]: Starting worker portinari.local:21178:bar
151
+ resque-pool-manager[21134]: Reaped resque worker[21141] (status: 0) queues: foo
152
+ resque-pool-manager[21134]: Reaped resque worker[21138] (status: 0) queues: foo
153
+ resque-pool-worker[21180]: Starting worker portinari.local:21180:*
154
+ resque-pool-worker[21181]: Starting worker portinari.local:21181:foo
155
+ ^C&gt;&gt; exit
156
+ Bye!
157
+ Shutting down resque-pool-master 21134
158
+ resque-pool-manager[21134]: INT: immediate shutdown (graceful worker shutdown)
159
+ resque-pool-manager[21134]: manager finished
160
+ $ _
161
+ </code></pre>
162
+
163
+ <h2 id="command-line-interface">Command Line Interface</h2>
164
+
165
+ <p>The CLI is a slightly customized Ripl shell (an IRB replacement)
166
+ started in context of a <code>Resque::Pool::Dynamic</code> instance. You can use
167
+ all Ruby you want, and call all the methods for controlling the resque
168
+ pool. A simple built-in help for the custom methods is included:</p>
169
+
170
+ <pre class="code ruby"><code>&gt;&gt; help
171
+ Known commands:
172
+ config - &quot;WORKERS&quot;, as interpreted by the #parse_config_string method.
173
+ config_path - Path to the temporary config file.
174
+ exit - Finish work
175
+ has_log? - True if we have an open log file.
176
+ kill! - Send signal to a running resque-pool.
177
+ log - Open log of resque-pool process.
178
+ log.ff - Fast forward until end of file (see IO::Tail#backward).
179
+ log.rew - Rewind until beginning of file (see IO::Tail#forward).
180
+ log.rewind - Rewind until beginning of file (see IO::Tail#forward).
181
+ log.tail - Show last n lines of the file, or follow the file.
182
+ log.tail_to_eof - Print contents of the file until current end of file.
183
+ log.tail_until - Follow the file until a line matches regexp.
184
+ log_path - Logfile path.
185
+ pid - PID of the resque-pool process, or nil if it's not running.
186
+ pstree - Show child process tree by calling `pstree` system command.
187
+ reload - Reload resque-pool configuration.
188
+ start - Start resque-pool, showing startup logs.
189
+ start! - Fork a child process for resque-pool master.
190
+ status - Show current status.
191
+ stop - Stop running resque-pool, show shutdown logs.
192
+ stop! - Stop running resque-pool.
193
+ write_config - Write temporary configuration file.
194
+ For details, type: help command
195
+ &gt;&gt; help config
196
+ -------------------------------- Method: #config (Resque::Pool::Dynamic)
197
+ (Defined in: lib/resque/pool/dynamic/process.rb)
198
+ dynamic.config -&gt; Hash
199
+ dynamic.config(opts) -&gt; Hash
200
+ ------------------------------------------------------------------------
201
+ Note: Default configuration is taken from environment variable
202
+ &quot;WORKERS&quot;, as interpreted by the #parse_config_string method.
203
+ Examples:
204
+ ---------
205
+ # config
206
+ #=&gt; {&quot;foo&quot;=&gt;1, &quot;bar&quot;=&gt;2}
207
+ config :foo =&gt; 2, :baz =&gt; 7
208
+ #=&gt; {&quot;foo&quot;=&gt;2, &quot;baz&quot;=&gt;7, &quot;bar&quot;=&gt;2}
209
+ config :bar =&gt; 0
210
+ #=&gt; {&quot;foo&quot;=&gt;2, &quot;baz&quot;=&gt;7}
211
+ Overloads:
212
+ ----------
213
+ ------------------------------------------------------------------------
214
+ dynamic.config -&gt; Hash
215
+ ------------------------------------------------------------------------
216
+ Show current workers configuration
217
+ Returns:
218
+ --------
219
+ (Hash) -
220
+ ------------------------------------------------------------------------
221
+ dynamic.config(opts) -&gt; Hash
222
+ ------------------------------------------------------------------------
223
+ Update workers configuration. If configuration has change and
224
+ resque-pool is running, it is reloaded.
225
+ Parameters:
226
+ -----------
227
+ (Hash) opts - Dictionary of worker process counts to update
228
+ (pass 0 or nil as value to delete all workers for a queue from pool)
229
+ Returns:
230
+ --------
231
+ (Hash) - Updated config
232
+ &gt;&gt; _
233
+ </code></pre>
234
+
235
+ <p>The built-in help is autogenerated from Yard documentation on methods;
236
+ if you’re curious, look into the Rakefile for details.</p>
237
+
238
+ <h2 id="rake-task-documentation">Rake task documentation</h2>
239
+
240
+ <pre class="code ruby"><code>$ bundle exec rake -D resque:pool:dynamic
241
+ rake resque:pool:dynamic
242
+ Starts a dynamic pool of Resque worker processes.
243
+
244
+ Variables:
245
+ WORKERS=worker spec - specify initial numbers of workers
246
+ (look below for format description)
247
+ NO_START=1 - don't start resque-pool master automatically
248
+ (default is to start)
249
+
250
+ Workers spec format:
251
+ queue_name=process_number[:queue_name=process_number[:...]]
252
+
253
+ queue_name can be whatever resque:work will accept as QUEUE=
254
+ parameter.
255
+
256
+ Example:
257
+ $ rake resque:pool:dynamic WORKERS=foo=1:bar=1:baz,quux,xyzzy=5:*=2
258
+ will start:
259
+ - one worker for queue 'foo'
260
+ - one worker for queue 'bar'
261
+ - five workers for queues baz,quux,xyzzy
262
+ - two workers for all queues
263
+ </code></pre>
264
+
265
+ <h2 id="api-and-code-structure">API and code structure</h2>
266
+
267
+ <p>The Rake task, defined in <code>resque/pool/dynamic/tasks.rb</code>, is a
268
+ one-liner calling out to <code>Resque::Pool::Dynamic.shell</code>, which starts
269
+ the command line interface.</p>
270
+
271
+ <p>The CLI itself, defined in <code>resque/pool/dynamic/shell.rb</code>, is a thin
272
+ layer (8 lines of code + Ripl UI tweaks) over an instance of the
273
+ <code>Resque::Pool::Dynamic</code> class.</p>
274
+
275
+ <p>The workhorse <code>Resque::Pool::Dynamic</code> class is defined in
276
+ <code>resque/pool/dynamic.rb</code>. It can be easily instantiated in the code
277
+ independent from the CLI; the CLI commands are just instance methods
278
+ you can call then.</p>
279
+
280
+ <h3 id="ideas">Ideas</h3>
281
+
282
+ <p>Besides interactive, supervised work, <code>Resque::Pool::Dynamic</code> can be
283
+ used separately from the UI for dynamic, reactive management of
284
+ workers, implementing some kind of autoscaling and load distribution
285
+ logic.</p>
286
+
287
+ <p>It can be used during load tests to automatically optimize best number
288
+ of workers for maximum processing speed, while keeping load average
289
+ and memory or CPU usage goals.</p>
290
+
291
+ <p>It may also react to queue depths, e.g. temporarily reallocating idle
292
+ workers to overloaded queues or shutting them down to conserve memory,
293
+ or actively managing sudden short peak usage by stopping other workers
294
+ and switching them to peaking queue.</p>
295
+
296
+ <p>It may also simply allocate workers dynamically based on time of day,
297
+ to e.g. have more workers performing UI-related task in the day, which
298
+ increases perceived responsiveness, and switch them to process
299
+ long-running reports overnight when there are less users.</p>
300
+
301
+ <h2 id="detailed-documentation-and-source-code">Detailed documentation and source code</h2>
302
+
303
+ <ul>
304
+ <li>http://rubydoc.info/FIXME</li>
305
+ <li>http://github.com/mpasternacki/resque-pool-dynamic</li>
306
+ </ul>
307
+ </div></div>
308
+
309
+ <div id="footer">
310
+ Generated on Wed Apr 25 21:03:02 2012 by
311
+ <a href="http://yardoc.org" title="Yay! A Ruby Documentation Tool" target="_parent">yard</a>
312
+ 0.7.5 (ruby-1.9.2).
313
+ </div>
314
+
315
+ </body>
316
+ </html>