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
data/lib/rush/dir.rb ADDED
@@ -0,0 +1,158 @@
1
+ # A dir is a subclass of Rush::Entry that contains other entries. Also known
2
+ # as a directory or a folder.
3
+ #
4
+ # Dirs can be operated on with Rush::Commands the same as an array of files.
5
+ # They also offer a square bracket accessor which can use globbing to get a
6
+ # list of files.
7
+ #
8
+ # Example:
9
+ #
10
+ # dir = box['/home/adam/']
11
+ # dir['**/*.rb'].line_count
12
+ #
13
+ # In the interactive shell, dir.ls is a useful command.
14
+ class Rush::Dir < Rush::Entry
15
+ def dir?
16
+ true
17
+ end
18
+
19
+ def full_path
20
+ "#{super}/"
21
+ end
22
+
23
+ # Entries contained within this dir - not recursive.
24
+ def contents
25
+ find_by_glob('*')
26
+ end
27
+
28
+ # Files contained in this dir only.
29
+ def files
30
+ contents.select { |entry| !entry.dir? }
31
+ end
32
+
33
+ # Other dirs contained in this dir only.
34
+ def dirs
35
+ contents.select { |entry| entry.dir? }
36
+ end
37
+
38
+ # Access subentries with square brackets, e.g. dir['subdir/file']
39
+ def [](key)
40
+ key = key.to_s
41
+ if key == '**'
42
+ files_flattened
43
+ elsif key.match(/\*/)
44
+ find_by_glob(key)
45
+ else
46
+ find_by_name(key)
47
+ end
48
+ end
49
+
50
+ def find_by_name(name) # :nodoc:
51
+ Rush::Entry.factory("#{full_path}/#{name}", box)
52
+ end
53
+
54
+ def find_by_glob(glob) # :nodoc:
55
+ connection.index(full_path, glob).map do |fname|
56
+ Rush::Entry.factory("#{full_path}/#{fname}", box)
57
+ end
58
+ end
59
+
60
+ # A list of all the recursively contained entries in flat form.
61
+ def entries_tree
62
+ find_by_glob('**/*')
63
+ end
64
+
65
+ # Recursively contained files.
66
+ def files_flattened
67
+ entries_tree.select { |e| !e.dir? }
68
+ end
69
+
70
+ # Recursively contained dirs.
71
+ def dirs_flattened
72
+ entries_tree.select { |e| e.dir? }
73
+ end
74
+
75
+ # Given a list of flat filenames, product a list of entries under this dir.
76
+ # Mostly for internal use.
77
+ def make_entries(filenames)
78
+ filenames.map do |fname|
79
+ Rush::Entry.factory("#{full_path}/#{fname}")
80
+ end
81
+ end
82
+
83
+ # Create a blank file within this dir.
84
+ def create_file(name)
85
+ file = self[name].create
86
+ file.write('')
87
+ file
88
+ end
89
+
90
+ # Create an empty subdir within this dir.
91
+ def create_dir(name)
92
+ name += '/' unless name.tail(1) == '/'
93
+ self[name].create
94
+ end
95
+
96
+ # Create an instantiated but not yet filesystem-created dir.
97
+ def create
98
+ connection.create_dir(full_path)
99
+ self
100
+ end
101
+
102
+ # Get the total disk usage of the dir and all its contents.
103
+ def size
104
+ connection.size(full_path)
105
+ end
106
+
107
+ # Contained dirs that are not hidden.
108
+ def nonhidden_dirs
109
+ dirs.select do |dir|
110
+ !dir.hidden?
111
+ end
112
+ end
113
+
114
+ # Contained files that are not hidden.
115
+ def nonhidden_files
116
+ files.select do |file|
117
+ !file.hidden?
118
+ end
119
+ end
120
+
121
+ # Run a bash command starting in this directory. Options are the same as Rush::Box#bash.
122
+ def bash(command, options={})
123
+ box.bash "cd #{full_path} && #{command}", options
124
+ end
125
+
126
+ # Destroy all of the contents of the directory, leaving it fresh and clean.
127
+ def purge
128
+ connection.purge full_path
129
+ end
130
+
131
+ # Text output of dir listing, equivalent to the regular unix shell's ls command.
132
+ def ls
133
+ out = [ "#{self}" ]
134
+ nonhidden_dirs.each do |dir|
135
+ out << " #{dir.name}/"
136
+ end
137
+ nonhidden_files.each do |file|
138
+ out << " #{file.name}"
139
+ end
140
+ out.join("\n")
141
+ end
142
+
143
+ # Run rake within this dir.
144
+ def rake(*args)
145
+ bash "rake #{args.join(' ')}"
146
+ end
147
+
148
+ # Run git within this dir.
149
+ def git(*args)
150
+ bash "git #{args.join(' ')}"
151
+ end
152
+
153
+ include Rush::Commands
154
+
155
+ def entries
156
+ contents
157
+ end
158
+ end
@@ -0,0 +1,26 @@
1
+ require 'rush/shell'
2
+
3
+ module Rush
4
+ # This is a class that can be embedded in other applications
5
+ # rake tasks, utility scripts, etc
6
+ #
7
+ # Delegates unknown method calls to a Rush::Shell instance
8
+ class EmbeddableShell
9
+ attr_accessor :shell
10
+ def initialize(suppress_output = true)
11
+ self.shell = Rush::Shell.new
12
+ shell.suppress_output = suppress_output
13
+ end
14
+
15
+ # evalutes and unkown method call agains the rush shell
16
+ def method_missing(sym, *args, &block)
17
+ shell.execute sym.to_s
18
+ $last_res
19
+ end
20
+
21
+ # take a whole block and execute it as if it were inside a shell
22
+ def execute_in_shell(&block)
23
+ self.instance_eval(&block)
24
+ end
25
+ end
26
+ end
data/lib/rush/entry.rb ADDED
@@ -0,0 +1,178 @@
1
+ # Rush::Entry is the base class for Rush::File and Rush::Dir. One or more of
2
+ # these is instantiated whenever you use square brackets to access the
3
+ # filesystem on a box, as well as any other operation that returns an entry or
4
+ # list of entries.
5
+ class Rush::Entry
6
+ attr_reader :box, :name, :path
7
+
8
+ # Initialize with full path to the file or dir, and the box it resides on.
9
+ def initialize(full_path, box=nil)
10
+ full_path = ::File.expand_path(full_path, '/')
11
+ @path = ::File.dirname(full_path)
12
+ @name = ::File.basename(full_path)
13
+ @box = box || Rush::Box.new('localhost')
14
+ end
15
+
16
+ # The factory checks to see if the full path has a trailing slash for
17
+ # creating a Rush::Dir rather than the default Rush::File.
18
+ def self.factory(full_path, box=nil)
19
+ if full_path.tail(1) == '/'
20
+ Rush::Dir.new(full_path, box)
21
+ else
22
+ Rush::File.new(full_path, box)
23
+ end
24
+ end
25
+
26
+ def to_s # :nodoc:
27
+ if box.host == 'localhost'
28
+ "#{full_path}"
29
+ else
30
+ inspect
31
+ end
32
+ end
33
+
34
+ def inspect # :nodoc:
35
+ "#{box}:#{full_path}"
36
+ end
37
+
38
+ def connection
39
+ box ? box.connection : Rush::Connection::Local.new
40
+ end
41
+
42
+ # The parent dir. For example, box['/etc/hosts'].parent == box['etc/']
43
+ def parent
44
+ @parent ||= Rush::Dir.new(@path)
45
+ end
46
+
47
+ def full_path
48
+ "#{@path}/#{@name}"
49
+ end
50
+
51
+ # Return true if the entry currently exists on the filesystem of the box.
52
+ def exists?
53
+ stat
54
+ true
55
+ rescue Rush::DoesNotExist
56
+ false
57
+ end
58
+
59
+ # Timestamp of entry creation.
60
+ def created_at
61
+ stat[:ctime]
62
+ end
63
+
64
+ # Timestamp that entry was last modified on.
65
+ def last_modified
66
+ stat[:mtime]
67
+ end
68
+
69
+ # Timestamp that entry was last accessed on.
70
+ def last_accessed
71
+ stat[:atime]
72
+ end
73
+
74
+ # Attempts to rename, copy, or otherwise place an entry into a dir that already contains an entry by that name will fail with this exception.
75
+ class NameAlreadyExists < Exception; end
76
+
77
+ # Do not use rename or duplicate with a slash; use copy_to or move_to instead.
78
+ class NameCannotContainSlash < Exception; end
79
+
80
+ # Rename an entry to another name within the same dir. The object's name
81
+ # will be updated to match the change on the filesystem.
82
+ def rename(new_name)
83
+ connection.rename(@path, @name, new_name)
84
+ @name = new_name
85
+ end
86
+
87
+ # Rename an entry to another name within the same dir. The existing object
88
+ # will not be affected, but a new object representing the newly-created
89
+ # entry will be returned.
90
+ def duplicate(new_name)
91
+ raise NameCannotContainSlash if new_name.match(/\//)
92
+ new_full_path = "#{@path}/#{new_name}"
93
+ connection.copy(full_path, new_full_path)
94
+ self.class.new(new_full_path, box)
95
+ end
96
+
97
+ # Copy the entry to another dir. Returns an object representing the new
98
+ # copy.
99
+ def copy_to(dir)
100
+ raise NotADir unless dir.class == Rush::Dir
101
+
102
+ if box == dir.box
103
+ connection.copy(full_path, dir.full_path)
104
+ else
105
+ archive = connection.read_archive(full_path)
106
+ dir.box.connection.write_archive(archive, dir.full_path)
107
+ end
108
+
109
+ new_full_path = "#{dir.full_path}#{name}"
110
+ self.class.new(new_full_path, dir.box)
111
+ end
112
+
113
+ # Move the entry to another dir. The object will be updated to show its new
114
+ # location.
115
+ def move_to(dir)
116
+ moved = copy_to(dir)
117
+ destroy
118
+ mimic(moved)
119
+ end
120
+
121
+ def mimic(from) # :nodoc:
122
+ @box = from.box
123
+ @path = from.path
124
+ @name = from.name
125
+ end
126
+
127
+ # Unix convention considers entries starting with a . to be hidden.
128
+ def hidden?
129
+ name.slice(0, 1) == '.'
130
+ end
131
+
132
+ # Set the access permissions for the entry.
133
+ #
134
+ # Permissions are set by role and permissions combinations which can be specified individually
135
+ # or grouped together. :user_can => :read, :user_can => :write is the same
136
+ # as :user_can => :read_write.
137
+ #
138
+ # You can also insert 'and' if you find it reads better, like :user_and_group_can => :read_and_write.
139
+ #
140
+ # Any permission excluded is set to deny access. The access call does not set partial
141
+ # permissions which combine with the existing state of the entry, like "chmod o+r" would.
142
+ #
143
+ # Examples:
144
+ #
145
+ # file.access = { :user_can => :read_write, :group_other_can => :read }
146
+ # dir.access = { :user => 'adam', :group => 'users', :read_write_execute => :user_group }
147
+ #
148
+ def access=(options)
149
+ connection.set_access(full_path, Rush::Access.parse(options))
150
+ end
151
+
152
+ # Returns a hash with up to nine values, combining user/group/other with read/write/execute.
153
+ # The key is omitted if the value is false.
154
+ #
155
+ # Examples:
156
+ #
157
+ # entry.access # -> { :user_can_read => true, :user_can_write => true, :group_can_read => true }
158
+ # entry.access[:other_can_read] # -> true or nil
159
+ #
160
+ def access
161
+ Rush::Access.new.from_octal(stat[:mode]).display_hash
162
+ end
163
+
164
+ # Destroy the entry. If it is a dir, everything inside it will also be destroyed.
165
+ def destroy
166
+ connection.destroy(full_path)
167
+ end
168
+
169
+ def ==(other) # :nodoc:
170
+ full_path == other.full_path and box == other.box
171
+ end
172
+
173
+ private
174
+
175
+ def stat
176
+ connection.stat(full_path)
177
+ end
178
+ end
@@ -0,0 +1,31 @@
1
+ module Rush
2
+ # Base class for all rush exceptions.
3
+ class Exception < ::RuntimeError; end
4
+
5
+ # Client was not authorized by remote server; check credentials.
6
+ class NotAuthorized < Exception; end
7
+
8
+ # rushd is not running on the remote box.
9
+ class RushdNotRunning < Exception; end
10
+
11
+ # An unrecognized status code was returned by rushd.
12
+ class FailedTransmit < Exception; end
13
+
14
+ # The entry (file or dir) referenced does not exist. Message is the entry's full path.
15
+ class DoesNotExist < Exception; end
16
+
17
+ # The bash command had a non-zero return value. Message is stderr.
18
+ class BashFailed < Exception; end
19
+
20
+ # There's already an entry by the given name in the given dir.
21
+ class NameAlreadyExists < Exception; end
22
+
23
+ # The name cannot contain a slash; use two operations, rename and then move, instead.
24
+ class NameCannotContainSlash < Exception; end
25
+
26
+ # You cannot move or copy entries to a path that is not a dir (should end with trailing slash).
27
+ class NotADir < Exception; end
28
+
29
+ # A user or permission value specified to set access was not valid.
30
+ class BadAccessSpecifier < Exception; end
31
+ end
data/lib/rush/file.rb ADDED
@@ -0,0 +1,77 @@
1
+ # Files are a subclass of Rush::Entry. Most of the file-specific operations
2
+ # relate to manipulating the file's contents, like search and replace.
3
+ class Rush::File < Rush::Entry
4
+ def dir?
5
+ false
6
+ end
7
+
8
+ # Create a blank file.
9
+ def create
10
+ write('')
11
+ self
12
+ end
13
+
14
+ # Size in bytes on disk.
15
+ def size
16
+ stat[:size]
17
+ end
18
+
19
+ # Raw contents of the file. For non-text files, you probably want to avoid
20
+ # printing this on the screen.
21
+ def contents
22
+ connection.file_contents(full_path)
23
+ end
24
+
25
+ # Write to the file, overwriting whatever was already in it.
26
+ #
27
+ # Example: file.write "hello, world\n"
28
+ def write(new_contents)
29
+ connection.write_file(full_path, new_contents)
30
+ end
31
+
32
+ # Return an array of lines from the file, similar to stdlib's File#readlines.
33
+ def lines
34
+ contents.split("\n")
35
+ end
36
+
37
+ # Search the file's for a regular expression. Returns nil if no match, or
38
+ # each of the matching lines in its entirety.
39
+ #
40
+ # Example: box['/etc/hosts'].search(/localhost/) # -> [ "127.0.0.1 localhost\n", "::1 localhost\n" ]
41
+ def search(pattern)
42
+ matching_lines = lines.select { |line| line.match(pattern) }
43
+ matching_lines.size == 0 ? nil : matching_lines
44
+ end
45
+
46
+ # Search-and-replace file contents.
47
+ #
48
+ # Example: box['/etc/hosts'].replace_contents!(/localhost/, 'local.host')
49
+ def replace_contents!(pattern, replace_with)
50
+ write contents.gsub(pattern, replace_with)
51
+ end
52
+
53
+ # Return the file's contents, or if it doesn't exist, a blank string.
54
+ def contents_or_blank
55
+ contents
56
+ rescue Rush::DoesNotExist
57
+ ""
58
+ end
59
+
60
+ # Count the number of lines in the file.
61
+ def line_count
62
+ lines.size
63
+ end
64
+
65
+ # Return an array of lines, or an empty array if the file does not exist.
66
+ def lines_or_empty
67
+ lines
68
+ rescue Rush::DoesNotExist
69
+ []
70
+ end
71
+
72
+ include Rush::Commands
73
+
74
+ def entries
75
+ [ self ]
76
+ end
77
+ end