HEX
Server: Apache
System: Linux s198.coreserver.jp 5.15.0-151-generic #161-Ubuntu SMP Tue Jul 22 14:25:40 UTC 2025 x86_64
User: nagasaki (10062)
PHP: 7.1.33
Disabled: NONE
Upload Files
File: //usr/local/rvm/src/ruby-2.5.9/spec/mspec/lib/mspec/matchers/output_to_fd.rb
require 'mspec/helpers/tmp'

# Lower-level output speccing mechanism for a single
# output stream. Unlike OutputMatcher which provides
# methods to capture the output, we actually replace
# the FD itself so that there is no reliance on a
# certain method being used.
class OutputToFDMatcher
  def initialize(expected, to)
    @to, @expected = to, expected

    case @to
    when STDOUT
      @to_name = "STDOUT"
    when STDERR
      @to_name = "STDERR"
    when IO
      @to_name = @to.object_id.to_s
    else
      raise ArgumentError, "#{@to.inspect} is not a supported output target"
    end
  end

  def with_tmp
    path = tmp("mspec_output_to_#{$$}_#{Time.now.to_i}")
    File.open(path, 'w+') { |io|
      yield(io)
    }
  ensure
    File.delete path if path
  end

  def matches?(block)
    old_to = @to.dup
    with_tmp do |out|
      # Replacing with a file handle so that Readline etc. work
      @to.reopen out
      begin
        block.call
      ensure
        @to.reopen old_to
        old_to.close
      end

      out.rewind
      @actual = out.read

      case @expected
      when Regexp
        !(@actual =~ @expected).nil?
      else
        @actual == @expected
      end
    end
  end

  def failure_message()
    ["Expected (#{@to_name}): #{@expected.inspect}\n",
     "#{'but got'.rjust(@to_name.length + 10)}: #{@actual.inspect}\nBacktrace"]
  end

  def negative_failure_message()
    ["Expected output (#{@to_name}) to NOT be:\n", @actual.inspect]
  end
end

module MSpecMatchers
  private def output_to_fd(what, where = STDOUT)
    OutputToFDMatcher.new what, where
  end
end