childprocess 0.9.0

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 (49) hide show
  1. checksums.yaml +7 -0
  2. data/.document +6 -0
  3. data/.gitignore +28 -0
  4. data/.rspec +1 -0
  5. data/.travis.yml +44 -0
  6. data/CHANGELOG.md +49 -0
  7. data/Gemfile +15 -0
  8. data/LICENSE +20 -0
  9. data/README.md +196 -0
  10. data/Rakefile +61 -0
  11. data/appveyor.yml +60 -0
  12. data/childprocess.gemspec +30 -0
  13. data/lib/childprocess.rb +205 -0
  14. data/lib/childprocess/abstract_io.rb +36 -0
  15. data/lib/childprocess/abstract_process.rb +192 -0
  16. data/lib/childprocess/errors.rb +26 -0
  17. data/lib/childprocess/jruby.rb +56 -0
  18. data/lib/childprocess/jruby/io.rb +16 -0
  19. data/lib/childprocess/jruby/process.rb +159 -0
  20. data/lib/childprocess/jruby/pump.rb +53 -0
  21. data/lib/childprocess/tools/generator.rb +146 -0
  22. data/lib/childprocess/unix.rb +9 -0
  23. data/lib/childprocess/unix/fork_exec_process.rb +70 -0
  24. data/lib/childprocess/unix/io.rb +21 -0
  25. data/lib/childprocess/unix/lib.rb +186 -0
  26. data/lib/childprocess/unix/platform/i386-linux.rb +12 -0
  27. data/lib/childprocess/unix/platform/i386-solaris.rb +11 -0
  28. data/lib/childprocess/unix/platform/x86_64-linux.rb +12 -0
  29. data/lib/childprocess/unix/platform/x86_64-macosx.rb +11 -0
  30. data/lib/childprocess/unix/posix_spawn_process.rb +134 -0
  31. data/lib/childprocess/unix/process.rb +89 -0
  32. data/lib/childprocess/version.rb +3 -0
  33. data/lib/childprocess/windows.rb +33 -0
  34. data/lib/childprocess/windows/handle.rb +91 -0
  35. data/lib/childprocess/windows/io.rb +25 -0
  36. data/lib/childprocess/windows/lib.rb +416 -0
  37. data/lib/childprocess/windows/process.rb +130 -0
  38. data/lib/childprocess/windows/process_builder.rb +175 -0
  39. data/lib/childprocess/windows/structs.rb +149 -0
  40. data/spec/abstract_io_spec.rb +12 -0
  41. data/spec/childprocess_spec.rb +422 -0
  42. data/spec/io_spec.rb +228 -0
  43. data/spec/jruby_spec.rb +24 -0
  44. data/spec/pid_behavior.rb +12 -0
  45. data/spec/platform_detection_spec.rb +86 -0
  46. data/spec/spec_helper.rb +261 -0
  47. data/spec/unix_spec.rb +57 -0
  48. data/spec/windows_spec.rb +23 -0
  49. metadata +179 -0
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 2f7bcd4a74bf8315015b61ff4a5dcf5774fec372
4
+ data.tar.gz: ebb386c0ffd4b05121cbc8c3f18ded1d8721552b
5
+ SHA512:
6
+ metadata.gz: 264821e0ea3147f6d19f63d76f5e290462e0ae8991d6858839b949a470f8ef173b716f5000f24fdfe9b86446eab1da593e7ad81e7cac83880fa7af81e7441563
7
+ data.tar.gz: 26f5534f0cb73ad3482e513706ca7a53d87c7f610efeaf09f64f0c4205b293db565548c097290d27a4b49a2b831f9593edbfc4cf7c0da73903d05c1c3fd98b23
@@ -0,0 +1,6 @@
1
+ README.rdoc
2
+ lib/**/*.rb
3
+ bin/*
4
+ features/**/*.feature
5
+ -
6
+ LICENSE
@@ -0,0 +1,28 @@
1
+ ## MAC OS
2
+ .DS_Store
3
+
4
+ ## TEXTMATE
5
+ *.tmproj
6
+ tmtags
7
+
8
+ ## EMACS
9
+ *~
10
+ \#*
11
+ .\#*
12
+
13
+ ## VIM
14
+ *.swp
15
+
16
+ ## RubyMine
17
+ .idea/*
18
+
19
+ ## PROJECT::GENERAL
20
+ coverage
21
+ rdoc
22
+ pkg
23
+ .rbx
24
+ Gemfile.lock
25
+ .ruby-version
26
+ .bundle
27
+
28
+ ## PROJECT::SPECIFIC
data/.rspec ADDED
@@ -0,0 +1 @@
1
+ --color
@@ -0,0 +1,44 @@
1
+ os:
2
+ - linux
3
+ - osx
4
+
5
+ rvm:
6
+ - 1.9.3
7
+ - jruby-9.1.9.0
8
+ - rbx-3
9
+ - 2.0.0
10
+ - 2.1
11
+ - 2.2
12
+ - 2.3
13
+ - 2.4
14
+ - 2.5
15
+ - ruby-head
16
+
17
+ sudo: true # Necessary to fix JRuby
18
+
19
+ cache: bundler
20
+
21
+ before_install:
22
+ - "echo 'gem: --no-document' > ~/.gemrc"
23
+ - gem update --system
24
+ - gem install bundler # Necessary to fix 1.9.3
25
+
26
+ env:
27
+ global:
28
+ matrix:
29
+ - CHILDPROCESS_POSIX_SPAWN=true CHILDPROCESS_UNSET=should-be-unset
30
+ - CHILDPROCESS_POSIX_SPAWN=false CHILDPROCESS_UNSET=should-be-unset
31
+
32
+ matrix:
33
+ allow_failures:
34
+ - rvm: rbx-3
35
+ - rvm: jruby-9.1.9.0
36
+ - rvm: ruby-head
37
+ - env: "CHILDPROCESS_POSIX_SPAWN=true"
38
+ exclude:
39
+ # Travis does not provide 1.9.3 on OSX
40
+ - rvm: 1.9.3
41
+ os: osx
42
+ # Travis does not provide 2.0.0 on it latest version of OSX
43
+ - rvm: 2.0.0
44
+ os: osx
@@ -0,0 +1,49 @@
1
+ ### Version 0.9.0 / 2018-03-10
2
+
3
+ * Added support for DragonFly BSD.
4
+
5
+
6
+ ### Version 0.8.0 / 2017-09-23
7
+
8
+ * Added a method for determining whether or not a process had been started.
9
+
10
+
11
+ ### Version 0.7.1 / 2017-06-26
12
+
13
+ * Fixed a noisy uninitialized variable warning
14
+
15
+
16
+ ### Version 0.7.0 / 2017-05-07
17
+
18
+ * Debugging information now uses a Logger, which can be configured.
19
+
20
+
21
+ ### Version 0.6.3 / 2017-03-24
22
+
23
+ See beta release notes.
24
+
25
+
26
+ ### Version 0.6.3.beta.1 / 2017-03-10
27
+
28
+ * Bug fix: Fixed child process creation problems on Windows 7 when a child was declared as a leader.
29
+
30
+
31
+ ### Version 0.6.2 / 2017-02-25
32
+
33
+ * Bug fix: Fixed a potentially broken edge case that could occur on older 32-bit OSX systems.
34
+
35
+
36
+ ### Version 0.6.1 / 2017-01-22
37
+
38
+ * Bug fix: Fixed a dependency that was accidentally declared as a runtime
39
+ dependency instead of a development dependency.
40
+
41
+
42
+ ### Version 0.6.0 / 2017-01-22
43
+
44
+ * Support for Ruby 2.4 added
45
+
46
+
47
+ ### Version 0.5.9 / 2016-01-06
48
+
49
+ * The Great Before Times...
data/Gemfile ADDED
@@ -0,0 +1,15 @@
1
+ source "http://rubygems.org"
2
+
3
+ # Specify your gem's dependencies in child_process.gemspec
4
+ gemspec
5
+
6
+
7
+ if RUBY_VERSION =~ /^1\./
8
+ gem 'tins', '< 1.7' # The 'tins' gem requires Ruby 2.x on/after this version
9
+ gem 'json', '< 2.0' # The 'json' gem drops pre-Ruby 2.x support on/after this version
10
+ gem 'term-ansicolor', '< 1.4' # The 'term-ansicolor' gem requires Ruby 2.x on/after this version
11
+
12
+ if RbConfig::CONFIG['host_os'].downcase =~ /mswin|msys|mingw32/
13
+ gem 'ffi', '< 1.9.15' # The 'ffi' gem, for Windows, requires Ruby 2.x on/after this version
14
+ end
15
+ end
data/LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright (c) 2010-2015 Jari Bakken
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,196 @@
1
+ # childprocess
2
+
3
+ This gem aims at being a simple and reliable solution for controlling
4
+ external programs running in the background on any Ruby / OS combination.
5
+
6
+ The code originated in the [selenium-webdriver](https://rubygems.org/gems/selenium-webdriver) gem, but should prove useful as
7
+ a standalone library.
8
+
9
+ [![Build Status](https://secure.travis-ci.org/enkessler/childprocess.svg)](http://travis-ci.org/enkessler/childprocess)
10
+ [![Build status](https://ci.appveyor.com/api/projects/status/fn2snbcd7kku5myk/branch/dev?svg=true)](https://ci.appveyor.com/project/enkessler/childprocess/branch/dev)
11
+ [![Gem Version](https://badge.fury.io/rb/childprocess.svg)](http://badge.fury.io/rb/childprocess)
12
+ [![Code Climate](https://codeclimate.com/github/enkessler/childprocess.svg)](https://codeclimate.com/github/enkessler/childprocess)
13
+ [![Coverage Status](https://coveralls.io/repos/enkessler/childprocess/badge.svg?branch=master)](https://coveralls.io/r/enkessler/childprocess?branch=master)
14
+
15
+ ***
16
+ **This project currently needs a new maintainer. If anyone is interested, please contact me, [enkessler](https://github.com/enkessler).**
17
+ ***
18
+
19
+ # Usage
20
+
21
+ The object returned from `ChildProcess.build` will implement `ChildProcess::AbstractProcess`.
22
+
23
+ ### Basic examples
24
+
25
+ ```ruby
26
+ process = ChildProcess.build("ruby", "-e", "sleep")
27
+
28
+ # inherit stdout/stderr from parent...
29
+ process.io.inherit!
30
+
31
+ # ...or pass an IO
32
+ process.io.stdout = Tempfile.new("child-output")
33
+
34
+ # modify the environment for the child
35
+ process.environment["a"] = "b"
36
+ process.environment["c"] = nil
37
+
38
+ # set the child's working directory
39
+ process.cwd = '/some/path'
40
+
41
+ # start the process
42
+ process.start
43
+
44
+ # check process status
45
+ process.alive? #=> true
46
+ process.exited? #=> false
47
+
48
+ # wait indefinitely for process to exit...
49
+ process.wait
50
+ process.exited? #=> true
51
+
52
+ # get the exit code
53
+ process.exit_code #=> 0
54
+
55
+ # ...or poll for exit + force quit
56
+ begin
57
+ process.poll_for_exit(10)
58
+ rescue ChildProcess::TimeoutError
59
+ process.stop # tries increasingly harsher methods to kill the process.
60
+ end
61
+ ```
62
+
63
+ ### Advanced examples
64
+
65
+ #### Output to pipe
66
+
67
+ ```ruby
68
+ r, w = IO.pipe
69
+
70
+ proc = ChildProcess.build("echo", "foo")
71
+ proc.io.stdout = proc.io.stderr = w
72
+ proc.start
73
+ w.close
74
+
75
+ begin
76
+ loop { print r.readpartial(8192) }
77
+ rescue EOFError
78
+ end
79
+
80
+ proc.wait
81
+ ```
82
+
83
+ Note that if you just want to get the output of a command, the backtick method on Kernel may be a better fit.
84
+
85
+ #### Write to stdin
86
+
87
+ ```ruby
88
+ process = ChildProcess.build("cat")
89
+
90
+ out = Tempfile.new("duplex")
91
+ out.sync = true
92
+
93
+ process.io.stdout = process.io.stderr = out
94
+ process.duplex = true # sets up pipe so process.io.stdin will be available after .start
95
+
96
+ process.start
97
+ process.io.stdin.puts "hello world"
98
+ process.io.stdin.close
99
+
100
+ process.poll_for_exit(exit_timeout_in_seconds)
101
+
102
+ out.rewind
103
+ out.read #=> "hello world\n"
104
+ ```
105
+
106
+ #### Pipe output to another ChildProcess
107
+
108
+ ```ruby
109
+ search = ChildProcess.build("grep", '-E', %w(redis memcached).join('|'))
110
+ search.duplex = true # sets up pipe so search.io.stdin will be available after .start
111
+ search.io.stdout = $stdout
112
+ search.start
113
+
114
+ listing = ChildProcess.build("ps", "aux")
115
+ listing.io.stdout = search.io.stdin
116
+ listing.start
117
+ listing.wait
118
+
119
+ search.io.stdin.close
120
+ search.wait
121
+ ```
122
+
123
+ #### Prefer posix_spawn on *nix
124
+
125
+ If the parent process is using a lot of memory, `fork+exec` can be very expensive. The `posix_spawn()` API removes this overhead.
126
+
127
+ ```ruby
128
+ ChildProcess.posix_spawn = true
129
+ process = ChildProcess.build(*args)
130
+ ```
131
+
132
+ ### Ensure entire process tree dies
133
+
134
+ By default, the child process does not create a new process group. This means there's no guarantee that the entire process tree will die when the child process is killed. To solve this:
135
+
136
+ ```ruby
137
+ process = ChildProcess.build(*args)
138
+ process.leader = true
139
+ process.start
140
+ ```
141
+
142
+ #### Detach from parent
143
+
144
+ ```ruby
145
+ process = ChildProcess.build("sleep", "10")
146
+ process.detach = true
147
+ process.start
148
+ ```
149
+
150
+ #### Invoking a shell
151
+
152
+ As opposed to `Kernel#system`, `Kernel#exec` et al., ChildProcess will not automatically execute your command in a shell (like `/bin/sh` or `cmd.exe`) depending on the arguments.
153
+ This means that if you try to execute e.g. gem executables (like `bundle` or `gem`) or Windows executables (with `.com` or `.bat` extensions) you may see a `ChildProcess::LaunchError`.
154
+ You can work around this by being explicit about what interpreter to invoke:
155
+
156
+ ```ruby
157
+ ChildProcess.build("cmd.exe", "/c", "bundle")
158
+ ChildProcess.build("ruby", "-S", "bundle")
159
+ ```
160
+
161
+ #### Log to file
162
+
163
+ Errors and debugging information are logged to `$stderr` by default but a custom logger can be used instead.
164
+
165
+ ```ruby
166
+ logger = Logger.new('logfile.log')
167
+ logger.level = Logger::DEBUG
168
+ ChildProcess.logger = logger
169
+ ```
170
+
171
+ ## Caveats
172
+
173
+ * With JRuby on Unix, modifying `ENV["PATH"]` before using childprocess could lead to 'Command not found' errors, since JRuby is unable to modify the environment used for PATH searches in `java.lang.ProcessBuilder`. This can be avoided by setting `ChildProcess.posix_spawn = true`.
174
+
175
+ # Implementation
176
+
177
+ How the process is launched and killed depends on the platform:
178
+
179
+ * Unix : `fork + exec` (or `posix_spawn` if enabled)
180
+ * Windows : `CreateProcess()` and friends
181
+ * JRuby : `java.lang.{Process,ProcessBuilder}`
182
+
183
+ # Note on Patches/Pull Requests
184
+
185
+ 1. Fork it
186
+ 2. Create your feature branch (off of the development branch)
187
+ `git checkout -b my-new-feature dev`
188
+ 3. Commit your changes
189
+ `git commit -am 'Add some feature'`
190
+ 4. Push to the branch
191
+ `git push origin my-new-feature`
192
+ 5. Create new Pull Request
193
+
194
+ # Copyright
195
+
196
+ Copyright (c) 2010-2015 Jari Bakken. See LICENSE for details.
@@ -0,0 +1,61 @@
1
+ require 'rubygems'
2
+ require 'rake'
3
+ require 'tmpdir'
4
+
5
+ require 'bundler'
6
+ Bundler::GemHelper.install_tasks
7
+
8
+ include Rake::DSL if defined?(::Rake::DSL)
9
+
10
+ require 'rspec/core/rake_task'
11
+ RSpec::Core::RakeTask.new(:spec) do |spec|
12
+ spec.ruby_opts = "-I lib:spec -w"
13
+ spec.pattern = 'spec/**/*_spec.rb'
14
+ end
15
+
16
+ desc 'Run specs for rcov'
17
+ RSpec::Core::RakeTask.new(:rcov) do |spec|
18
+ spec.ruby_opts = "-I lib:spec"
19
+ spec.pattern = 'spec/**/*_spec.rb'
20
+ spec.rcov = true
21
+ spec.rcov_opts = %w[--exclude spec,ruby-debug,/Library/Ruby,.gem --include lib/childprocess]
22
+ end
23
+
24
+ task :default => :spec
25
+
26
+ begin
27
+ require 'yard'
28
+ YARD::Rake::YardocTask.new
29
+ rescue LoadError
30
+ task :yardoc do
31
+ abort "YARD is not available. In order to run yardoc, you must: sudo gem install yard"
32
+ end
33
+ end
34
+
35
+ task :clean do
36
+ rm_rf "pkg"
37
+ rm_rf "childprocess.jar"
38
+ end
39
+
40
+ desc 'Create jar to bundle in selenium-webdriver'
41
+ task :jar => [:clean, :build] do
42
+ tmpdir = Dir.mktmpdir("childprocess-jar")
43
+ gem_to_package = Dir['pkg/*.gem'].first
44
+ gem_name = File.basename(gem_to_package, ".gem")
45
+ p :gem_to_package => gem_to_package, :gem_name => gem_name
46
+
47
+ sh "gem install -i #{tmpdir} #{gem_to_package} --ignore-dependencies --no-rdoc --no-ri"
48
+ sh "jar cf childprocess.jar -C #{tmpdir}/gems/#{gem_name}/lib ."
49
+ sh "jar tf childprocess.jar"
50
+ end
51
+
52
+ task :env do
53
+ $:.unshift File.expand_path("../lib", __FILE__)
54
+ require 'childprocess'
55
+ end
56
+
57
+ desc 'Calculate size of posix_spawn structs for the current platform'
58
+ task :generate => :env do
59
+ require 'childprocess/tools/generator'
60
+ ChildProcess::Tools::Generator.generate
61
+ end
@@ -0,0 +1,60 @@
1
+ version: '1.0.{build}'
2
+
3
+ environment:
4
+ matrix:
5
+ - CHILDPROCESS_POSIX_SPAWN: true
6
+ CHILDPROCESS_UNSET: should-be-unset
7
+ RUBY_VERSION: 193-x64
8
+ - CHILDPROCESS_POSIX_SPAWN: false
9
+ CHILDPROCESS_UNSET: should-be-unset
10
+ RUBY_VERSION: 193-x64
11
+ - CHILDPROCESS_POSIX_SPAWN: true
12
+ CHILDPROCESS_UNSET: should-be-unset
13
+ RUBY_VERSION: 200-x64
14
+ - CHILDPROCESS_POSIX_SPAWN: false
15
+ CHILDPROCESS_UNSET: should-be-unset
16
+ RUBY_VERSION: 200-x64
17
+ - CHILDPROCESS_POSIX_SPAWN: true
18
+ CHILDPROCESS_UNSET: should-be-unset
19
+ RUBY_VERSION: 21-x64
20
+ - CHILDPROCESS_POSIX_SPAWN: false
21
+ CHILDPROCESS_UNSET: should-be-unset
22
+ RUBY_VERSION: 21-x64
23
+ - CHILDPROCESS_POSIX_SPAWN: true
24
+ CHILDPROCESS_UNSET: should-be-unset
25
+ RUBY_VERSION: 22-x64
26
+ - CHILDPROCESS_POSIX_SPAWN: false
27
+ CHILDPROCESS_UNSET: should-be-unset
28
+ RUBY_VERSION: 22-x64
29
+ - CHILDPROCESS_POSIX_SPAWN: true
30
+ CHILDPROCESS_UNSET: should-be-unset
31
+ RUBY_VERSION: 23-x64
32
+ - CHILDPROCESS_POSIX_SPAWN: false
33
+ CHILDPROCESS_UNSET: should-be-unset
34
+ RUBY_VERSION: 23-x64
35
+ - CHILDPROCESS_POSIX_SPAWN: true
36
+ CHILDPROCESS_UNSET: should-be-unset
37
+ RUBY_VERSION: 24-x64
38
+ - CHILDPROCESS_POSIX_SPAWN: false
39
+ CHILDPROCESS_UNSET: should-be-unset
40
+ RUBY_VERSION: 24-x64
41
+ - CHILDPROCESS_POSIX_SPAWN: true
42
+ CHILDPROCESS_UNSET: should-be-unset
43
+ RUBY_VERSION: 25-x64
44
+ - CHILDPROCESS_POSIX_SPAWN: false
45
+ CHILDPROCESS_UNSET: should-be-unset
46
+ RUBY_VERSION: 25-x64
47
+
48
+ install:
49
+ - set PATH=C:\Ruby%RUBY_VERSION%\bin;%PATH%
50
+ - bundle install
51
+
52
+ build: off
53
+
54
+ before_test:
55
+ - ruby -v
56
+ - gem -v
57
+ - bundle -v
58
+
59
+ test_script:
60
+ - bundle exec rake