ng 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,19 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
18
+ log
19
+ test.rb
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in ng.gemspec
4
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,23 @@
1
+ Copyright (c) 2012 Ryan Taylor Long
2
+
3
+
4
+ MIT License
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining
7
+ a copy of this software and associated documentation files (the
8
+ "Software"), to deal in the Software without restriction, including
9
+ without limitation the rights to use, copy, modify, merge, publish,
10
+ distribute, sublicense, and/or sell copies of the Software, and to
11
+ permit persons to whom the Software is furnished to do so, subject to
12
+ the following conditions:
13
+
14
+ The above copyright notice and this permission notice shall be
15
+ included in all copies or substantial portions of the Software.
16
+
17
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,29 @@
1
+ # ng - Pure-Ruby Nailgun client port
2
+
3
+ Eliminates the need to shell-out when using Nailgun
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'ng'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install ng
18
+
19
+ ## Usage
20
+
21
+ TODO: Write usage instructions here
22
+
23
+ ## Contributing
24
+
25
+ 1. Fork it
26
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
27
+ 3. Commit your changes (`git commit -am 'Added some feature'`)
28
+ 4. Push to the branch (`git push origin my-new-feature`)
29
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env rake
2
+ require "bundler/gem_tasks"
@@ -0,0 +1,25 @@
1
+ module Nailgun
2
+ class Client
3
+ class Chunk
4
+ attr_reader :content, :type
5
+
6
+ def initialize(type, content)
7
+ @content = content.to_s
8
+ @type = type
9
+ end
10
+
11
+ def header
12
+ ChunkHeader.new(type, length)
13
+ end
14
+
15
+ def length
16
+ content.length
17
+ end
18
+
19
+ def to_s
20
+ header.to_s + content
21
+ end
22
+
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,18 @@
1
+ module Nailgun
2
+ class Client
3
+ class ChunkHeader
4
+ def initialize(type, content_length)
5
+ @type = Nailgun::CHUNK_TYPES[type]
6
+ @content_length = content_length
7
+ end
8
+
9
+ def to_a
10
+ [@content_length, @type]
11
+ end
12
+
13
+ def to_s
14
+ to_a.pack('NA')
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,194 @@
1
+ require_relative 'client/chunk'
2
+ require_relative 'client/chunk_header'
3
+ require 'socket'
4
+ require 'io/wait'
5
+ require 'logger'
6
+ require 'timeout'
7
+
8
+ module Nailgun
9
+ class Client
10
+ LOGGER = Logger.new('log')
11
+
12
+ attr_reader :opts, :socket
13
+
14
+ # Public: Convinience method to instantiate and run the command
15
+ #
16
+ # command - see #run
17
+ # args - see #run
18
+ # opts = {} - see #initialize
19
+ #
20
+ # Returns the duplicated String.
21
+ def self.run(command, args, opts = {})
22
+ self.new(opts).run(command, args)
23
+ end
24
+
25
+ # Public: Initialize a Client.
26
+ #
27
+ # opts = {} - a Hash of options to override the defaults in Nailgun::DEFAULTS
28
+ def initialize(opts = {})
29
+ @opts = Nailgun::DEFAULTS.merge(opts)
30
+ @socket = TCPSocket.new(*@opts.values_at(:hostname, :port))
31
+ debug "Opened new #{@socket.inspect}"
32
+ end
33
+
34
+ # Public: Run a command on the Client instance
35
+ #
36
+ # command - the command string
37
+ # *args - any arguments to send
38
+ def run(command, *args)
39
+ receive_loop # start the loop
40
+
41
+ send_args args.flatten
42
+ send_env opts[:env]
43
+ send_dir opts[:dir]
44
+ send_command command
45
+ send_stdin opts[:stdin]
46
+
47
+ receive_loop.join
48
+ return nil
49
+ end
50
+
51
+ # Public: Start the receiver loop Thread, memoize it, and return the Thread
52
+ #
53
+ # Returns the Thread object, whose value will eventually be the exit status from the Nailgun
54
+ # server
55
+ def receive_loop
56
+ @loop ||= Thread.new {
57
+ catch(:exit) do
58
+ loop { receive_chunk }
59
+ end
60
+ }
61
+ end
62
+
63
+ private
64
+
65
+ # Private: Send the argument chunks
66
+ #
67
+ # *args - an Array of the arguments to send
68
+ def send_args(*args)
69
+ args.flatten.each do |arg|
70
+ send_chunk :arg, arg
71
+ end
72
+ end
73
+
74
+ # Private: Send the environment vars.
75
+ #
76
+ # env - a Hash in the format of Ruby's ENV constant
77
+ def send_env(env)
78
+ env.each do |var|
79
+ send_chunk :env, var.join(?=)
80
+ end
81
+ end
82
+
83
+ # Private: Send the working directory
84
+ #
85
+ # dir - the working directory
86
+ def send_dir(dir)
87
+ send_chunk :dir, dir.to_s
88
+ end
89
+
90
+ # Private: Send the command to be run
91
+ #
92
+ # command - the Nail command (usually a Java class name)
93
+ def send_command(command)
94
+ send_chunk :cmd, command
95
+ end
96
+
97
+ # Private: Send the STDIN stream for the Nail to read from.
98
+ #
99
+ # io = nil - an IO of the stdin stream to send.
100
+ def send_stdin(io = nil)
101
+ unless io.nil?
102
+ begin
103
+ send_chunk :stdin, io.read(2048)
104
+ end until io.eof?
105
+ io.close
106
+ end
107
+ send_chunk :stdin_eof
108
+ end
109
+
110
+ # Private: Send a chunk. Used by the higher-level methods
111
+ #
112
+ # type - the chunk type
113
+ # content = nil - the actual content
114
+ def send_chunk(type, content = nil)
115
+ chunk = Chunk.new(type, content).to_s
116
+ debug "Sending #{type} chunk: #{content.inspect}"
117
+ socket.write chunk
118
+ end
119
+
120
+ # Private: get the next chunk from the socket, and then determine what to do with it.
121
+ def receive_chunk
122
+ Timeout.timeout(Nailgun::TIMEOUT, Nailgun::TimeoutError) do
123
+ length, type = receive_header
124
+ if length == 0
125
+ debug "Received #{type} chunk with no content"
126
+ else
127
+ debug "About to read #{type} chunk (#{length} B). Content follows (until <<<< END CHUNK >>>>) "
128
+
129
+ content = socket.read(length)
130
+
131
+ LOGGER << (content + "<<<< END CHUNK >>>>\n")
132
+ end
133
+
134
+ handle_chunk(type, content)
135
+ end
136
+ end
137
+
138
+ # Private: Block while waiting for a header for the next chunk
139
+ #
140
+ # Returns [length, type]
141
+ def receive_header
142
+ socket.read(Nailgun::CHUNK_HEADER_LEN).unpack('NA')
143
+ end
144
+
145
+ # Private: Determine what to do with the received chunk
146
+ #
147
+ # type - chunk type
148
+ # content - chunk content
149
+ def handle_chunk(type, content)
150
+ case t = Nailgun::CHUNK_TYPES.key(type)
151
+ when :stdout, :stderr
152
+ opts[t].write content
153
+ when :exit
154
+ socket.close
155
+ handle_exit(content.to_i)
156
+ else
157
+ raise Nailgun::UnexpectedChunktypeError.new([type, content].join(?;))
158
+ end
159
+ end
160
+
161
+ def handle_exit(code)
162
+ if code == 0
163
+ throw :exit
164
+ elsif ex = Nailgun::EXIT_CODE_EXCEPTIONS[code]
165
+ raise ex.new
166
+ else
167
+ raise Nailgun::OtherError.new(code.to_s)
168
+ end
169
+ end
170
+
171
+ # Private: Debug log message
172
+ def debug(message)
173
+ LOGGER.debug(message)
174
+ end
175
+
176
+ # Extend the return value with a success? method akin to Process::Status, which I can't figure
177
+ # out how to instantiate manually
178
+ class ExitStatus
179
+ def initialize(value)
180
+ @val = value
181
+ end
182
+ def success?
183
+ @val == 0
184
+ end
185
+ def method_missing(*args)
186
+ @val.send(*args)
187
+ end
188
+ def inspect
189
+ @val.inspect
190
+ end
191
+ end
192
+
193
+ end
194
+ end
@@ -0,0 +1,3 @@
1
+ module Nailgun
2
+ VERSION = '0.0.1'
3
+ end
data/lib/nailgun.rb ADDED
@@ -0,0 +1,49 @@
1
+ require_relative 'nailgun/version'
2
+ require_relative 'nailgun/client'
3
+
4
+ module Nailgun
5
+
6
+ DEFAULTS = {
7
+ hostname: 'localhost',
8
+ port: 2113,
9
+ stdin: nil,
10
+ stdout: STDOUT,
11
+ stderr: STDERR,
12
+ env: ENV,
13
+ dir: Dir.pwd
14
+ }.freeze
15
+
16
+ CHUNK_HEADER_LEN = 5
17
+
18
+ TIMEOUT = 5
19
+
20
+ TimeoutError = Class.new(StandardError)
21
+ SocketFailedError = Class.new(StandardError)
22
+ ConnectFailedError = Class.new(StandardError)
23
+ UnexpectedChunktypeError = Class.new(StandardError)
24
+ ServerExceptionError = Class.new(StandardError)
25
+ ConnectionBrokenError = Class.new(StandardError)
26
+ BadArgumentsError = Class.new(StandardError)
27
+ OtherError = Class.new(StandardError)
28
+
29
+ EXIT_CODE_EXCEPTIONS = {
30
+ 999 => SocketFailedError,
31
+ 998 => ConnectFailedError,
32
+ 997 => UnexpectedChunktypeError,
33
+ 996 => ServerExceptionError,
34
+ 995 => ConnectionBrokenError,
35
+ 994 => BadArgumentsError
36
+ }.freeze
37
+
38
+ CHUNK_TYPES = {
39
+ stdin: '0',
40
+ stdout: '1',
41
+ stderr: '2',
42
+ stdin_eof: '.',
43
+ arg: 'A',
44
+ env: 'E',
45
+ dir: 'D',
46
+ cmd: 'C',
47
+ exit: 'X'
48
+ }.freeze
49
+ end
data/ng.gemspec ADDED
@@ -0,0 +1,17 @@
1
+ # -*- encoding: utf-8 -*-
2
+ require File.expand_path('../lib/nailgun/version', __FILE__)
3
+
4
+ Gem::Specification.new do |gem|
5
+ gem.authors = ['Ryan Taylor Long']
6
+ gem.email = ['ryan@rtlong.com']
7
+ gem.description = %q{Pure-Ruby Nailgun client port}
8
+ gem.summary = %q{Eliminates the need to shell-out when using Nailgun}
9
+ gem.homepage = ''
10
+
11
+ gem.files = `git ls-files`.split($\)
12
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
13
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
14
+ gem.name = 'ng'
15
+ gem.require_paths = ['lib']
16
+ gem.version = Nailgun::VERSION
17
+ end
metadata ADDED
@@ -0,0 +1,57 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: ng
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Ryan Taylor Long
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2012-10-18 00:00:00.000000000 Z
13
+ dependencies: []
14
+ description: Pure-Ruby Nailgun client port
15
+ email:
16
+ - ryan@rtlong.com
17
+ executables: []
18
+ extensions: []
19
+ extra_rdoc_files: []
20
+ files:
21
+ - .gitignore
22
+ - Gemfile
23
+ - LICENSE
24
+ - README.md
25
+ - Rakefile
26
+ - lib/nailgun.rb
27
+ - lib/nailgun/client.rb
28
+ - lib/nailgun/client/chunk.rb
29
+ - lib/nailgun/client/chunk_header.rb
30
+ - lib/nailgun/version.rb
31
+ - ng.gemspec
32
+ homepage: ''
33
+ licenses: []
34
+ post_install_message:
35
+ rdoc_options: []
36
+ require_paths:
37
+ - lib
38
+ required_ruby_version: !ruby/object:Gem::Requirement
39
+ none: false
40
+ requirements:
41
+ - - ! '>='
42
+ - !ruby/object:Gem::Version
43
+ version: '0'
44
+ required_rubygems_version: !ruby/object:Gem::Requirement
45
+ none: false
46
+ requirements:
47
+ - - ! '>='
48
+ - !ruby/object:Gem::Version
49
+ version: '0'
50
+ requirements: []
51
+ rubyforge_project:
52
+ rubygems_version: 1.8.23
53
+ signing_key:
54
+ specification_version: 3
55
+ summary: Eliminates the need to shell-out when using Nailgun
56
+ test_files: []
57
+ has_rdoc: