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/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
|