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,39 @@
1
+ # Generic find_by (returns first match) and find_all_by (returns all matches)
2
+ # against arrays.
3
+ #
4
+ # Examples:
5
+ #
6
+ # processes.find_by_pid(::Process.pid)
7
+ # processes.find_all_by_cmdline(/mongrel_rails/)
8
+ #
9
+ module Rush::FindBy
10
+ def method_missing(meth, *args)
11
+ if m = meth.to_s.match(/^find_by_([a-z_]+)$/)
12
+ find_by(m[1], args.first)
13
+ elsif m = meth.to_s.match(/^find_all_by_([a-z_]+)$/)
14
+ find_all_by(m[1], args.first)
15
+ else
16
+ super
17
+ end
18
+ end
19
+
20
+ def find_by(field, arg)
21
+ detect do |item|
22
+ item.respond_to?(field) and compare_or_match(item.send(field), arg)
23
+ end
24
+ end
25
+
26
+ def find_all_by(field, arg)
27
+ select do |item|
28
+ item.respond_to?(field) and compare_or_match(item.send(field), arg)
29
+ end
30
+ end
31
+
32
+ def compare_or_match(value, against)
33
+ if against.class == Regexp
34
+ value.match(against) ? true : false
35
+ else
36
+ value == against
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,18 @@
1
+ # Integer extensions for file and dir sizes (returned in bytes).
2
+ #
3
+ # Example:
4
+ #
5
+ # box['/assets/'].files_flattened.select { |f| f.size > 10.mb }
6
+ class Fixnum
7
+ def kb
8
+ self * 1024
9
+ end
10
+
11
+ def mb
12
+ kb * 1024
13
+ end
14
+
15
+ def gb
16
+ mb * 1024
17
+ end
18
+ end
@@ -0,0 +1,11 @@
1
+ # Mixin for array and string for methods I wish they had.
2
+ module Rush::HeadTail
3
+ def head(n)
4
+ slice(0, n)
5
+ end
6
+
7
+ def tail(n)
8
+ n = [ n, length ].min
9
+ slice(-n, n)
10
+ end
11
+ end
data/lib/rush/local.rb ADDED
@@ -0,0 +1,374 @@
1
+ require 'fileutils'
2
+ require 'yaml'
3
+
4
+ # Rush::Box uses a connection object to execute all rush commands. If the box
5
+ # is local, Rush::Connection::Local is created. The local connection is the
6
+ # heart of rush's internals. (Users of the rush shell or library need never
7
+ # access the connection object directly, so the docs herein are intended for
8
+ # developers wishing to modify rush.)
9
+ #
10
+ # The local connection has a series of methods which do the actual work of
11
+ # modifying files, getting process lists, and so on. RushServer creates a
12
+ # local connection to handle incoming requests; the translation from a raw hash
13
+ # of parameters to an executed method is handled by
14
+ # Rush::Connection::Local#receive.
15
+ class Rush::Connection::Local
16
+ # Write raw bytes to a file.
17
+ def write_file(full_path, contents)
18
+ ::File.open(full_path, 'w') do |f|
19
+ f.write contents
20
+ end
21
+ true
22
+ end
23
+
24
+ # Read raw bytes from a file.
25
+ def file_contents(full_path)
26
+ ::File.read(full_path)
27
+ rescue Errno::ENOENT
28
+ raise Rush::DoesNotExist, full_path
29
+ end
30
+
31
+ # Destroy a file or dir.
32
+ def destroy(full_path)
33
+ raise "No." if full_path == '/'
34
+ FileUtils.rm_rf(full_path)
35
+ true
36
+ end
37
+
38
+ # Purge the contents of a dir.
39
+ def purge(full_path)
40
+ raise "No." if full_path == '/'
41
+ Dir.chdir(full_path) do
42
+ all = Dir.glob("*", File::FNM_DOTMATCH).reject { |f| f == '.' or f == '..' }
43
+ FileUtils.rm_rf all
44
+ end
45
+ true
46
+ end
47
+
48
+ # Create a dir.
49
+ def create_dir(full_path)
50
+ FileUtils.mkdir_p(full_path)
51
+ true
52
+ end
53
+
54
+ # Rename an entry within a dir.
55
+ def rename(path, name, new_name)
56
+ raise(Rush::NameCannotContainSlash, "#{path} rename #{name} to #{new_name}") if new_name.match(/\//)
57
+ old_full_path = "#{path}/#{name}"
58
+ new_full_path = "#{path}/#{new_name}"
59
+ raise(Rush::NameAlreadyExists, "#{path} rename #{name} to #{new_name}") if ::File.exists?(new_full_path)
60
+ FileUtils.mv(old_full_path, new_full_path)
61
+ true
62
+ end
63
+
64
+ # Copy ane entry from one path to another.
65
+ def copy(src, dst)
66
+ FileUtils.cp_r(src, dst)
67
+ true
68
+ rescue Errno::ENOENT
69
+ raise Rush::DoesNotExist, File.dirname(dst)
70
+ rescue RuntimeError
71
+ raise Rush::DoesNotExist, src
72
+ end
73
+
74
+ # Create an in-memory archive (tgz) of a file or dir, which can be
75
+ # transmitted to another server for a copy or move. Note that archive
76
+ # operations have the dir name implicit in the archive.
77
+ def read_archive(full_path)
78
+ `cd #{::File.dirname(full_path)}; tar c #{::File.basename(full_path)}`
79
+ end
80
+
81
+ # Extract an in-memory archive to a dir.
82
+ def write_archive(archive, dir)
83
+ IO.popen("cd #{dir}; tar x", "w") do |p|
84
+ p.write archive
85
+ end
86
+ end
87
+
88
+ # Get an index of files from the given path with the glob. Could return
89
+ # nested values if the glob contains a doubleglob. The return value is an
90
+ # array of full paths, with directories listed first.
91
+ def index(base_path, glob)
92
+ glob = '*' if glob == '' or glob.nil?
93
+ dirs = []
94
+ files = []
95
+ ::Dir.chdir(base_path) do
96
+ ::Dir.glob(glob).each do |fname|
97
+ if ::File.directory?(fname)
98
+ dirs << fname + '/'
99
+ else
100
+ files << fname
101
+ end
102
+ end
103
+ end
104
+ dirs.sort + files.sort
105
+ rescue Errno::ENOENT
106
+ raise Rush::DoesNotExist, base_path
107
+ end
108
+
109
+ # Fetch stats (size, ctime, etc) on an entry. Size will not be accurate for dirs.
110
+ def stat(full_path)
111
+ s = ::File.stat(full_path)
112
+ {
113
+ :size => s.size,
114
+ :ctime => s.ctime,
115
+ :atime => s.atime,
116
+ :mtime => s.mtime,
117
+ :mode => s.mode
118
+ }
119
+ rescue Errno::ENOENT
120
+ raise Rush::DoesNotExist, full_path
121
+ end
122
+
123
+ def set_access(full_path, access)
124
+ access.apply(full_path)
125
+ end
126
+
127
+ # Fetch the size of a dir, since a standard file stat does not include the
128
+ # size of the contents.
129
+ def size(full_path)
130
+ `du -sb #{full_path}`.match(/(\d+)/)[1].to_i
131
+ end
132
+
133
+ # Get the list of processes as an array of hashes.
134
+ def processes
135
+ if ::File.directory? "/proc"
136
+ resolve_unix_uids(linux_processes)
137
+ elsif ::File.directory? "C:/WINDOWS"
138
+ windows_processes
139
+ else
140
+ os_x_processes
141
+ end
142
+ end
143
+
144
+ # Process list on Linux using /proc.
145
+ def linux_processes
146
+ list = []
147
+ ::Dir["/proc/*/stat"].select { |file| file =~ /\/proc\/\d+\// }.each do |file|
148
+ begin
149
+ list << read_proc_file(file)
150
+ rescue
151
+ # process died between the dir listing and accessing the file
152
+ end
153
+ end
154
+ list
155
+ end
156
+
157
+ def resolve_unix_uids(list)
158
+ @uid_map = {} # reset the cache between uid resolutions.
159
+ list.each do |process|
160
+ process[:user] = resolve_unix_uid_to_user(process[:uid])
161
+ end
162
+ list
163
+ end
164
+
165
+ # resolve uid to user
166
+ def resolve_unix_uid_to_user(uid)
167
+ require 'etc'
168
+
169
+ @uid_map ||= {}
170
+ uid = uid.to_i
171
+
172
+ return @uid_map[uid] if !@uid_map[uid].nil?
173
+
174
+ begin
175
+ record = Etc.getpwuid(uid)
176
+ rescue ArgumentError
177
+ return nil
178
+ end
179
+
180
+ @uid_map[uid] = record.name
181
+ @uid_map[uid]
182
+ end
183
+
184
+ # Read a single file in /proc and store the parsed values in a hash suitable
185
+ # for use in the Rush::Process#new.
186
+ def read_proc_file(file)
187
+ data = ::File.read(file).split(" ")
188
+ uid = ::File.stat(file).uid
189
+ pid = data[0]
190
+ command = data[1].match(/^\((.*)\)$/)[1]
191
+ cmdline = ::File.read("/proc/#{pid}/cmdline").gsub(/\0/, ' ')
192
+ parent_pid = data[3].to_i
193
+ utime = data[13].to_i
194
+ ktime = data[14].to_i
195
+ vss = data[22].to_i / 1024
196
+ rss = data[23].to_i * 4
197
+ time = utime + ktime
198
+
199
+ {
200
+ :pid => pid,
201
+ :uid => uid,
202
+ :command => command,
203
+ :cmdline => cmdline,
204
+ :parent_pid => parent_pid,
205
+ :mem => rss,
206
+ :cpu => time,
207
+ }
208
+ end
209
+
210
+ # Process list on OS X or other unixes without a /proc.
211
+ def os_x_processes
212
+ raw = os_x_raw_ps.split("\n").slice(1, 99999)
213
+ raw.map do |line|
214
+ parse_ps(line)
215
+ end
216
+ end
217
+
218
+ # ps command used to generate list of processes on non-/proc unixes.
219
+ def os_x_raw_ps
220
+ `COLUMNS=9999 ps ax -o "pid uid ppid rss cpu command"`
221
+ end
222
+
223
+ # Parse a single line of the ps command and return the values in a hash
224
+ # suitable for use in the Rush::Process#new.
225
+ def parse_ps(line)
226
+ m = line.split(" ", 6)
227
+ params = {}
228
+ params[:pid] = m[0]
229
+ params[:uid] = m[1]
230
+ params[:parent_pid] = m[2].to_i
231
+ params[:mem] = m[3].to_i
232
+ params[:cpu] = m[4].to_i
233
+ params[:cmdline] = m[5]
234
+ params[:command] = params[:cmdline].split(" ").first
235
+ params
236
+ end
237
+
238
+ # Process list on Windows.
239
+ def windows_processes
240
+ require 'win32ole'
241
+ wmi = WIN32OLE.connect("winmgmts://")
242
+ wmi.ExecQuery("select * from win32_process").map do |proc_info|
243
+ parse_oleprocinfo(proc_info)
244
+ end
245
+ end
246
+
247
+ # Parse the Windows OLE process info.
248
+ def parse_oleprocinfo(proc_info)
249
+ command = proc_info.Name
250
+ pid = proc_info.ProcessId
251
+ uid = 0
252
+ cmdline = proc_info.CommandLine
253
+ rss = proc_info.MaximumWorkingSetSize
254
+ time = proc_info.KernelModeTime.to_i + proc_info.UserModeTime.to_i
255
+
256
+ {
257
+ :pid => pid,
258
+ :uid => uid,
259
+ :command => command,
260
+ :cmdline => cmdline,
261
+ :mem => rss,
262
+ :cpu => time,
263
+ }
264
+ end
265
+
266
+ # Returns true if the specified pid is running.
267
+ def process_alive(pid)
268
+ ::Process.kill(0, pid)
269
+ true
270
+ rescue Errno::ESRCH
271
+ false
272
+ end
273
+
274
+ # Terminate a process, by pid.
275
+ def kill_process(pid)
276
+ ::Process.kill('TERM', pid)
277
+
278
+ # keep trying until it's dead (technique borrowed from god)
279
+ 5.times do
280
+ return if !process_alive(pid)
281
+ sleep 0.5
282
+ ::Process.kill('TERM', pid) rescue nil
283
+ end
284
+
285
+ ::Process.kill('KILL', pid) rescue nil
286
+
287
+ rescue Errno::ESRCH
288
+ # if it's dead, great - do nothing
289
+ end
290
+
291
+ def bash(command, user=nil, background=false)
292
+ return bash_background(command, user) if background
293
+
294
+ require 'session'
295
+
296
+ sh = Session::Bash.new
297
+
298
+ if user and user != ""
299
+ out, err = sh.execute "cd /; sudo -H -u #{user} bash", :stdin => command
300
+ else
301
+ out, err = sh.execute command
302
+ end
303
+
304
+ retval = sh.status
305
+ sh.close!
306
+
307
+ raise(Rush::BashFailed, err) if retval != 0
308
+
309
+ out
310
+ end
311
+
312
+ def bash_background(command, user)
313
+ inpipe, outpipe = IO.pipe
314
+
315
+ pid = fork do
316
+ outpipe.write command
317
+ outpipe.close
318
+ STDIN.reopen(inpipe)
319
+
320
+ if user and user != ''
321
+ exec "cd /; sudo -H -u #{user} bash"
322
+ else
323
+ exec "bash"
324
+ end
325
+ end
326
+ outpipe.close
327
+
328
+ Process::detach pid
329
+
330
+ pid
331
+ end
332
+
333
+ ####################################
334
+
335
+ # Raised when the action passed in by RushServer is not known.
336
+ class UnknownAction < Exception; end
337
+
338
+ # RushServer uses this method to transform a hash (:action plus parameters
339
+ # specific to that action type) into a method call on the connection. The
340
+ # returned value must be text so that it can be transmitted across the wire
341
+ # as an HTTP response.
342
+ def receive(params)
343
+ case params[:action]
344
+ when 'write_file' then write_file(params[:full_path], params[:payload])
345
+ when 'file_contents' then file_contents(params[:full_path])
346
+ when 'destroy' then destroy(params[:full_path])
347
+ when 'purge' then purge(params[:full_path])
348
+ when 'create_dir' then create_dir(params[:full_path])
349
+ when 'rename' then rename(params[:path], params[:name], params[:new_name])
350
+ when 'copy' then copy(params[:src], params[:dst])
351
+ when 'read_archive' then read_archive(params[:full_path])
352
+ when 'write_archive' then write_archive(params[:payload], params[:dir])
353
+ when 'index' then index(params[:base_path], params[:glob]).join("\n") + "\n"
354
+ when 'stat' then YAML.dump(stat(params[:full_path]))
355
+ when 'set_access' then set_access(params[:full_path], Rush::Access.from_hash(params))
356
+ when 'size' then size(params[:full_path])
357
+ when 'processes' then YAML.dump(processes)
358
+ when 'process_alive' then process_alive(params[:pid]) ? '1' : '0'
359
+ when 'kill_process' then kill_process(params[:pid].to_i)
360
+ when 'bash' then bash(params[:payload], params[:user], params[:background] == 'true')
361
+ else
362
+ raise UnknownAction
363
+ end
364
+ end
365
+
366
+ # No-op for duck typing with remote connection.
367
+ def ensure_tunnel(options={})
368
+ end
369
+
370
+ # Local connections are always alive.
371
+ def alive?
372
+ true
373
+ end
374
+ end
@@ -0,0 +1,55 @@
1
+ # An array of these objects is returned by Rush::Box#processes.
2
+ class Rush::Process
3
+ attr_reader :box, :pid, :uid, :parent_pid, :command, :cmdline, :mem, :cpu, :user
4
+
5
+ # params is a hash returned by the system-specific method of looking up the
6
+ # process list.
7
+ def initialize(params, box)
8
+ @box = box
9
+
10
+ @pid = params[:pid].to_i
11
+ @uid = params[:uid].to_i
12
+ @user = params[:user]
13
+ @command = params[:command]
14
+ @cmdline = params[:cmdline]
15
+ @mem = params[:mem]
16
+ @cpu = params[:cpu]
17
+ @parent_pid = params[:parent_pid]
18
+ end
19
+
20
+ def to_s # :nodoc:
21
+ inspect
22
+ end
23
+
24
+ def inspect # :nodoc:
25
+ "#{box} process #{@pid}: #{@cmdline}"
26
+ end
27
+
28
+ # Returns the Rush::Process parent of this process.
29
+ def parent
30
+ box.processes.select { |p| p.pid == parent_pid }.first
31
+ end
32
+
33
+ # Returns an array of child processes owned by this process.
34
+ def children
35
+ box.processes.select { |p| p.parent_pid == pid }
36
+ end
37
+
38
+ # Returns true if the process is currently running.
39
+ def alive?
40
+ box.connection.process_alive(pid)
41
+ end
42
+
43
+ # Terminate the process.
44
+ def kill
45
+ box.connection.kill_process(pid)
46
+ end
47
+
48
+ def ==(other) # :nodoc:
49
+ pid == other.pid and box == other.box
50
+ end
51
+
52
+ def self.all
53
+ Rush::Box.new('localhost').processes
54
+ end
55
+ end