dysinger-rush 0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. data/Rakefile +65 -0
  2. data/bin/rush +13 -0
  3. data/bin/rushd +7 -0
  4. data/lib/rush.rb +27 -0
  5. data/lib/rush/access.rb +130 -0
  6. data/lib/rush/array_ext.rb +19 -0
  7. data/lib/rush/box.rb +112 -0
  8. data/lib/rush/commands.rb +55 -0
  9. data/lib/rush/config.rb +154 -0
  10. data/lib/rush/dir.rb +158 -0
  11. data/lib/rush/embeddable_shell.rb +26 -0
  12. data/lib/rush/entry.rb +178 -0
  13. data/lib/rush/exceptions.rb +31 -0
  14. data/lib/rush/file.rb +77 -0
  15. data/lib/rush/find_by.rb +39 -0
  16. data/lib/rush/fixnum_ext.rb +18 -0
  17. data/lib/rush/head_tail.rb +11 -0
  18. data/lib/rush/local.rb +374 -0
  19. data/lib/rush/process.rb +55 -0
  20. data/lib/rush/process_set.rb +62 -0
  21. data/lib/rush/remote.rb +152 -0
  22. data/lib/rush/search_results.rb +58 -0
  23. data/lib/rush/server.rb +117 -0
  24. data/lib/rush/shell.rb +148 -0
  25. data/lib/rush/ssh_tunnel.rb +122 -0
  26. data/lib/rush/string_ext.rb +3 -0
  27. data/spec/access_spec.rb +134 -0
  28. data/spec/array_ext_spec.rb +15 -0
  29. data/spec/base.rb +24 -0
  30. data/spec/box_spec.rb +64 -0
  31. data/spec/commands_spec.rb +47 -0
  32. data/spec/config_spec.rb +108 -0
  33. data/spec/dir_spec.rb +159 -0
  34. data/spec/embeddable_shell_spec.rb +17 -0
  35. data/spec/entry_spec.rb +129 -0
  36. data/spec/file_spec.rb +79 -0
  37. data/spec/find_by_spec.rb +58 -0
  38. data/spec/fixnum_ext_spec.rb +19 -0
  39. data/spec/local_spec.rb +313 -0
  40. data/spec/process_set_spec.rb +50 -0
  41. data/spec/process_spec.rb +73 -0
  42. data/spec/remote_spec.rb +135 -0
  43. data/spec/search_results_spec.rb +44 -0
  44. data/spec/shell_spec.rb +12 -0
  45. data/spec/ssh_tunnel_spec.rb +122 -0
  46. data/spec/string_ext_spec.rb +23 -0
  47. metadata +126 -0
@@ -0,0 +1,313 @@
1
+ require File.dirname(__FILE__) + '/base'
2
+
3
+ describe Rush::Connection::Local do
4
+ before do
5
+ @sandbox_dir = "/tmp/rush_spec.#{Process.pid}"
6
+ system "rm -rf #{@sandbox_dir}; mkdir -p #{@sandbox_dir}"
7
+
8
+ @con = Rush::Connection::Local.new
9
+ end
10
+
11
+ after do
12
+ system "rm -rf #{@sandbox_dir}"
13
+ end
14
+
15
+ it "receive -> write_file(file, contents)" do
16
+ @con.should_receive(:write_file).with('file', 'contents')
17
+ @con.receive(:action => 'write_file', :full_path => 'file', :payload => 'contents')
18
+ end
19
+
20
+ it "receive -> file_contents(file)" do
21
+ @con.should_receive(:file_contents).with('file').and_return('the contents')
22
+ @con.receive(:action => 'file_contents', :full_path => 'file').should == 'the contents'
23
+ end
24
+
25
+ it "receive -> destroy(file or dir)" do
26
+ @con.should_receive(:destroy).with('file')
27
+ @con.receive(:action => 'destroy', :full_path => 'file')
28
+ end
29
+
30
+ it "receive -> purge(dir)" do
31
+ @con.should_receive(:purge).with('dir')
32
+ @con.receive(:action => 'purge', :full_path => 'dir')
33
+ end
34
+
35
+ it "receive -> create_dir(path)" do
36
+ @con.should_receive(:create_dir).with('dir')
37
+ @con.receive(:action => 'create_dir', :full_path => 'dir')
38
+ end
39
+
40
+ it "receive -> rename(path, name, new_name)" do
41
+ @con.should_receive(:rename).with('path', 'name', 'new_name')
42
+ @con.receive(:action => 'rename', :path => 'path', :name => 'name', :new_name => 'new_name')
43
+ end
44
+
45
+ it "receive -> copy(src, dst)" do
46
+ @con.should_receive(:copy).with('src', 'dst')
47
+ @con.receive(:action => 'copy', :src => 'src', :dst => 'dst')
48
+ end
49
+
50
+ it "receive -> read_archive(full_path)" do
51
+ @con.should_receive(:read_archive).with('full_path').and_return('archive data')
52
+ @con.receive(:action => 'read_archive', :full_path => 'full_path').should == 'archive data'
53
+ end
54
+
55
+ it "receive -> write_archive(archive, dir)" do
56
+ @con.should_receive(:write_archive).with('archive', 'dir')
57
+ @con.receive(:action => 'write_archive', :dir => 'dir', :payload => 'archive')
58
+ end
59
+
60
+ it "receive -> index(base_path, glob)" do
61
+ @con.should_receive(:index).with('base_path', '*').and_return(%w(1 2))
62
+ @con.receive(:action => 'index', :base_path => 'base_path', :glob => '*').should == "1\n2\n"
63
+ end
64
+
65
+ it "receive -> stat(full_path)" do
66
+ @con.should_receive(:stat).with('full_path').and_return(1 => 2)
67
+ @con.receive(:action => 'stat', :full_path => 'full_path').should == YAML.dump(1 => 2)
68
+ end
69
+
70
+ it "receive -> set_access(full_path, user, group, permissions)" do
71
+ access = mock("access")
72
+ Rush::Access.should_receive(:from_hash).with(:action => 'set_access', :full_path => 'full_path', :user => 'joe').and_return(access)
73
+
74
+ @con.should_receive(:set_access).with('full_path', access)
75
+ @con.receive(:action => 'set_access', :full_path => 'full_path', :user => 'joe')
76
+ end
77
+
78
+ it "receive -> size(full_path)" do
79
+ @con.should_receive(:size).with('full_path').and_return("1024")
80
+ @con.receive(:action => 'size', :full_path => 'full_path').should == "1024"
81
+ end
82
+
83
+ it "receive -> processes" do
84
+ @con.should_receive(:processes).with().and_return([ { :pid => 1 } ])
85
+ @con.receive(:action => 'processes').should == YAML.dump([ { :pid => 1 } ])
86
+ end
87
+
88
+ it "receive -> process_alive" do
89
+ @con.should_receive(:process_alive).with(123).and_return(true)
90
+ @con.receive(:action => 'process_alive', :pid => 123).should == '1'
91
+ end
92
+
93
+ it "receive -> kill_process" do
94
+ @con.should_receive(:kill_process).with(123).and_return(true)
95
+ @con.receive(:action => 'kill_process', :pid => '123')
96
+ end
97
+
98
+ it "receive -> bash (foreground)" do
99
+ @con.should_receive(:bash).with('cmd', 'user', false).and_return('output')
100
+ @con.receive(:action => 'bash', :payload => 'cmd', :user => 'user', :background => 'false').should == 'output'
101
+ end
102
+
103
+ it "receive -> bash (background)" do
104
+ @con.should_receive(:bash).with('cmd', 'user', true).and_return('output')
105
+ @con.receive(:action => 'bash', :payload => 'cmd', :user => 'user', :background => 'true').should == 'output'
106
+ end
107
+
108
+ it "receive -> unknown action exception" do
109
+ lambda { @con.receive(:action => 'does_not_exist') }.should raise_error(Rush::Connection::Local::UnknownAction)
110
+ end
111
+
112
+ it "write_file writes contents to a file" do
113
+ fname = "#{@sandbox_dir}/a_file"
114
+ data = "some data"
115
+ @con.write_file(fname, data)
116
+ File.read(fname).should == data
117
+ end
118
+
119
+ it "file_contents reads a file's contents" do
120
+ fname = "#{@sandbox_dir}/a_file"
121
+ system "echo stuff > #{fname}"
122
+ @con.file_contents(fname).should == "stuff\n"
123
+ end
124
+
125
+ it "file_contents raises DoesNotExist if the file does not exist" do
126
+ fname = "#{@sandbox_dir}/does_not_exist"
127
+ lambda { @con.file_contents(fname) }.should raise_error(Rush::DoesNotExist, fname)
128
+ end
129
+
130
+ it "destroy to destroy a file or dir" do
131
+ fname = "#{@sandbox_dir}/delete_me"
132
+ system "touch #{fname}"
133
+ @con.destroy(fname)
134
+ File.exists?(fname).should be_false
135
+ end
136
+
137
+ it "purge to purge a dir" do
138
+ system "cd #{@sandbox_dir}; touch {1,2}; mkdir 3; touch 3/4"
139
+ @con.purge(@sandbox_dir)
140
+ File.exists?(@sandbox_dir).should be_true
141
+ Dir.glob("#{@sandbox_dir}/*").should == []
142
+ end
143
+
144
+ it "purge kills hidden (dotfile) entries too" do
145
+ system "cd #{@sandbox_dir}; touch .killme"
146
+ @con.purge(@sandbox_dir)
147
+ File.exists?(@sandbox_dir).should be_true
148
+ `cd #{@sandbox_dir}; ls -lA | grep -v total | wc -l`.to_i.should == 0
149
+ end
150
+
151
+ it "create_dir creates a directory" do
152
+ fname = "#{@sandbox_dir}/a/b/c/"
153
+ @con.create_dir(fname)
154
+ File.directory?(fname).should be_true
155
+ end
156
+
157
+ it "rename to rename entries within a dir" do
158
+ system "touch #{@sandbox_dir}/a"
159
+ @con.rename(@sandbox_dir, 'a', 'b')
160
+ File.exists?("#{@sandbox_dir}/a").should be_false
161
+ File.exists?("#{@sandbox_dir}/b").should be_true
162
+ end
163
+
164
+ it "copy to copy an entry to another dir on the same box" do
165
+ system "mkdir #{@sandbox_dir}/subdir"
166
+ system "touch #{@sandbox_dir}/a"
167
+ @con.copy("#{@sandbox_dir}/a", "#{@sandbox_dir}/subdir")
168
+ File.exists?("#{@sandbox_dir}/a").should be_true
169
+ File.exists?("#{@sandbox_dir}/subdir/a").should be_true
170
+ end
171
+
172
+ it "copy raises DoesNotExist with source path if it doesn't exist or otherwise can't be accessed" do
173
+ lambda { @con.copy('/does/not/exist', '/tmp') }.should raise_error(Rush::DoesNotExist, '/does/not/exist')
174
+ end
175
+
176
+ it "copy raises DoesNotExist with destination path if it can't access the destination" do
177
+ lambda { @con.copy('/tmp', '/does/not/exist') }.should raise_error(Rush::DoesNotExist, '/does/not')
178
+ end
179
+
180
+ it "read_archive to pull an archive of a dir into a byte stream" do
181
+ system "touch #{@sandbox_dir}/a"
182
+ @con.read_archive(@sandbox_dir).size.should > 50
183
+ end
184
+
185
+ it "write_archive to turn a byte stream into a dir" do
186
+ system "cd #{@sandbox_dir}; mkdir -p a; touch a/b; tar cf xfer.tar a; mkdir dst"
187
+ archive = File.read("#{@sandbox_dir}/xfer.tar")
188
+ @con.write_archive(archive, "#{@sandbox_dir}/dst")
189
+ File.directory?("#{@sandbox_dir}/dst/a").should be_true
190
+ File.exists?("#{@sandbox_dir}/dst/a/b").should be_true
191
+ end
192
+
193
+ it "index fetches list of all files and dirs in a dir when pattern is empty" do
194
+ system "cd #{@sandbox_dir}; mkdir dir; touch file"
195
+ @con.index(@sandbox_dir, '').should == [ 'dir/', 'file' ]
196
+ end
197
+
198
+ it "index fetches only files with a certain extension with a flat pattern, *.rb" do
199
+ system "cd #{@sandbox_dir}; touch a.rb; touch b.txt"
200
+ @con.index(@sandbox_dir, '*.rb').should == [ 'a.rb' ]
201
+ end
202
+
203
+ it "index raises DoesNotExist when the base path is invalid" do
204
+ lambda { @con.index('/does/not/exist', '*') }.should raise_error(Rush::DoesNotExist, '/does/not/exist')
205
+ end
206
+
207
+ it "stat gives file stats like size and timestamps" do
208
+ @con.stat(@sandbox_dir).should have_key(:ctime)
209
+ @con.stat(@sandbox_dir).should have_key(:size)
210
+ end
211
+
212
+ it "stat fetches the octal permissions" do
213
+ @con.stat(@sandbox_dir)[:mode].should be_kind_of(Fixnum)
214
+ end
215
+
216
+ it "stat raises DoesNotExist if the entry does not exist" do
217
+ fname = "#{@sandbox_dir}/does_not_exist"
218
+ lambda { @con.stat(fname) }.should raise_error(Rush::DoesNotExist, fname)
219
+ end
220
+
221
+ it "set_access invokes the access object" do
222
+ access = mock("access")
223
+ access.should_receive(:apply).with('/some/path')
224
+ @con.set_access('/some/path', access)
225
+ end
226
+
227
+ if !RUBY_PLATFORM.match(/darwin/) # doesn't work on OS X 'cause du switches are different
228
+ it "size gives size of a directory and all its contents recursively" do
229
+ system "mkdir -p #{@sandbox_dir}/a/b/; echo 1234 > #{@sandbox_dir}/a/b/c"
230
+ @con.size(@sandbox_dir).should == (4096*3 + 5)
231
+ end
232
+ end
233
+
234
+ it "parses ps output on os x" do
235
+ @con.parse_ps("21712 501 21711 1236 0 /usr/bin/vi somefile.rb").should == {
236
+ :pid => "21712",
237
+ :uid => "501",
238
+ :parent_pid => 21711,
239
+ :mem => 1236,
240
+ :cpu => 0,
241
+ :command => '/usr/bin/vi',
242
+ :cmdline => '/usr/bin/vi somefile.rb',
243
+ }
244
+ end
245
+
246
+ it "gets the list of processes on os x via the ps command" do
247
+ @con.should_receive(:os_x_raw_ps).and_return <<EOPS
248
+ PID UID PPID RSS CPU COMMAND
249
+ 1 0 1 1111 0 cmd1 args
250
+ 2 501 1 222 1 cmd2
251
+ EOPS
252
+ @con.os_x_processes.should == [
253
+ { :pid => "1", :uid => "0", :parent_pid => 1, :mem => 1111, :cpu => 0, :command => "cmd1", :cmdline => "cmd1 args" },
254
+ { :pid => "2", :uid => "501", :parent_pid => 1, :mem => 222, :cpu => 1, :command => "cmd2", :cmdline => "cmd2" },
255
+ ]
256
+ end
257
+
258
+ it "the current process should be alive" do
259
+ @con.process_alive(Process.pid).should be_true
260
+ end
261
+
262
+ it "a made-up process should not be alive" do
263
+ @con.process_alive(99999).should be_false
264
+ end
265
+
266
+ it "kills a process by pid" do
267
+ ::Process.should_receive(:kill).at_least(:once)
268
+ @con.kill_process(123)
269
+ end
270
+
271
+ it "does not raise an error if the process is already dead" do
272
+ ::Process.should_receive(:kill).and_raise(Errno::ESRCH)
273
+ lambda { @con.kill_process(123) }.should_not raise_error
274
+ end
275
+
276
+ it "executes a bash command, returning stdout when successful" do
277
+ @con.bash("echo test").should == "test\n"
278
+ end
279
+
280
+ it "executes a bash command, raising and error (with stderr as the message) when return value is nonzero" do
281
+ lambda { @con.bash("no_such_bin") }.should raise_error(Rush::BashFailed, /command not found/)
282
+ end
283
+
284
+ it "executes a bash command as another user using sudo" do
285
+ @con.bash("echo test2", ENV['USER']).should == "test2\n"
286
+ end
287
+
288
+ it "executes a bash command in the background, returning the pid" do
289
+ @con.bash("true", nil, true).should > 0
290
+ end
291
+
292
+ it "ensure_tunnel to match with remote connection" do
293
+ @con.ensure_tunnel
294
+ end
295
+
296
+ it "always returns true on alive?" do
297
+ @con.should be_alive
298
+ end
299
+
300
+ it "resolves a unix uid to a user" do
301
+ @con.resolve_unix_uid_to_user(0).should == "root"
302
+ @con.resolve_unix_uid_to_user('0').should == "root"
303
+ end
304
+
305
+ it "returns nil if the unix uid does not exist" do
306
+ @con.resolve_unix_uid_to_user(9999).should be_nil
307
+ end
308
+
309
+ it "iterates through a process list and resolves the unix uid for each" do
310
+ list = [ { :uid => 0, :command => 'pureftpd' }, { :uid => 9999, :command => 'defunk' } ]
311
+ @con.resolve_unix_uids(list).should == [ { :uid => 0, :user => 'root', :command => 'pureftpd' }, { :uid => 9999, :command => 'defunk', :user => nil } ]
312
+ end
313
+ end
@@ -0,0 +1,50 @@
1
+ require File.dirname(__FILE__) + '/base'
2
+
3
+ describe Rush::ProcessSet do
4
+ before do
5
+ @process = mock('process')
6
+ @set = Rush::ProcessSet.new([ @process ])
7
+ end
8
+
9
+ it "is Enumerable" do
10
+ @set.select { |s| s == @process }.should == [ @process ]
11
+ end
12
+
13
+ it "defines size" do
14
+ @set.size.should == 1
15
+ end
16
+
17
+ it "defines first" do
18
+ @set.first.should == @process
19
+ end
20
+
21
+ it "is equal to sets with the same contents" do
22
+ @set.should == Rush::ProcessSet.new([ @process ])
23
+ end
24
+
25
+ it "is equal to arrays with the same contents" do
26
+ @set.should == [ @process ]
27
+ end
28
+
29
+ it "kills all processes in the set" do
30
+ @process.should_receive(:kill)
31
+ @set.kill
32
+ end
33
+
34
+ it "checks the alive? state of all processes in the set" do
35
+ @process.should_receive(:alive?).and_return(true)
36
+ @set.alive?.should == [ true ]
37
+ end
38
+
39
+ it "filters the set from a conditions hash and returns the filtered set" do
40
+ @process.stub!(:pid).and_return(123)
41
+ @set.filter(:pid => 123).first.should == @process
42
+ @set.filter(:pid => 456).size.should == 0
43
+ end
44
+
45
+ it "filters with regexps if provided in the conditions" do
46
+ @process.stub!(:command).and_return('foobaz')
47
+ @set.filter(:command => /baz/).first.should == @process
48
+ @set.filter(:command => /blerg/).size.should == 0
49
+ end
50
+ end
@@ -0,0 +1,73 @@
1
+ require File.dirname(__FILE__) + '/base'
2
+
3
+ describe Rush::Process do
4
+ before do
5
+ @pid = fork do
6
+ sleep 999
7
+ end
8
+ @process = Rush::Process.all.detect { |p| p.pid == @pid }
9
+ end
10
+
11
+ after do
12
+ system "kill -9 #{@pid}"
13
+ end
14
+
15
+ if !RUBY_PLATFORM.match(/darwin/) # OS x reports pids weird
16
+ it "knows all its child processes" do
17
+ parent = Rush::Process.all.detect { |p| p.pid == Process.pid }
18
+ parent.children.should == [ @process ]
19
+ end
20
+ end
21
+
22
+ it "gets the list of all processes" do
23
+ list = Rush::Process.all
24
+ list.size.should > 5
25
+ list.first.should be_kind_of(Rush::Process)
26
+ end
27
+
28
+ it "knows the pid" do
29
+ @process.pid.should == @pid
30
+ end
31
+
32
+ it "knows the uid" do
33
+ @process.uid.should == ::Process.uid
34
+ end
35
+
36
+ it "knows the executed binary" do
37
+ @process.command.should match(/^ruby/)
38
+ end
39
+
40
+ it "knows the command line" do
41
+ @process.cmdline.should match(/process_spec.rb/)
42
+ end
43
+
44
+ it "knows the memory used" do
45
+ @process.mem.should > 0
46
+ end
47
+
48
+ it "knows the cpu used" do
49
+ @process.cpu.should >= 0
50
+ end
51
+
52
+ it "knows the parent process pid" do
53
+ @process.parent_pid.should == Process.pid
54
+ end
55
+
56
+ it "knows the parent process" do
57
+ this = Rush::Box.new.processes.select { |p| p.pid == Process.pid }.first
58
+ @process.parent.should == this
59
+ end
60
+
61
+ it "can kill itself" do
62
+ system "sleep 30 &"
63
+ process = Rush::Process.all.detect { |p| p.command == "sleep" }
64
+ process.kill
65
+ sleep 0.1
66
+ process.alive?.should be_false
67
+ end
68
+
69
+ it "if box and pid are the same, process is equal" do
70
+ other = Rush::Process.new({ :pid => @process.pid }, @process.box)
71
+ @process.should == other
72
+ end
73
+ end
@@ -0,0 +1,135 @@
1
+ require File.dirname(__FILE__) + '/base'
2
+
3
+ describe Rush::Connection::Local do
4
+ before do
5
+ @sandbox_dir = "/tmp/rush_spec.#{Process.pid}"
6
+ system "rm -rf #{@sandbox_dir}; mkdir -p #{@sandbox_dir}"
7
+
8
+ @con = Rush::Connection::Remote.new('spec.example.com')
9
+ end
10
+
11
+ after do
12
+ system "rm -rf #{@sandbox_dir}"
13
+ end
14
+
15
+ it "transmits write_file" do
16
+ @con.should_receive(:transmit).with(:action => 'write_file', :full_path => 'file', :payload => 'contents')
17
+ @con.write_file('file', 'contents')
18
+ end
19
+
20
+ it "transmits file_contents" do
21
+ @con.should_receive(:transmit).with(:action => 'file_contents', :full_path => 'file').and_return('contents')
22
+ @con.file_contents('file').should == 'contents'
23
+ end
24
+
25
+ it "transmits destroy" do
26
+ @con.should_receive(:transmit).with(:action => 'destroy', :full_path => 'file')
27
+ @con.destroy('file')
28
+ end
29
+
30
+ it "transmits purge" do
31
+ @con.should_receive(:transmit).with(:action => 'purge', :full_path => 'dir')
32
+ @con.purge('dir')
33
+ end
34
+
35
+ it "transmits create_dir" do
36
+ @con.should_receive(:transmit).with(:action => 'create_dir', :full_path => 'file')
37
+ @con.create_dir('file')
38
+ end
39
+
40
+ it "transmits rename" do
41
+ @con.should_receive(:transmit).with(:action => 'rename', :path => 'path', :name => 'name', :new_name => 'new_name')
42
+ @con.rename('path', 'name', 'new_name')
43
+ end
44
+
45
+ it "transmits copy" do
46
+ @con.should_receive(:transmit).with(:action => 'copy', :src => 'src', :dst => 'dst')
47
+ @con.copy('src', 'dst')
48
+ end
49
+
50
+ it "transmits read_archive" do
51
+ @con.should_receive(:transmit).with(:action => 'read_archive', :full_path => 'full_path').and_return('archive data')
52
+ @con.read_archive('full_path').should == 'archive data'
53
+ end
54
+
55
+ it "transmits write_archive" do
56
+ @con.should_receive(:transmit).with(:action => 'write_archive', :dir => 'dir', :payload => 'archive')
57
+ @con.write_archive('archive', 'dir')
58
+ end
59
+
60
+ it "transmits index" do
61
+ @con.should_receive(:transmit).with(:action => 'index', :base_path => 'base_path', :glob => '*').and_return("1\n2\n")
62
+ @con.index('base_path', '*').should == %w(1 2)
63
+ end
64
+
65
+ it "transmits stat" do
66
+ @con.should_receive(:transmit).with(:action => 'stat', :full_path => 'full_path').and_return(YAML.dump(1 => 2))
67
+ @con.stat('full_path').should == { 1 => 2 }
68
+ end
69
+
70
+ it "transmits set_access" do
71
+ @con.should_receive(:transmit).with(:action => 'set_access', :full_path => 'full_path', :user => 'joe', :user_read => 1)
72
+ @con.set_access('full_path', :user => 'joe', :user_read => 1)
73
+ end
74
+
75
+ it "transmits size" do
76
+ @con.should_receive(:transmit).with(:action => 'size', :full_path => 'full_path').and_return("123")
77
+ @con.size('full_path').should == 123
78
+ end
79
+
80
+ it "transmits processes" do
81
+ @con.should_receive(:transmit).with(:action => 'processes').and_return(YAML.dump([ { :pid => 1 } ]))
82
+ @con.processes.should == [ { :pid => 1 } ]
83
+ end
84
+
85
+ it "transmits process_alive" do
86
+ @con.should_receive(:transmit).with(:action => 'process_alive', :pid => 123).and_return(true)
87
+ @con.process_alive(123).should == true
88
+ end
89
+
90
+ it "transmits kill_process" do
91
+ @con.should_receive(:transmit).with(:action => 'kill_process', :pid => 123)
92
+ @con.kill_process(123)
93
+ end
94
+
95
+ it "transmits bash" do
96
+ @con.should_receive(:transmit).with(:action => 'bash', :payload => 'cmd', :user => 'user', :background => 'bg').and_return('output')
97
+ @con.bash('cmd', 'user', 'bg').should == 'output'
98
+ end
99
+
100
+ it "an http result code of 401 raises NotAuthorized" do
101
+ lambda { @con.process_result("401", "") }.should raise_error(Rush::NotAuthorized)
102
+ end
103
+
104
+ it "an http result code of 400 raises the exception passed in the result body" do
105
+ @con.stub!(:parse_exception).and_return(Rush::DoesNotExist, "message")
106
+ lambda { @con.process_result("400", "") }.should raise_error(Rush::DoesNotExist)
107
+ end
108
+
109
+ it "an http result code of 501 (or anything other than the other defined codes) raises FailedTransmit" do
110
+ lambda { @con.process_result("501", "") }.should raise_error(Rush::FailedTransmit)
111
+ end
112
+
113
+ it "parse_exception takes the class from the first line and the message from the second" do
114
+ @con.parse_exception("Rush::DoesNotExist\nthe message\n").should == [ Rush::DoesNotExist, "the message" ]
115
+ end
116
+
117
+ it "parse_exception rejects unrecognized exceptions" do
118
+ lambda { @con.parse_exception("NotARushException\n") }.should raise_error
119
+ end
120
+
121
+ it "passes through ensure_tunnel" do
122
+ @con.tunnel.should_receive(:ensure_tunnel)
123
+ @con.ensure_tunnel
124
+ end
125
+
126
+ it "is alive if the box is responding to commands" do
127
+ @con.should_receive(:index).and_return(:dummy)
128
+ @con.should be_alive
129
+ end
130
+
131
+ it "not alive if an attempted command throws an exception" do
132
+ @con.should_receive(:index).and_raise(RuntimeError)
133
+ @con.should_not be_alive
134
+ end
135
+ end