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

XIRR financial function not calculating as in Excel #2469

Closed
blacknell opened this issue Dec 27, 2021 · 2 comments · Fixed by #2487
Closed

XIRR financial function not calculating as in Excel #2469

blacknell opened this issue Dec 27, 2021 · 2 comments · Fixed by #2487

Comments

@blacknell
Copy link
Contributor

This is:

- [X] a bug report
- [ ] a feature request
- [ ] **not** a usage question (ask them on https://stackoverflow.com/questions/tagged/phpspreadsheet or https://gitter.im/PHPOffice/PhpSpreadsheet)

What is the expected behavior?

XIRR should be able to return a value -0.6422991 for this simple use case

What is the current behavior?

XIRR returns #NUM!

What are the steps to reproduce?

Please provide a Minimal, Complete, and Verifiable example of code that exhibits the issue without relying on an external Excel file or a web server:

<?php

use PhpOffice\PhpSpreadsheet\Calculation\Financial\CashFlow\Variable\NonPeriodic;

require __DIR__."/vendor/autoload.php";

ini_set('memory_limit', '8192M');
ini_set('display_errors', 1);
ini_set('log_errors', true);

$cashFlows = [55600, -51094.83];
$cashFlowDates = ['24/11/2021', '24/12/2021'];

# Excel returns -0.6422991
echo NonPeriodic::rate($cashFlows, $cashFlowDates).PHP_EOL;

Which versions of PhpSpreadsheet and PHP are affected?

PhpSpreadsheet 1.20.0 (tested with 1.15 also using pre namespace changes)
PHP 7.4 and 8.0

@oleibman
Copy link
Collaborator

oleibman commented Jan 5, 2022

I am able to verify that there is a problem. However, your result -0.6422991 does not quite agree with what I see -0.642308. I have confirmed the latter on several platforms/libraries. Can you please confirm the numbers you specified in your sample code?

XIRR is calculated by making an initial guess and then using an algorithm which is fast but is not guaranteed to converge. Your example falls into one of those non-converging areas with the initial default guess of 0.1. However, noticing that the correct answer is negative, I took a shot and supplied an initial guess of -0.1, and that did converge to the correct result. It is certainly possible to change the code to try the negative of the original guess if the original guess fails. It solves this problem and doesn't break anything. I'm sure that's not the smartest possible approach, but I will at least have a PR out within a day or two that adopts that approach while I study the problem further. Note that the existing unit tests already have at least one example of a converging negative result even with the default guess.

@blacknell
Copy link
Contributor Author

blacknell commented Jan 5, 2022

Firstly, thank you @oleibman for taking a look at this.

Indeed something has been miskeyed because when I re-keyed the above into a new Excel worksheet I get ​-0.64230761304498 - I didn't keep the workbook from before so I can't explain my error above.

In the library, when I enter -0.1 as an initial guess I too get -0.64230760881305. Weirdly, when this known result is used as an initial guess it also fails to converge.

I take your point about the nature of the proposed solution. I don't know how Excel converges but it is doing so with default 0.1 guess. I've rarely (if ever) seen it fail.

Would be good to have this scenario added to the test suite. Let me know if I can help

oleibman pushed a commit to oleibman/PhpSpreadsheet that referenced this issue Jan 6, 2022
Fix PHPOffice#2469. The algorithm used for XIRR is known not to converge in some cases, some of which are because the value is legitimately unsolvable; for others, using a different guess might help.

The algorithm uses continual guesses at a rate to hopefully converge on the solution. The code in Python package xirr (https://github.com/tarioch/xirr/) suggests a refinement when this rate falls below -1. Adopting this refinement solves the problem for the data in issue 2469 without any adverse effect on the existing tests. My thanks to @tarioch for that refinement.

The data from 2469 is, of course, added to the test cases. The user also mentions that an initial guess equal to the actual result doesn't converge either. A test is also added to confirm that that case now works.

The test cases are changed to run in the context of a spreadsheet rather than by direct calls to XIRR calculation routine. This revealed some data validation errors which are also cleaned up with this PR. This suggests that other financial tests might benefit from the same change; I will look into that.
@oleibman oleibman mentioned this issue Jan 6, 2022
5 tasks
oleibman added a commit that referenced this issue Jan 14, 2022
* Refinement for XIRR

Fix #2469. The algorithm used for XIRR is known not to converge in some cases, some of which are because the value is legitimately unsolvable; for others, using a different guess might help.

The algorithm uses continual guesses at a rate to hopefully converge on the solution. The code in Python package xirr (https://github.com/tarioch/xirr/) suggests a refinement when this rate falls below -1. Adopting this refinement solves the problem for the data in issue 2469 without any adverse effect on the existing tests. My thanks to @tarioch for that refinement.

The data from 2469 is, of course, added to the test cases. The user also mentions that an initial guess equal to the actual result doesn't converge either. A test is also added to confirm that that case now works.

The test cases are changed to run in the context of a spreadsheet rather than by direct calls to XIRR calculation routine. This revealed some data validation errors which are also cleaned up with this PR. This suggests that other financial tests might benefit from the same change; I will look into that.

* More Unit Tests

From https://github.com/RayDeCampo/java-xirr/blob/master/src/test/java/org/decampo/xirr/XirrTest.java
https://github.com/tarioch/xirr/blob/master/tests/test_math.py

Note that there are some cases where the PHP tests do not converge, but the non-PHP tests do. I have confirmed in each of those cases that Excel does not converge, so the PhpSpreadsheet results are good, at least for now. The discrepancies are noted in comments in the test member.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging a pull request may close this issue.

2 participants