Skip to content

Failed unit test for nrel 2008 solar position algorithm #2077

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

Closed
kurt-rhee opened this issue Jun 5, 2024 · 5 comments · Fixed by #2249
Closed

Failed unit test for nrel 2008 solar position algorithm #2077

kurt-rhee opened this issue Jun 5, 2024 · 5 comments · Fixed by #2249

Comments

@kurt-rhee
Copy link
Contributor

Describe the bug
In the documentation for the nrel 2008 solar position algorithm, there are a set of "unit tests" in table A4.1

image
https://www.nrel.gov/docs/fy08osti/34302.pdf

To Reproduce
Steps to reproduce the behavior:

  1. run julian_day_dt() with the following parameters
year = 837,
month = 4,
day = 10,
hour = 7,
minute = 12,
second = 0,
microsecond = 0
  1. result is 2026867.8

Expected behavior
expected result in table A4.1 is 2026871.8

Screenshots
image

Versions:

  • ``pvlib.10.5`:
  • python: 3.11

Additional context
Perhaps this has something to do with the gregorian / julian date cutoff? All of the other tests complete as expected.

@kandersolar
Copy link
Member

kandersolar commented Jun 5, 2024

Perhaps this has something to do with the gregorian / julian date cutoff?

It seems so. I reproduce the expected values exactly using this modified version of the function that follows the text description around equation 4 regarding the calendar difference:

import pvlib

def julian_day_dt(year, month, day, hour, minute, second, microsecond):
    """This is the original way to calculate the julian day from the NREL paper.
    However, it is much faster to convert to unix/epoch time and then convert
    to julian day. Note that the date must be UTC."""
    if month <= 2:
        year = year-1
        month = month+12
    a = int(year/100)
    b = 2 - a + int(a * 0.25)
    frac_of_day = (microsecond / 1e6 + (second + minute * 60 + hour * 3600)
                   ) * 1.0 / (3600*24)
    d = day + frac_of_day
    jd = int(365.25 * (year + 4716)) + int(30.6001 * (month + 1)) + d - 1524.5
    if jd > 2299160.0:
        jd += b

    return jd


tests = [
    ((2000, 1, 1, 12, 0, 0), 2451545.0),
    ((1999, 1, 1, 0, 0, 0), 2451179.5),
    ((1987, 1, 27, 0, 0, 0), 2446822.5),
    ((1987, 6, 19, 12, 0, 0), 2446966.0),
    ((1988, 1, 27, 0, 0, 0), 2447187.5),
    ((1988, 6, 19, 12, 0, 0), 2447332.0),
    ((1900, 1, 1, 0, 0, 0), 2415020.5),
    ((1600, 1, 1, 0, 0, 0), 2305447.5),
    
    ((1600, 12, 31, 0, 0, 0), 2305812.5),
    ((837, 4, 10, 7, 12, 0), 2026871.8),
    ((-123, 12, 31, 0, 0, 0), 1676496.5),
    ((-122, 1, 1, 0, 0, 0), 1676497.5),
    ((-1000, 7, 12, 12, 0, 0), 1356001.0),
    ((-1000, 2, 29, 0, 0, 0), 1355866.5),
    ((-1001, 8, 17, 21, 36, 0), 1355671.4),
    ((-4712, 1, 1, 12, 0, 0), 0.0),
]

print("pvlib\tmodified")
for inputs, expected in tests:
    ret1 = pvlib.spa.julian_day_dt(*inputs, microsecond=0)
    ret2 = julian_day_dt(*inputs, microsecond=0)
    print(ret1 - expected, '\t', ret2 - expected)

@kurt-rhee
Copy link
Contributor Author

Nice catch @kandersolar! Quickest issue resolution of all time.

Do you want me to put in a PR for this or is it small enough that an outside commit isn't worth it.

@kandersolar
Copy link
Member

It's worth fixing IMHO. We know our current implementation doesn't follow its reference, and we even have test cases to prove it. This function isn't used in pvlib's main SPA calculation flow, so I doubt this issue affects many (if any) real-life applications, but in service of pvlib's goal of providing "reference" implementations I think it deserves to be fixed. A PR would be welcome!

@kurt-rhee
Copy link
Contributor Author

kurt-rhee commented Jun 6, 2024 via email

@kurt-rhee
Copy link
Contributor Author

On second thought, @kylefmacdonald perhaps you would like a entry level introduction to contributing to pvlib?

dgapitts added a commit to dgapitts/pvlib-python that referenced this issue Oct 9, 2024
@kandersolar kandersolar added this to the v0.11.2 milestone Oct 10, 2024
@kandersolar kandersolar linked a pull request Oct 25, 2024 that will close this issue
3 tasks
kandersolar added a commit that referenced this issue Oct 25, 2024
#2249)

* fix to pvlib/spa.py for issue #2077

* add-test_julian_day_issue_2207

* correct-test-indention-and-class

* fix-typos-cleanup

* fix-linting-errors-add-blank-lines

* 2 blank lines before a function, one blank line at the end of a file.

* one last linter fix

* add whatsnew entry, contributors

---------

Co-authored-by: Kevin Anderson <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants