{
  "id": "pvlib__pvlib-python-1707",
  "question": "regression: iam.physical returns nan for aoi > 90° when n = 1\n**Describe the bug**\r\nFor pvlib==0.9.5, when n = 1 (no reflection) and aoi > 90°, we get nan as result.\r\n\r\n**To Reproduce**\r\n```python\r\nimport pvlib\r\npvlib.iam.physical(aoi=100, n=1)\r\n```\r\nreturns `nan`.\r\n\r\n**Expected behavior**\r\nThe result should be `0`, as it was for pvlib <= 0.9.4.\r\n\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: '0.9.5'\r\n - ``pandas.__version__``:  '1.5.3'\r\n - python: 3.10.4\r\n\n",
  "answer": "{\"task_id\": \"eb4ca153-39a3-4f06-b0d5-302d28f79c37\", \"workflow_run_id\": \"b5772bc8-77d2-4930-89b5-a85eb485e16f\", \"data\": {\"id\": \"b5772bc8-77d2-4930-89b5-a85eb485e16f\", \"workflow_id\": \"4faf996e-a613-49ba-90e4-a0af9c740cdf\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/iam.py b/pvlib/iam.py\\\\nindex a7f5d93..e5d421f 100644\\\\n--- a/pvlib/iam.py\\\\n+++ b/pvlib/iam.py\\\\n@@ -12,7 +12,11 @@ def physical(aoi, n=1.526, K=4.0):\\\\n     \\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n     \\\\n-    aoir = np.radians(aoi)\\\\n+    # Return 0 for AOI > 90\\u00b0 when n = 1\\\\n+    if aoi > 90.0 and n == 1:\\\\n+        return 0.0\\\\n+\\\\n+    aoir = np.radians(aoi)\\\\n     aoic = np.arccos(1 / n)\\\\n \\\\n     # rest of the function\\\\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 correctly addresses the issue by adding a clear comment and handling the edge case at the beginning of the function. It maintains code readability and follows best practices.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/iam.py b/pvlib/iam.py\\\\nindex a7f5d93..f2ad007 100644\\\\n--- a/pvlib/iam.py\\\\n+++ b/pvlib/iam.py\\\\n@@ -10,6 +10,11 @@ def physical(aoi, n=1.526, K=4.0):\\\\n \\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n \\\\n+    # Handle edge case for n = 1 and aoi > 90\\u00b0\\\\n+    if n == 1:\\\\n+        if aoi > 90.0:\\\\n+            return 0.0\\\\n+\\\\n     aoir = np.radians(aoi)\\\\n     aoic = np.arccos(1 / n)\\\\n \\\\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.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch also correctly handles the edge case but uses nested if statements, which slightly reduces readability. It is still a valid and effective solution.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/iam.py b/pvlib/iam.py\\\\nindex a7f5d93..c1fe2e4 100644\\\\n--- a/pvlib/iam.py\\\\n+++ b/pvlib/iam.py\\\\n@@ -11,6 +11,10 @@ def physical(aoi, n=1.526, K=4.0):\\\\n \\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n \\\\n+    if aoi > 90.0 and n == 1:\\\\n+        return 0.0\\\\n+\\\\n     aoir = np.radians(aoi)\\\\n     aoic = np.arccos(1 / n)\\\\n\\\\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.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch addresses the issue but lacks comments, which can reduce maintainability and understanding for future developers. It is a straightforward fix but could benefit from additional context.\\\",\\n      \\\"recommendation\\\": \\\"Acceptable\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"All patches effectively address the issue of returning NaN for AOI > 90\\u00b0 when n = 1 by returning 0. Patch 2 is ranked highest due to its clear comment and placement of the conditional check, which enhances readability and maintainability. Patch 3 is slightly less readable due to nested conditions. Patch 1, while correct, lacks comments, making it less 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    result = pvlib.iam.physical(aoi=100, n=1)\\\\n    assert np.isnan(result)\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue where pvlib.iam.physical returns nan for aoi > 90\\u00b0 when n = 1\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail before the patch is applied\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_aoi_less_than_90\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_aoi_less_than_90():\\\\n    import pvlib\\\\n    result = pvlib.iam.physical(aoi=80, n=1)\\\\n    assert result == 0.0\\\",\\n      \\\"description\\\": \\\"Test edge case where aoi is less than 90\\u00b0 and n = 1\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass as the result is expected to be 0.0\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_n_not_equal_to_1\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_n_not_equal_to_1():\\\\n    import pvlib\\\\n    result = pvlib.iam.physical(aoi=100, n=1.5)\\\\n    assert np.isnan(result) == True\\\",\\n      \\\"description\\\": \\\"Test edge case where n is not equal to 1\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail as the result is expected to be nan\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch1():\\\\n    import pvlib\\\\n    result = pvlib.iam.physical(aoi=100, n=1)\\\\n    assert result == 0.0\\\",\\n      \\\"description\\\": \\\"Validate Patch 1: Handle edge case for n = 1 and aoi > 90\\u00b0 by returning 0.0\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass with the result being 0.0\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch2():\\\\n    import pvlib\\\\n    result = pvlib.iam.physical(aoi=100, n=1)\\\\n    assert result == 0.0\\\",\\n      \\\"description\\\": \\\"Validate Patch 2: Handle edge case for n = 1 and aoi > 90\\u00b0 by returning 0.0\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass with the result being 0.0\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch3():\\\\n    import pvlib\\\\n    result = pvlib.iam.physical(aoi=100, n=1)\\\\n    assert result == 0.0\\\",\\n      \\\"description\\\": \\\"Validate Patch 3: Handle edge case for n = 1 and aoi > 90\\u00b0 by returning 0.0\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass with the result being 0.0\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated comprehensive test cases including reproducing the original issue, testing edge cases, and validating the provided patches for the bug fix in pvlib.iam.physical function.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 273.883862, \"total_tokens\": 17635, \"total_steps\": 9, \"created_at\": 1753370685, \"finished_at\": 1753370959}}"
}