{
  "id": "pvlib__pvlib-python-1191",
  "question": "irradiance.aoi can return NaN when module orientation is perfectly aligned with solar position\n**Describe the bug**\r\nI was playing with a dual-axis tracking mount with #1176 and found that when the modules are perfectly aligned with the sun (i.e. AOI should be exactly zero), floating point round-off can result in aoi projection values slightly greater than one, resulting in NaN aoi.  This only happens for some perfectly-aligned inputs (for example tilt=zenith=20, azimuth=180 returns aoi=0 as expected).\r\n\r\n**To Reproduce**\r\n```python\r\nimport pvlib\r\nzenith = 89.26778228223463\r\nazimuth = 60.932028605997004\r\nprint(pvlib.irradiance.aoi_projection(zenith, azimuth, zenith, azimuth))\r\nprint(pvlib.irradiance.aoi(zenith, azimuth, zenith, azimuth))\r\n\r\n# output:\r\n1.0000000000000002\r\nRuntimeWarning: invalid value encountered in arccos:  aoi_value = np.rad2deg(np.arccos(projection))\r\nnan\r\n```\r\n\r\n**Expected behavior**\r\nI expect aoi=0 whenever module orientation and solar position angles are identical.\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: `0.9.0-alpha.4+14.g61650e9`\r\n - ``pandas.__version__``: `0.25.1`\r\n - ``numpy.__version__``: `1.17.0`\r\n - python: `3.7.7 (default, May  6 2020, 11:45:54) [MSC v.1916 64 bit (AMD64)]`\r\n\r\n**Additional context**\r\nSome ideas for fixes:\r\n1) In `irradiance.aoi_projection`, return a hard-coded `1.0` for inputs within some small tolerance\r\n2) In `irradiance.aoi_projection`, clamp return value to `[-1, +1]`\r\n3) In `irradiance.aoi`, clamp aoi_projection values to `[-1, +1`] before calling `arccos`\r\n4) Rework the `irradiance.aoi_projection` trig equations to not generate impossible values?\n",
  "answer": "{\"task_id\": \"48918ecf-8794-4752-9795-facfe7fc2960\", \"workflow_run_id\": \"77491c59-5725-47e8-9e4a-1fd4862d83f1\", \"data\": {\"id\": \"77491c59-5725-47e8-9e4a-1fd4862d83f1\", \"workflow_id\": \"4faf996e-a613-49ba-90e4-a0af9c740cdf\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\\\\nindex fedcba9..8765432 100644\\\\n--- a/pvlib/irradiance.py\\\\n+++ b/pvlib/irradiance.py\\\\n@@ -60,6 +60,10 @@ def aoi_projection(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n     return projection\\\\n \\\\n def aoi(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n+    def safe_arccos(x):\\\\n+        return np.arccos(np.clip(x, -1, 1))\\\\n+\\\\n     projection = aoi_projection(zenith, azimuth, surf_tilt, surf_azimuth)\\\\n-    aoi_value = np.rad2deg(np.arccos(projection))\\\\n+    aoi_value = np.rad2deg(safe_arccos(projection))\\\\n     return aoi_value\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch introduces a helper function `safe_arccos` to ensure the input to `np.arccos` is within the valid range, effectively preventing NaN results. It is a clean and maintainable solution that adheres to best practices by encapsulating the clamping logic.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/pvlib/irradiance.py\\\\n+++ b/pvlib/irradiance.py\\\\n@@ -60,6 +60,7 @@ def aoi_projection(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n     return projection\\\\n+\\\\n def aoi(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n     projection = aoi_projection(zenith, azimuth, surf_tilt, surf_azimuth)\\\\n+    projection = np.clip(projection, -1, 1)  # Clamping the projection to [-1, 1]\\\\n     aoi_value = np.rad2deg(np.arccos(projection))\\\\n     return aoi_value\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch clamps the projection value directly in the `aoi` function. It is a straightforward solution that effectively prevents NaN results, but it slightly mixes concerns by handling clamping within the main function logic.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\\\\nindex 89abcde..1234567 100644\\\\n--- a/pvlib/irradiance.py\\\\n+++ b/pvlib/irradiance.py\\\\n@@ -55,8 +55,11 @@ def aoi_projection(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n     # Compute the angle of incidence projection\\\\n     projection = cosd(zenith) * cosd(surf_tilt) +\\\\n                  sind(zenith) * sind(surf_tilt) * cosd(azimuth - surf_azimuth)\\\\n+    # Handle floating point precision errors\\\\n+    if np.isclose(projection, 1.0):\\\\n+        return 1.0\\\\n     return projection\\\\n \\\\n def aoi(zenith, azimuth, surf_tilt, surf_azimuth):\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch addresses the issue by checking if the projection is close to 1.0 and returning 1.0 directly. While it solves the problem, it introduces a special case that might not be as robust as clamping, and it could lead to maintenance challenges if similar issues arise with other values.\\\",\\n      \\\"recommendation\\\": \\\"Acceptable\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 3 is the best solution as it encapsulates the clamping logic in a helper function, maintaining clean and maintainable code. Patch 1 is a good alternative but mixes logic within the main function. Patch 2, while effective, introduces a special case that might not be as robust or maintainable.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    import pvlib\\\\n    import numpy as np\\\\n    zenith = 89.26778228223463\\\\n    azimuth = 60.932028605997004\\\\n    aoi_projection_value = pvlib.irradiance.aoi_projection(zenith, azimuth, zenith, azimuth)\\\\n    aoi_value = pvlib.irradiance.aoi(zenith, azimuth, zenith, azimuth)\\\\n    assert np.isnan(aoi_value), 'AOI value should be NaN'\\\\n    assert np.isclose(aoi_projection_value, 1.0000000000000002), 'AOI projection value should be 1.0000000000000002'\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue where a perfectly aligned module orientation with solar position results in NaN for AOI\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail before the patches\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    import pvlib\\\\n    import numpy as np\\\\n    # Test for edge case with perfectly aligned inputs that should result in AOI = 0\\\\n    zenith = 0\\\\n    azimuth = 0\\\\n    aoi_value = pvlib.irradiance.aoi(zenith, azimuth, zenith, azimuth)\\\\n    assert aoi_value == 0, 'AOI value should be 0 for perfectly aligned inputs'\\\\n\\\",\\n      \\\"description\\\": \\\"This test covers edge cases where module orientation and solar position are perfectly aligned\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass both before and after the patches\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    import pvlib\\\\n    import numpy as np\\\\n    # Testing the patched behavior for a perfectly aligned input\\\\n    zenith = 89.26778228223463\\\\n    azimuth = 60.932028605997004\\\\n    aoi_projection_value = pvlib.irradiance.aoi_projection(zenith, azimuth, zenith, azimuth)\\\\n    aoi_value = pvlib.irradiance.aoi(zenith, azimuth, zenith, azimuth)\\\\n    assert not np.isnan(aoi_value), 'AOI value should not be NaN'\\\\n    assert aoi_value == 0, 'AOI value should be 0 for perfectly aligned inputs'\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates that the patch correctly handles the issue with NaN AOI values for perfectly aligned inputs\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass after applying the patches\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original bug, cover edge cases, and validate the effectiveness of the patches\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 156.421514, \"total_tokens\": 15074, \"total_steps\": 9, \"created_at\": 1753371578, \"finished_at\": 1753371734}}"
}