Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allowing for small float rounding errors when calculating column widths #639

Closed
wants to merge 4 commits into from

Conversation

hbrandl
Copy link
Contributor

@hbrandl hbrandl commented Jan 15, 2014

I found another bug in the table width calculation algorithm.

In some cases arithmetic errors lead to errors.

Testcase to illustrate the problem

    it "table widths should be correctly calculated even for column widths with lots of decimals", :focus do
      data=[["a", "b ", "c ", "d", "e", "f", "g", "h", "i", "j", "k", "l"],
            [{:content=>"Foobar", :colspan=>12}]
          ]
      #we need values with lots of decimals so that arithmetic errors will occur
      #the values are not arbitrary but where found converting mm to pdf pt
      column_widths=[137, 40, 40, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678]
      pdf = Prawn::Document.new({:page_size => 'A4', :page_layout => :landscape})
      table = Prawn::Table.new data, pdf, :column_widths => column_widths
      table.column_widths.should == column_widths
    end

Output of current master

1) Prawn::Table You can explicitly set the column widths and use a colspan > 1 table widths should be correctly calculated even for column widths with lots of decimals
     Failure/Error: table.column_widths.should == column_widths
       expected: [137, 40, 40, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678, 54.69291338582678]
            got: [59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342, 59.10301837270342] (using ==)
     # ./spec/table_spec.rb:59:in `block (3 levels) in <top (required)>'

This pull request addresses the issue by allowing for an small arithmetic error (e-09) in the two affected algorithms.

@hbrandl
Copy link
Contributor Author

hbrandl commented Jan 15, 2014

The test case only addresses the change in column_width_calculator.rb.
I'm pretty sure that the change in cells.rb addresses a bug too. But I don't know how to produce the floating point arithmetic error needed to prove it.

@practicingruby
Copy link
Member

@bradediger I am curious of your thoughts on the problem and the fix here. @hbrandl and I chatted over IRC about it a bit and both agree it's not ideal.

Should we thinking about using BigDecimal here? Or something else entirely? Or just stick with the fuzzy float comparison?

@fidothe
Copy link
Contributor

fidothe commented Jan 21, 2014

I think you want to avoid BigDecimal like the plague here. PostScript Points -> any sane measurement system involves lots of silly floating point, BigDecimal assumes something rather saner. I was wondering whether some precision limit was worth enforcing (AFAICT InDesign et al don't output anything like the precision Prawn does, they round out at 3 or 4 places (IIRC)). That might, in turn, make this kind of problem largely go away. [I realise that this is not an entirely helpful comment in this context]

@bradediger
Copy link
Member

@sandal Yes, I think using a FP epsilon and changing equality tests to closeness tests is the best (least insane) way to go here. BigDecimal is going to be a huge rabbit hole that offers little payoff, IMO.

@practicingruby
Copy link
Member

@bradediger OK, Should we decide on a precision and create a constant for it? I feel like 10^-9 is more than enough, but we could use something close to machine precision via Float::EPSILON. I just always forget the right way to do those calculations 😢

@bradediger
Copy link
Member

@sandal Errors accumulate during consecutive numerical operations, so we can't necessarily just use the machine epsilon. But I definitely think we could standardize on an epsilon around 1e-9 (give or take a few orders of magnitude) for all FP comparisons.

@practicingruby
Copy link
Member

Okay, so it sounds like we just need to add a test for the uncovered case here, and clean up the checks a little bit. I'm unblocked to work on this, but won't get around to it for at least another week. Anyone who wants to take a stab at this sooner than that is welcome to do so.

@practicingruby
Copy link
Member

I can't really think of how to construct the test case the possible rounding error in the natural column widths code, so I think we'll need to build one in response to a regression rather than proactively.
That said, if someone has an idea of how to pin this behavior down, please submit a pull request with an updated spec.

I made some minor tweaks, and I'm going to go ahead and squash+merge this now.

practicingruby added a commit that referenced this pull request Feb 3, 2014
This is a squashed commit of the following:

commit 58fa18f
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 12:05:10 2014 -0500

    Wrap spec text

commit 8565dca
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 11:59:35 2014 -0500

    Extract float precision constant and clarify documentation

commit 54432f9
Merge: f2b2638 7ea6dad
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 11:47:13 2014 -0500

    Merge branch 'master' of github.com:prawnpdf/prawn into column_width_arithmetic_errors

commit f2b2638
Author: Hartwig Brandl <[email protected]>
Date:   Wed Jan 15 20:51:38 2014 +0100

    allowing for small arithmetic errors when calculating column widths
practicingruby added a commit that referenced this pull request Feb 3, 2014
This is a squashed commit of the following:

commit 58fa18f
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 12:05:10 2014 -0500

    Wrap spec text

commit 8565dca
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 11:59:35 2014 -0500

    Extract float precision constant and clarify documentation

commit 54432f9
Merge: f2b2638 7ea6dad
Author: Gregory Brown <[email protected]>
Date:   Mon Feb 3 11:47:13 2014 -0500

    Merge branch 'master' of github.com:prawnpdf/prawn into column_width_arithmetic_errors

commit f2b2638
Author: Hartwig Brandl <[email protected]>
Date:   Wed Jan 15 20:51:38 2014 +0100

    allowing for small arithmetic errors when calculating column widths
@practicingruby practicingruby deleted the column_width_arithmetic_errors branch February 16, 2014 14:29
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

4 participants