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.
- data/Rakefile +65 -0
- data/bin/rush +13 -0
- data/bin/rushd +7 -0
- data/lib/rush.rb +27 -0
- data/lib/rush/access.rb +130 -0
- data/lib/rush/array_ext.rb +19 -0
- data/lib/rush/box.rb +112 -0
- data/lib/rush/commands.rb +55 -0
- data/lib/rush/config.rb +154 -0
- data/lib/rush/dir.rb +158 -0
- data/lib/rush/embeddable_shell.rb +26 -0
- data/lib/rush/entry.rb +178 -0
- data/lib/rush/exceptions.rb +31 -0
- data/lib/rush/file.rb +77 -0
- data/lib/rush/find_by.rb +39 -0
- data/lib/rush/fixnum_ext.rb +18 -0
- data/lib/rush/head_tail.rb +11 -0
- data/lib/rush/local.rb +374 -0
- data/lib/rush/process.rb +55 -0
- data/lib/rush/process_set.rb +62 -0
- data/lib/rush/remote.rb +152 -0
- data/lib/rush/search_results.rb +58 -0
- data/lib/rush/server.rb +117 -0
- data/lib/rush/shell.rb +148 -0
- data/lib/rush/ssh_tunnel.rb +122 -0
- data/lib/rush/string_ext.rb +3 -0
- data/spec/access_spec.rb +134 -0
- data/spec/array_ext_spec.rb +15 -0
- data/spec/base.rb +24 -0
- data/spec/box_spec.rb +64 -0
- data/spec/commands_spec.rb +47 -0
- data/spec/config_spec.rb +108 -0
- data/spec/dir_spec.rb +159 -0
- data/spec/embeddable_shell_spec.rb +17 -0
- data/spec/entry_spec.rb +129 -0
- data/spec/file_spec.rb +79 -0
- data/spec/find_by_spec.rb +58 -0
- data/spec/fixnum_ext_spec.rb +19 -0
- data/spec/local_spec.rb +313 -0
- data/spec/process_set_spec.rb +50 -0
- data/spec/process_spec.rb +73 -0
- data/spec/remote_spec.rb +135 -0
- data/spec/search_results_spec.rb +44 -0
- data/spec/shell_spec.rb +12 -0
- data/spec/ssh_tunnel_spec.rb +122 -0
- data/spec/string_ext_spec.rb +23 -0
- metadata +126 -0
data/spec/local_spec.rb
ADDED
@@ -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
|
data/spec/remote_spec.rb
ADDED
@@ -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
|