mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
95e8c48dd3
* For easier modifications of ruby/spec by MRI developers. * .gitignore: track changes under spec. * spec/mspec, spec/rubyspec: add in-tree mspec and ruby/spec. These files can therefore be updated like any other file in MRI. Instructions are provided in spec/README. [Feature #13156] [ruby-core:79246] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@58595 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
313 lines
8.4 KiB
Ruby
313 lines
8.4 KiB
Ruby
# -*- encoding: utf-8 -*-
|
|
require File.expand_path('../../../spec_helper', __FILE__)
|
|
require File.expand_path('../fixtures/classes', __FILE__)
|
|
require File.expand_path('../shared/gets_ascii', __FILE__)
|
|
|
|
describe "IO#gets" do
|
|
it_behaves_like :io_gets_ascii, :gets
|
|
end
|
|
|
|
describe "IO#gets" do
|
|
before :each do
|
|
@io = IOSpecs.io_fixture "lines.txt"
|
|
@count = 0
|
|
end
|
|
|
|
after :each do
|
|
@io.close if @io
|
|
end
|
|
|
|
it "assigns the returned line to $_" do
|
|
IOSpecs.lines.each do |line|
|
|
@io.gets
|
|
$_.should == line
|
|
end
|
|
end
|
|
|
|
it "returns nil if called at the end of the stream" do
|
|
IOSpecs.lines.length.times { @io.gets }
|
|
@io.gets.should == nil
|
|
end
|
|
|
|
it "raises IOError on closed stream" do
|
|
lambda { IOSpecs.closed_io.gets }.should raise_error(IOError)
|
|
end
|
|
|
|
describe "with no separator" do
|
|
it "returns the next line of string that is separated by $/" do
|
|
IOSpecs.lines.each { |line| line.should == @io.gets }
|
|
end
|
|
|
|
it "returns tainted strings" do
|
|
while line = @io.gets
|
|
line.tainted?.should == true
|
|
end
|
|
end
|
|
|
|
it "updates lineno with each invocation" do
|
|
while @io.gets
|
|
@io.lineno.should == @count += 1
|
|
end
|
|
end
|
|
|
|
it "updates $. with each invocation" do
|
|
while @io.gets
|
|
$..should == @count += 1
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "with nil separator" do
|
|
it "returns the entire contents" do
|
|
@io.gets(nil).should == IOSpecs.lines.join("")
|
|
end
|
|
|
|
it "returns tainted strings" do
|
|
while line = @io.gets(nil)
|
|
line.tainted?.should == true
|
|
end
|
|
end
|
|
|
|
it "updates lineno with each invocation" do
|
|
while @io.gets(nil)
|
|
@io.lineno.should == @count += 1
|
|
end
|
|
end
|
|
|
|
it "updates $. with each invocation" do
|
|
while @io.gets(nil)
|
|
$..should == @count += 1
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "with an empty String separator" do
|
|
# Two successive newlines in the input separate paragraphs.
|
|
# When there are more than two successive newlines, only two are kept.
|
|
it "returns the next paragraph" do
|
|
@io.gets("").should == IOSpecs.lines[0,3].join("")
|
|
@io.gets("").should == IOSpecs.lines[4,3].join("")
|
|
@io.gets("").should == IOSpecs.lines[7,2].join("")
|
|
end
|
|
|
|
it "reads until the beginning of the next paragraph" do
|
|
# There are three newlines between the first and second paragraph
|
|
@io.gets("")
|
|
@io.gets.should == IOSpecs.lines[4]
|
|
end
|
|
|
|
it "returns tainted strings" do
|
|
while line = @io.gets("")
|
|
line.tainted?.should == true
|
|
end
|
|
end
|
|
|
|
it "updates lineno with each invocation" do
|
|
while @io.gets("")
|
|
@io.lineno.should == @count += 1
|
|
end
|
|
end
|
|
|
|
it "updates $. with each invocation" do
|
|
while @io.gets("")
|
|
$..should == @count += 1
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "with an arbitrary String separator" do
|
|
it "reads up to and including the separator" do
|
|
@io.gets("la linea").should == "Voici la ligne une.\nQui \303\250 la linea"
|
|
end
|
|
|
|
it "returns tainted strings" do
|
|
while line = @io.gets("la")
|
|
line.tainted?.should == true
|
|
end
|
|
end
|
|
|
|
it "updates lineno with each invocation" do
|
|
while (@io.gets("la"))
|
|
@io.lineno.should == @count += 1
|
|
end
|
|
end
|
|
|
|
it "updates $. with each invocation" do
|
|
while @io.gets("la")
|
|
$..should == @count += 1
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "IO#gets" do
|
|
before :each do
|
|
@name = tmp("io_gets")
|
|
end
|
|
|
|
after :each do
|
|
rm_r @name
|
|
end
|
|
|
|
it "raises an IOError if the stream is opened for append only" do
|
|
lambda { File.open(@name, fmode("a:utf-8")) { |f| f.gets } }.should raise_error(IOError)
|
|
end
|
|
|
|
it "raises an IOError if the stream is opened for writing only" do
|
|
lambda { File.open(@name, fmode("w:utf-8")) { |f| f.gets } }.should raise_error(IOError)
|
|
end
|
|
end
|
|
|
|
describe "IO#gets" do
|
|
before :each do
|
|
@name = tmp("io_gets")
|
|
touch(@name) { |f| f.write "one\n\ntwo\n\nthree\nfour\n" }
|
|
@io = new_io @name, fmode("r:utf-8")
|
|
end
|
|
|
|
after :each do
|
|
@io.close if @io
|
|
rm_r @name
|
|
end
|
|
|
|
it "calls #to_int to convert a single object argument to an Integer limit" do
|
|
obj = mock("io gets limit")
|
|
obj.should_receive(:to_int).and_return(6)
|
|
|
|
@io.gets(obj).should == "one\n"
|
|
end
|
|
|
|
it "calls #to_int to convert the second object argument to an Integer limit" do
|
|
obj = mock("io gets limit")
|
|
obj.should_receive(:to_int).and_return(2)
|
|
|
|
@io.gets(nil, obj).should == "on"
|
|
end
|
|
|
|
it "calls #to_str to convert the first argument to a String when passed a limit" do
|
|
obj = mock("io gets separator")
|
|
obj.should_receive(:to_str).and_return($/)
|
|
|
|
@io.gets(obj, 5).should == "one\n"
|
|
end
|
|
|
|
it "reads to the default seperator when passed a single argument greater than the number of bytes to the separator" do
|
|
@io.gets(6).should == "one\n"
|
|
end
|
|
|
|
it "reads limit bytes when passed a single argument less than the number of bytes to the default separator" do
|
|
@io.gets(3).should == "one"
|
|
end
|
|
|
|
it "reads limit bytes when passed nil and a limit" do
|
|
@io.gets(nil, 6).should == "one\n\nt"
|
|
end
|
|
|
|
it "reads all bytes when the limit is higher than the available bytes" do
|
|
@io.gets(nil, 100).should == "one\n\ntwo\n\nthree\nfour\n"
|
|
end
|
|
|
|
it "reads until the next paragraph when passed '' and a limit greater than the next paragraph" do
|
|
@io.gets("", 6).should == "one\n\n"
|
|
end
|
|
|
|
it "reads limit bytes when passed '' and a limit less than the next paragraph" do
|
|
@io.gets("", 3).should == "one"
|
|
end
|
|
|
|
it "reads all bytes when pass a separator and reading more than all bytes" do
|
|
@io.gets("\t", 100).should == "one\n\ntwo\n\nthree\nfour\n"
|
|
end
|
|
end
|
|
|
|
describe "IO#gets" do
|
|
before :each do
|
|
@name = tmp("io_gets")
|
|
# create data "朝日" + "\xE3\x81" * 100 to avoid utf-8 conflicts
|
|
data = "朝日" + ([227,129].pack('C*') * 100).force_encoding('utf-8')
|
|
touch(@name) { |f| f.write data }
|
|
@io = new_io @name, fmode("r:utf-8")
|
|
end
|
|
|
|
after :each do
|
|
@io.close if @io
|
|
rm_r @name
|
|
end
|
|
|
|
it "reads limit bytes and extra bytes when limit is reached not at character boundary" do
|
|
[@io.gets(1), @io.gets(1)].should == ["朝", "日"]
|
|
end
|
|
|
|
it "read limit bytes and extra bytes with maximum of 16" do
|
|
# create str "朝日\xE3" + "\x81\xE3" * 8 to avoid utf-8 conflicts
|
|
str = "朝日" + ([227] + [129,227] * 8).pack('C*').force_encoding('utf-8')
|
|
@io.gets(7).should == str
|
|
end
|
|
end
|
|
|
|
describe "IO#gets" do
|
|
before :each do
|
|
@external = Encoding.default_external
|
|
@internal = Encoding.default_internal
|
|
|
|
Encoding.default_external = Encoding::UTF_8
|
|
Encoding.default_internal = nil
|
|
|
|
@name = tmp("io_gets")
|
|
touch(@name) { |f| f.write "line" }
|
|
end
|
|
|
|
after :each do
|
|
@io.close if @io
|
|
rm_r @name
|
|
Encoding.default_external = @external
|
|
Encoding.default_internal = @internal
|
|
end
|
|
|
|
it "uses the default external encoding" do
|
|
@io = new_io @name, 'r'
|
|
@io.gets.encoding.should == Encoding::UTF_8
|
|
end
|
|
|
|
it "uses the IO object's external encoding, when set" do
|
|
@io = new_io @name, 'r'
|
|
@io.set_encoding Encoding::US_ASCII
|
|
@io.gets.encoding.should == Encoding::US_ASCII
|
|
end
|
|
|
|
it "transcodes into the default internal encoding" do
|
|
Encoding.default_internal = Encoding::US_ASCII
|
|
@io = new_io @name, 'r'
|
|
@io.gets.encoding.should == Encoding::US_ASCII
|
|
end
|
|
|
|
it "transcodes into the IO object's internal encoding, when set" do
|
|
Encoding.default_internal = Encoding::US_ASCII
|
|
@io = new_io @name, 'r'
|
|
@io.set_encoding Encoding::UTF_8, Encoding::UTF_16
|
|
@io.gets.encoding.should == Encoding::UTF_16
|
|
end
|
|
|
|
it "overwrites the default external encoding with the IO object's own external encoding" do
|
|
Encoding.default_external = Encoding::ASCII_8BIT
|
|
Encoding.default_internal = Encoding::UTF_8
|
|
@io = new_io @name, 'r'
|
|
@io.set_encoding Encoding::IBM866
|
|
@io.gets.encoding.should == Encoding::UTF_8
|
|
end
|
|
|
|
it "ignores the internal encoding if the default external encoding is ASCII-8BIT" do
|
|
Encoding.default_external = Encoding::ASCII_8BIT
|
|
Encoding.default_internal = Encoding::UTF_8
|
|
@io = new_io @name, 'r'
|
|
@io.gets.encoding.should == Encoding::ASCII_8BIT
|
|
end
|
|
|
|
it "transcodes to internal encoding if the IO object's external encoding is ASCII-8BIT" do
|
|
Encoding.default_external = Encoding::ASCII_8BIT
|
|
Encoding.default_internal = Encoding::UTF_8
|
|
@io = new_io @name, 'r'
|
|
@io.set_encoding Encoding::ASCII_8BIT, Encoding::UTF_8
|
|
@io.gets.encoding.should == Encoding::UTF_8
|
|
end
|
|
end
|