require_relative '../../spec_helper' require_relative 'shared/quo' require 'bigdecimal' describe "BigDecimal#div with precision set to 0" do # TODO: figure out if there is a better way to do these # shared specs rather than sending [0]. See other specs # that share :bigdecimal_quo. it_behaves_like :bigdecimal_quo, :div, [0] end describe "BigDecimal#div" do before :each do @one = BigDecimal("1") @zero = BigDecimal("0") @zero_plus = BigDecimal("+0") @zero_minus = BigDecimal("-0") @two = BigDecimal("2") @three = BigDecimal("3") @nan = BigDecimal("NaN") @infinity = BigDecimal("Infinity") @infinity_minus = BigDecimal("-Infinity") @one_minus = BigDecimal("-1") @frac_1 = BigDecimal("1E-99999") @frac_2 = BigDecimal("0.9E-99999") end it "returns a / b with optional precision" do @two.div(@one).should == @two @one.div(@two).should == @zero # ^^ is this really intended for a class with arbitrary precision? @one.div(@two, 1).should == BigDecimal("0.5") @one.div(@one_minus).should == @one_minus @one_minus.div(@one_minus).should == @one @frac_2.div(@frac_1, 1).should == BigDecimal("0.9") @frac_1.div(@frac_1).should == @one res = "0." + "3" * 1000 (1..100).each { |idx| @one.div(@three, idx).to_s("F").should == "0." + res[2, idx] } end describe "with Object" do it "tries to coerce the other operand to self" do object = mock("Object") object.should_receive(:coerce).with(@one).and_return([@one, @two]) @one.div(object).should == @zero end end it "raises FloatDomainError if NaN is involved" do -> { @one.div(@nan) }.should raise_error(FloatDomainError) -> { @nan.div(@one) }.should raise_error(FloatDomainError) -> { @nan.div(@nan) }.should raise_error(FloatDomainError) end it "returns 0 if divided by Infinity and no precision given" do @zero.div(@infinity).should == 0 @frac_2.div(@infinity).should == 0 end it "returns 0 if divided by Infinity with given precision" do @zero.div(@infinity, 0).should == 0 @frac_2.div(@infinity, 1).should == 0 @zero.div(@infinity, 100000).should == 0 @frac_2.div(@infinity, 100000).should == 0 end it "raises ZeroDivisionError if divided by zero and no precision given" do -> { @one.div(@zero) }.should raise_error(ZeroDivisionError) -> { @one.div(@zero_plus) }.should raise_error(ZeroDivisionError) -> { @one.div(@zero_minus) }.should raise_error(ZeroDivisionError) -> { @zero.div(@zero) }.should raise_error(ZeroDivisionError) -> { @zero_minus.div(@zero_plus) }.should raise_error(ZeroDivisionError) -> { @zero_minus.div(@zero_minus) }.should raise_error(ZeroDivisionError) -> { @zero_plus.div(@zero_minus) }.should raise_error(ZeroDivisionError) end it "returns NaN if zero is divided by zero" do @zero.div(@zero, 0).should.nan? @zero_minus.div(@zero_plus, 0).should.nan? @zero_plus.div(@zero_minus, 0).should.nan? @zero.div(@zero, 10).should.nan? @zero_minus.div(@zero_plus, 10).should.nan? @zero_plus.div(@zero_minus, 10).should.nan? end it "raises FloatDomainError if (+|-) Infinity divided by 1 and no precision given" do -> { @infinity_minus.div(@one) }.should raise_error(FloatDomainError) -> { @infinity.div(@one) }.should raise_error(FloatDomainError) -> { @infinity_minus.div(@one_minus) }.should raise_error(FloatDomainError) end it "returns (+|-)Infinity if (+|-)Infinity by 1 and precision given" do @infinity_minus.div(@one, 0).should == @infinity_minus @infinity.div(@one, 0).should == @infinity @infinity_minus.div(@one_minus, 0).should == @infinity end it "returns NaN if Infinity / ((+|-) Infinity)" do @infinity.div(@infinity_minus, 100000).should.nan? @infinity_minus.div(@infinity, 1).should.nan? end end