right_popen 1.0.20-x86-mswin32-60 → 1.0.21-x86-mswin32-60
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/lib/right_popen/version.rb +1 -1
- data/lib/win32/right_popen.so +0 -0
- data/spec/produce_mixed_output.rb +2 -1
- data/spec/right_popen_spec.rb +64 -23
- data/spec/runner.rb +36 -23
- metadata +4 -4
    
        data/lib/right_popen/version.rb
    CHANGED
    
    
    
        data/lib/win32/right_popen.so
    CHANGED
    
    | Binary file | 
    
        data/spec/right_popen_spec.rb
    CHANGED
    
    | @@ -1,5 +1,6 @@ | |
| 1 1 | 
             
            require File.expand_path(File.join(File.dirname(__FILE__), 'spec_helper'))
         | 
| 2 2 | 
             
            require File.expand_path(File.join(File.dirname(__FILE__), 'runner'))
         | 
| 3 | 
            +
            require 'stringio'
         | 
| 3 4 |  | 
| 4 5 | 
             
            module RightScale
         | 
| 5 6 | 
             
              describe 'popen3' do
         | 
| @@ -105,24 +106,62 @@ module RightScale | |
| 105 106 | 
             
                  status.pid.should > 0
         | 
| 106 107 | 
             
                end
         | 
| 107 108 |  | 
| 108 | 
            -
                it 'should preserve  | 
| 109 | 
            -
                   | 
| 110 | 
            -
                   | 
| 109 | 
            +
                it 'should preserve interleaved output when yielding CPU on consumer thread' do
         | 
| 110 | 
            +
                  lines = 11
         | 
| 111 | 
            +
                  exit_code = 42
         | 
| 112 | 
            +
                  repeats = 5
         | 
| 113 | 
            +
                  force_yield = 0.1
         | 
| 114 | 
            +
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'produce_mixed_output.rb'))}\" #{lines} #{exit_code}"
         | 
| 115 | 
            +
                  runner = Runner.new
         | 
| 116 | 
            +
                  actual_output = StringIO.new
         | 
| 117 | 
            +
                  actual_error = StringIO.new
         | 
| 118 | 
            +
                  puts
         | 
| 119 | 
            +
                  stats = runner.run_right_popen(command, :repeats=>repeats, :force_yield=>force_yield) do |status|
         | 
| 120 | 
            +
                    status.status.exitstatus.should == exit_code
         | 
| 121 | 
            +
                    status.pid.should > 0
         | 
| 122 | 
            +
                    actual_output << status.output_text
         | 
| 123 | 
            +
                    actual_error << status.error_text
         | 
| 124 | 
            +
                    print '+'
         | 
| 125 | 
            +
                  end
         | 
| 126 | 
            +
                  puts
         | 
| 127 | 
            +
                  stats.size.should == repeats
         | 
| 128 | 
            +
             | 
| 129 | 
            +
                  expected_output = StringIO.new
         | 
| 130 | 
            +
                  repeats.times do
         | 
| 131 | 
            +
                    lines.times do |i|
         | 
| 132 | 
            +
                      expected_output << "stdout #{i}\n"
         | 
| 133 | 
            +
                    end
         | 
| 134 | 
            +
                  end
         | 
| 135 | 
            +
                  actual_output.string.should == expected_output.string
         | 
| 136 | 
            +
             | 
| 137 | 
            +
                  expected_error = StringIO.new
         | 
| 138 | 
            +
                  repeats.times do
         | 
| 139 | 
            +
                    lines.times do |i|
         | 
| 140 | 
            +
                      (expected_error << "stderr #{i}\n") if 0 == i % 10
         | 
| 141 | 
            +
                    end
         | 
| 142 | 
            +
                  end
         | 
| 143 | 
            +
                  actual_error.string.should == expected_error.string
         | 
| 144 | 
            +
                end
         | 
| 145 | 
            +
             | 
| 146 | 
            +
                it 'should preserve interleaved output when process is spewing rapidly' do
         | 
| 147 | 
            +
                  lines = LARGE_OUTPUT_COUNTER
         | 
| 148 | 
            +
                  exit_code = 99
         | 
| 149 | 
            +
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'produce_mixed_output.rb'))}\" #{lines} #{exit_code}"
         | 
| 111 150 | 
             
                  runner = Runner.new
         | 
| 112 151 | 
             
                  status = runner.run_right_popen(command)
         | 
| 113 | 
            -
                  status.status.exitstatus.should ==  | 
| 152 | 
            +
                  status.status.exitstatus.should == exit_code
         | 
| 114 153 |  | 
| 115 | 
            -
                   | 
| 116 | 
            -
                   | 
| 117 | 
            -
                     | 
| 154 | 
            +
                  expected_output = StringIO.new
         | 
| 155 | 
            +
                  lines.times do |i|
         | 
| 156 | 
            +
                    expected_output << "stdout #{i}\n"
         | 
| 118 157 | 
             
                  end
         | 
| 119 | 
            -
                  status.output_text.should ==  | 
| 158 | 
            +
                  status.output_text.should == expected_output.string
         | 
| 120 159 |  | 
| 121 | 
            -
                   | 
| 122 | 
            -
                   | 
| 123 | 
            -
                    ( | 
| 160 | 
            +
                  expected_error = StringIO.new
         | 
| 161 | 
            +
                  lines.times do |i|
         | 
| 162 | 
            +
                    (expected_error << "stderr #{i}\n") if 0 == i % 10
         | 
| 124 163 | 
             
                  end
         | 
| 125 | 
            -
                  status.error_text.should ==  | 
| 164 | 
            +
                  status.error_text.should == expected_error.string
         | 
| 126 165 | 
             
                  status.pid.should > 0
         | 
| 127 166 | 
             
                end
         | 
| 128 167 |  | 
| @@ -132,7 +171,7 @@ module RightScale | |
| 132 171 | 
             
                  status = runner.run_right_popen(command)
         | 
| 133 172 | 
             
                  status.status.exitstatus.should == 0
         | 
| 134 173 | 
             
                  status.output_text.should_not include('_test_')
         | 
| 135 | 
            -
                  status = runner.run_right_popen(command, :__test__ => '42')
         | 
| 174 | 
            +
                  status = runner.run_right_popen(command, :env=>{ :__test__ => '42' })
         | 
| 136 175 | 
             
                  status.status.exitstatus.should == 0
         | 
| 137 176 | 
             
                  status.output_text.should match(/^__test__=42$/)
         | 
| 138 177 | 
             
                  status.pid.should > 0
         | 
| @@ -145,7 +184,7 @@ module RightScale | |
| 145 184 | 
             
                    ENV.each { |k, v| old_envs[k] = v }
         | 
| 146 185 | 
             
                    command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'print_env.rb'))}\""
         | 
| 147 186 | 
             
                    runner = Runner.new
         | 
| 148 | 
            -
                    status = runner.run_right_popen(command, :__test__ => '42')
         | 
| 187 | 
            +
                    status = runner.run_right_popen(command, :env=>{ :__test__ => '42' })
         | 
| 149 188 | 
             
                    status.status.exitstatus.should == 0
         | 
| 150 189 | 
             
                    status.output_text.should match(/^__test__=42$/)
         | 
| 151 190 | 
             
                    ENV.each { |k, v| old_envs[k].should == v }
         | 
| @@ -162,7 +201,7 @@ module RightScale | |
| 162 201 | 
             
                  it 'should merge the PATH variable instead of overriding it' do
         | 
| 163 202 | 
             
                    command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'print_env.rb'))}\""
         | 
| 164 203 | 
             
                    runner = Runner.new
         | 
| 165 | 
            -
                    status = runner.run_right_popen(command, 'PATH' => "c:/bogus\\bin")
         | 
| 204 | 
            +
                    status = runner.run_right_popen(command, :env=>{ 'PATH' => "c:/bogus\\bin" })
         | 
| 166 205 | 
             
                    status.status.exitstatus.should == 0
         | 
| 167 206 | 
             
                    status.output_text.should include('c:\\bogus\\bin;')
         | 
| 168 207 | 
             
                    status.pid.should > 0
         | 
| @@ -204,17 +243,19 @@ module RightScale | |
| 204 243 | 
             
                  pending 'Set environment variable TEST_LEAK to enable' unless ENV['TEST_LEAK']
         | 
| 205 244 | 
             
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'produce_output.rb'))}\" \"#{STANDARD_MESSAGE}\" \"#{ERROR_MESSAGE}\""
         | 
| 206 245 | 
             
                  runner = Runner.new
         | 
| 207 | 
            -
                   | 
| 208 | 
            -
                   | 
| 209 | 
            -
             | 
| 210 | 
            -
             | 
| 211 | 
            -
             | 
| 246 | 
            +
                  stats = runner.run_right_popen(command, :repeats=>REPEAT_TEST_COUNTER)
         | 
| 247 | 
            +
                  stats.each do |status|
         | 
| 248 | 
            +
                    status.status.exitstatus.should == 0
         | 
| 249 | 
            +
                    status.output_text.should == STANDARD_MESSAGE + "\n"
         | 
| 250 | 
            +
                    status.error_text.should == ERROR_MESSAGE + "\n"
         | 
| 251 | 
            +
                    status.pid.should > 0
         | 
| 252 | 
            +
                  end
         | 
| 212 253 | 
             
                end
         | 
| 213 254 |  | 
| 214 255 | 
             
                it 'should pass input to child process' do
         | 
| 215 256 | 
             
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'increment.rb'))}\""
         | 
| 216 257 | 
             
                  runner = Runner.new
         | 
| 217 | 
            -
                  status = runner.run_right_popen(command,  | 
| 258 | 
            +
                  status = runner.run_right_popen(command, :input=>"42\n")
         | 
| 218 259 | 
             
                  status.status.exitstatus.should == 0
         | 
| 219 260 | 
             
                  status.output_text.should == "43\n"
         | 
| 220 261 | 
             
                  status.error_text.should be_empty
         | 
| @@ -225,7 +266,7 @@ module RightScale | |
| 225 266 | 
             
            	  pending 'not implemented for windows' if is_windows?
         | 
| 226 267 | 
             
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'stdout.rb'))}\""
         | 
| 227 268 | 
             
                  runner = Runner.new
         | 
| 228 | 
            -
                  status = runner.run_right_popen(command,  | 
| 269 | 
            +
                  status = runner.run_right_popen(command, :expect_timeout=>true, :timeout=>2)
         | 
| 229 270 | 
             
                  status.did_timeout.should be_true
         | 
| 230 271 | 
             
                  status.output_text.should be_empty
         | 
| 231 272 | 
             
                  status.error_text.should == "Closing stdout\n"
         | 
| @@ -235,7 +276,7 @@ module RightScale | |
| 235 276 | 
             
            	  pending 'not implemented for windows' if is_windows?
         | 
| 236 277 | 
             
                  command = "\"#{RUBY_CMD}\" \"#{File.expand_path(File.join(File.dirname(__FILE__), 'background.rb'))}\""
         | 
| 237 278 | 
             
                  runner = Runner.new
         | 
| 238 | 
            -
                  status = runner.run_right_popen(command | 
| 279 | 
            +
                  status = runner.run_right_popen(command)
         | 
| 239 280 | 
             
                  status.status.exitstatus.should == 0
         | 
| 240 281 | 
             
                  status.did_timeout.should be_false
         | 
| 241 282 | 
             
                  status.output_text.should be_empty
         | 
    
        data/spec/runner.rb
    CHANGED
    
    | @@ -24,18 +24,20 @@ | |
| 24 24 | 
             
            module RightScale
         | 
| 25 25 | 
             
              class Runner
         | 
| 26 26 | 
             
                class RunnerStatus
         | 
| 27 | 
            -
                  def initialize(command,  | 
| 27 | 
            +
                  def initialize(command, options={}, &callback)
         | 
| 28 | 
            +
                    options = {:force_yield=>nil, :timeout=>nil, :expect_timeout=>false}.merge(options)
         | 
| 28 29 | 
             
                    @output_text = ""
         | 
| 29 30 | 
             
                    @error_text  = ""
         | 
| 30 31 | 
             
                    @status      = nil
         | 
| 31 32 | 
             
                    @did_timeout = false
         | 
| 32 | 
            -
                    @callback    =  | 
| 33 | 
            +
                    @callback    = callback
         | 
| 33 34 | 
             
                    @pid         = nil
         | 
| 35 | 
            +
                    @force_yield = options[:force_yield]
         | 
| 34 36 | 
             
                    EM.next_tick do
         | 
| 35 | 
            -
                      @timeout = EM::Timer.new(2) do
         | 
| 36 | 
            -
                        puts "\n** Failed to run #{command.inspect}: Timeout"
         | 
| 37 | 
            +
                      @timeout = EM::Timer.new(options[:timeout] || 2) do
         | 
| 38 | 
            +
                        puts "\n** Failed to run #{command.inspect}: Timeout" unless options[:expect_timeout]
         | 
| 37 39 | 
             
                        @did_timeout = true
         | 
| 38 | 
            -
                        @callback.call(self)
         | 
| 40 | 
            +
                        @callback.call(self) if options[:expect_timeout]
         | 
| 39 41 | 
             
                      end
         | 
| 40 42 | 
             
                    end
         | 
| 41 43 | 
             
                  end
         | 
| @@ -43,10 +45,12 @@ module RightScale | |
| 43 45 | 
             
                  attr_accessor :output_text, :error_text, :status, :did_timeout, :pid
         | 
| 44 46 |  | 
| 45 47 | 
             
                  def on_read_stdout(data)
         | 
| 48 | 
            +
                    sleep @force_yield if @force_yield
         | 
| 46 49 | 
             
                    @output_text << data
         | 
| 47 50 | 
             
                  end
         | 
| 48 51 |  | 
| 49 52 | 
             
                  def on_read_stderr(data)
         | 
| 53 | 
            +
                    sleep @force_yield if @force_yield
         | 
| 50 54 | 
             
                    @error_text << data
         | 
| 51 55 | 
             
                  end
         | 
| 52 56 |  | 
| @@ -69,42 +73,51 @@ module RightScale | |
| 69 73 | 
             
                  @last_iteration = 0
         | 
| 70 74 | 
             
                end
         | 
| 71 75 |  | 
| 72 | 
            -
                def do_right_popen(command,  | 
| 73 | 
            -
                   | 
| 76 | 
            +
                def do_right_popen(command, options={}, &callback)
         | 
| 77 | 
            +
                  options = {:env=>nil, :input=>nil, :timeout=>nil, :force_yield=>nil, :expect_timeout=>false}.merge(options)
         | 
| 78 | 
            +
                  status = RunnerStatus.new(command, options, &callback)
         | 
| 74 79 | 
             
                  RightScale.popen3(:command        => command,
         | 
| 75 | 
            -
             | 
| 76 | 
            -
             | 
| 77 | 
            -
             | 
| 78 | 
            -
             | 
| 79 | 
            -
             | 
| 80 | 
            -
             | 
| 81 | 
            -
             | 
| 80 | 
            +
                                    :input          => options[:input],
         | 
| 81 | 
            +
                                    :target         => status,
         | 
| 82 | 
            +
                                    :environment    => options[:env],
         | 
| 83 | 
            +
                                    :stdout_handler => :on_read_stdout,
         | 
| 84 | 
            +
                                    :stderr_handler => :on_read_stderr,
         | 
| 85 | 
            +
                                    :pid_handler    => :on_pid,
         | 
| 86 | 
            +
                                    :exit_handler   => :on_exit)
         | 
| 82 87 | 
             
                  status
         | 
| 83 88 | 
             
                end
         | 
| 84 89 |  | 
| 85 | 
            -
                def run_right_popen(command,  | 
| 90 | 
            +
                def run_right_popen(command, options={}, &callback)
         | 
| 91 | 
            +
                  options = {:repeats=>1, :env=>nil, :input=>nil, :timeout=>nil, :force_yield=>nil, :expect_timeout=>false}.merge(options)
         | 
| 86 92 | 
             
                  begin
         | 
| 87 93 | 
             
                    @iterations = 0
         | 
| 94 | 
            +
                    @repeats = options[:repeats]
         | 
| 95 | 
            +
                    @stats = []
         | 
| 88 96 | 
             
                    EM.run do
         | 
| 89 | 
            -
                      EM. | 
| 90 | 
            -
                        do_right_popen(command,  | 
| 91 | 
            -
                          maybe_continue(status)
         | 
| 97 | 
            +
                      EM.defer do
         | 
| 98 | 
            +
                        do_right_popen(command, options) do |status|
         | 
| 99 | 
            +
                          maybe_continue(status, command, options, &callback)
         | 
| 92 100 | 
             
                        end
         | 
| 93 101 | 
             
                      end
         | 
| 94 102 | 
             
                    end
         | 
| 95 | 
            -
                    @ | 
| 103 | 
            +
                    @stats.size < 2 ? @stats.first : @stats
         | 
| 96 104 | 
             
                  rescue Exception => e
         | 
| 97 105 | 
             
                    puts "\n** Failed: #{e.message} FROM\n#{e.backtrace.join("\n")}"
         | 
| 98 106 | 
             
                    raise e
         | 
| 99 107 | 
             
                  end
         | 
| 100 108 | 
             
                end
         | 
| 101 109 |  | 
| 102 | 
            -
                def maybe_continue(status)
         | 
| 110 | 
            +
                def maybe_continue(status, command, options, &callback)
         | 
| 103 111 | 
             
                  @iterations += 1
         | 
| 104 | 
            -
                   | 
| 105 | 
            -
             | 
| 112 | 
            +
                  @stats << status
         | 
| 113 | 
            +
                  callback.call(status) if callback
         | 
| 114 | 
            +
                  if @iterations < @repeats
         | 
| 115 | 
            +
                    EM.defer do
         | 
| 116 | 
            +
                      do_right_popen(command, options) do |status|
         | 
| 117 | 
            +
                        maybe_continue(status, command, options, &callback)
         | 
| 118 | 
            +
                      end
         | 
| 119 | 
            +
                    end
         | 
| 106 120 | 
             
                  else
         | 
| 107 | 
            -
                    @status = status
         | 
| 108 121 | 
             
                    EM.stop
         | 
| 109 122 | 
             
                  end
         | 
| 110 123 | 
             
                end
         | 
    
        metadata
    CHANGED
    
    | @@ -1,13 +1,13 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification 
         | 
| 2 2 | 
             
            name: right_popen
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version 
         | 
| 4 | 
            -
              hash:  | 
| 4 | 
            +
              hash: 61
         | 
| 5 5 | 
             
              prerelease: false
         | 
| 6 6 | 
             
              segments: 
         | 
| 7 7 | 
             
              - 1
         | 
| 8 8 | 
             
              - 0
         | 
| 9 | 
            -
              -  | 
| 10 | 
            -
              version: 1.0. | 
| 9 | 
            +
              - 21
         | 
| 10 | 
            +
              version: 1.0.21
         | 
| 11 11 | 
             
            platform: x86-mswin32-60
         | 
| 12 12 | 
             
            authors: 
         | 
| 13 13 | 
             
            - Scott Messier
         | 
| @@ -17,7 +17,7 @@ autorequire: | |
| 17 17 | 
             
            bindir: bin
         | 
| 18 18 | 
             
            cert_chain: []
         | 
| 19 19 |  | 
| 20 | 
            -
            date: 2012- | 
| 20 | 
            +
            date: 2012-04-25 00:00:00 -07:00
         | 
| 21 21 | 
             
            default_executable: 
         | 
| 22 22 | 
             
            dependencies: 
         | 
| 23 23 | 
             
            - !ruby/object:Gem::Dependency 
         |