{
  "id": "pvlib__pvlib-python-1395",
  "question": "Add retrieval function for horizon profile from MINES Paris Tech\n<!-- Thank you for your contribution! The following items must be addressed before the code can be merged. Please don't hesitate to ask for help if you're unsure of how to accomplish any of the items. Feel free to remove checklist items that are not relevant to your change. -->\r\n\r\n - [x] I am familiar with the [contributing guidelines](https://pvlib-python.readthedocs.io/en/latest/contributing.html)\r\n - [x] Tests added\r\n - [x] Updates entries to [`docs/sphinx/source/api.rst`](https://github.com/pvlib/pvlib-python/blob/master/docs/sphinx/source/api.rst) for API changes.\r\n - [x] Adds description and name entries in the appropriate \"what's new\" file in [`docs/sphinx/source/whatsnew`](https://github.com/pvlib/pvlib-python/tree/master/docs/sphinx/source/whatsnew) for all changes. Includes link to the GitHub Issue with `` :issue:`num` `` or this Pull Request with `` :pull:`num` ``. Includes contributor name and/or GitHub username (link with `` :ghuser:`user` ``).\r\n - [x] New code is fully documented. Includes [numpydoc](https://numpydoc.readthedocs.io/en/latest/format.html) compliant docstrings, examples, and comments where necessary.\r\n - [x] Pull request is nearly complete and ready for detailed review.\r\n - [x] Maintainer: Appropriate GitHub Labels and Milestone are assigned to the Pull Request and linked Issue.\r\n\r\n<!-- Brief description of the problem and proposed solution (if not already fully described in the issue linked to above): -->\r\n\r\nThe proposed function retrieves the local horizon profile for a specific location (latitude, longitude, and elevation). The returned horizon profile has a resolution of 1 degree in the azimuth direction. The service is provided by MINES ParisTech though I cannot find any official documentation for it.\r\n\r\nThe function added in this PR (``pvlib.iotools.get_mines_horizon``) is very similar to the function added in #1395 (``pvlib.iotools.get_pvgis_horizon``).\n",
  "answer": "{\"task_id\": \"c66185a9-3c4f-4384-a6d8-935f637d25c9\", \"workflow_run_id\": \"3d6c65b1-30e0-413d-be28-4d398ecb6796\", \"data\": {\"id\": \"3d6c65b1-30e0-413d-be28-4d398ecb6796\", \"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/iotools/mines.py b/pvlib/iotools/mines.py\\\\nindex 0000000..2222222 100644\\\\n--- a/pvlib/iotools/mines.py\\\\n+++ b/pvlib/iotools/mines.py\\\\n@@ -10,7 +10,45 @@\\\\n def get_mines_horizon(latitude, longitude, elevation=None):\\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n     Retrieve the horizon profile from MINES ParisTech.\\\\n\\\\n     Parameters\\\\n     ----------\\\\n     latitude : float\\\\n         Latitude of the location.\\\\n     longitude : float\\\\n         Longitude of the location.\\\\n+    elevation : float, optional\\\\n+        Elevation of the location, by default None.\\\\n+\\\\n+    Returns\\\\n+    -------\\\\n+    DataFrame\\\\n+        A pandas DataFrame with columns 'azimuth' and 'height' representing\\\\n+        the horizon profile.\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+\\\\n+    url = \\\\\\\"https://mines-paris-api/horizon\\\\\\\"\\\\n+    params = {'lat': latitude, 'lon': longitude}\\\\n+    if elevation is not None:\\\\n+        params['elevation'] = elevation\\\\n+    response = requests.get(url, params=params)\\\\n+    response.raise_for_status()\\\\n+    return parse_mines_response(response.text)\\\\n+\\\\n+def parse_mines_response(response_text):\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+    Parse the response from the MINES API.\\\\n+\\\\n+    Parameters\\\\n+    ----------\\\\n+    response_text : str\\\\n+        Raw text from the API response.\\\\n+\\\\n+    Returns\\\\n+    -------\\\\n+    DataFrame\\\\n+        Processed horizon data in a Pandas DataFrame.\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+    from io import StringIO\\\\n+    import pandas as pd\\\\n+\\\\n+    data = StringIO(response_text)\\\\n+    return pd.read_csv(data, sep=':', names=['azimuth', 'height'], header=None)\\\\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 uses a DataFrame to handle the response data, which is more efficient and aligns with best practices for data manipulation in Python. It also uses parameterized requests, reducing the risk of errors in URL construction.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/tests/test_iotools.py b/pvlib/tests/test_iotools.py\\\\nindex 0000000..4444444 100644\\\\n--- a/pvlib/tests/test_iotools.py\\\\n+++ b/pvlib/tests/test_iotools.py\\\\n@@ -100,6 +100,30 @@ def test_get_mines_horizon():\\\\n     \\\\\\\"\\\\\\\"\\\\\\\"Test the get_mines_horizon function.\\\\\\\"\\\\\\\"\\\\\\\"\\\\n     latitude, longitude = 48.8566, 2.3522  # example for Paris\\\\n     elevation = 35  # example elevation\\\\n+    mock_response = \\\\\\\"0:0\\\\\\\\n1:1\\\\\\\\n2:2\\\\\\\\n3:3\\\\\\\\n\\\\\\\"\\\\n+\\\\n+    with patch('requests.get') as mock_get:\\\\n+        # setup the mock to return the mock response\\\\n+        mock_get.return_value.status_code = 200\\\\n+        mock_get.return_value.text = mock_response\\\\n+\\\\n+        # valid case\\\\n+        horizon = get_mines_horizon(latitude, longitude, elevation)\\\\n+        assert len(horizon) == 4\\\\n+        assert horizon[0] == 0\\\\n+        assert horizon[3] == 3\\\\n+\\\\n+        # test without elevation\\\\n+        horizon_no_elev = get_mines_horizon(latitude, longitude)\\\\n+        assert len(horizon_no_elev) == 4\\\\n+\\\\n+        # test network error\\\\n+        mock_get.return_value.status_code = 404\\\\n+        with pytest.raises(requests.exceptions.HTTPError):\\\\n+            get_mines_horizon(latitude, longitude)\\\\n\\\\n+    # test invalid response\\\\n+    invalid_response = \\\\\\\"invalid:text\\\\\\\"\\\\n+    with patch('requests.get') as mock_get:\\\\n+        mock_get.return_value.status_code = 200\\\\n+        mock_get.return_value.text = invalid_response\\\\n+        with pytest.raises(ValueError):\\\\n+            get_mines_horizon(latitude, longitude)\\\\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.95\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds comprehensive tests for the function, covering various scenarios including valid responses, network errors, and invalid responses. It ensures robustness and reliability of the function.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/iotools/mines.py b/pvlib/iotools/mines.py\\\\nindex 0000000..1111111 100644\\\\n--- a/pvlib/iotools/mines.py\\\\n+++ b/pvlib/iotools/mines.py\\\\n@@ -10,7 +10,39 @@\\\\n def get_mines_horizon(latitude, longitude, elevation=None):\\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n     Retrieve the horizon profile from MINES ParisTech.\\\\n\\\\n     Parameters\\\\n     ----------\\\\n     latitude : float\\\\n         Latitude of the location.\\\\n     longitude : float\\\\n         Longitude of the location.\\\\n+    elevation : float, optional\\\\n+        Elevation of the location, by default None.\\\\n+\\\\n+    Returns\\\\n+    -------\\\\n+    dict\\\\n+        Dictionary containing azimuth angles and horizon height in degrees.\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+\\\\n+    url = f\\\\\\\"https://mines-paris-api/horizon?lat={latitude}&lon={longitude}\\\\\\\"\\\\n+    if elevation:\\\\n+        url += f\\\\\\\"&elevation={elevation}\\\\\\\"\\\\n+    response = requests.get(url)\\\\n+    response.raise_for_status()\\\\n+    return parse_mines_response(response.text)\\\\n+\\\\n+def parse_mines_response(response_text):\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+    Parse the response from the MINES API.\\\\n+\\\\n+    Parameters\\\\n+    ----------\\\\n+    response_text : str\\\\n+        Raw text from the API response.\\\\n+\\\\n+    Returns\\\\n+    -------\\\\n+    dict\\\\n+        Processed horizon data.\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+    data = {}\\\\n+    # assume the format is \\\\\\\"azimuth:height\\\\\\\\n\\\\\\\"\\\\n+    for line in response_text.split('\\\\\\\\n'):\\\\n+        if line:\\\\n+            azimuth, height = map(float, line.split(':'))\\\\n+            data[azimuth] = height\\\\n+    return data\\\\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 provides a basic implementation of the function using a dictionary to store the data. While functional, it is less efficient than using a DataFrame and has a higher risk of errors in URL construction.\\\",\\n      \\\"recommendation\\\": \\\"Acceptable, but consider improvements\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/iotools/__init__.py b/pvlib/iotools/__init__.py\\\\nindex 0000000..3333333 100644\\\\n--- a/pvlib/iotools/__init__.py\\\\n+++ b/pvlib/iotools/__init__.py\\\\n@@ -5,4 +5,5 @@\\\\n from . import pvgis\\\\n from . import solcast\\\\n from . import srml\\\\n+\\\\n+from .mines import get_mines_horizon\\\\n\\\\n __all__ = ['get_mines_horizon']\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.8,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds the new function to the module's init file, making it available for import. It is a necessary step but does not directly affect the functionality or quality of the function itself.\\\",\\n      \\\"recommendation\\\": \\\"Necessary for integration\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 5,\\n      \\\"patch\\\": \\\"diff --git a/docs/sphinx/source/api.rst b/docs/sphinx/source/api.rst\\\\nindex 0000000..5555555 100644\\\\n--- a/docs/sphinx/source/api.rst\\\\n+++ b/docs/sphinx/source/api.rst\\\\n@@ -200,6 +200,10 @@ iotools\\\\n     pvlib.iotools.get_tmy2\\\\n     pvlib.iotools.get_tmy3\\\\n     pvlib.iotools.get_solpos\\\\n+\\\\n+.. autofunction:: pvlib.iotools.get_mines_horizon\\\\n+\\\\n+   :noindex:\\\\n\\\\n     pvlib.iotools.get_srml\\\\n     pvlib.iotools.parse_solpy\\\\n\\\",\\n      \\\"rank\\\": 5,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.6\\n      },\\n      \\\"reasoning\\\": \\\"This patch updates the documentation to include the new function. While important for user guidance, it does not impact the functionality or quality of the code.\\\",\\n      \\\"recommendation\\\": \\\"Important for documentation\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the best choice due to its use of a DataFrame for efficient data handling and parameterized requests for safer URL construction. Patch 4 provides comprehensive testing, ensuring robustness. Patch 1 offers a basic implementation but is less efficient. Patches 3 and 5 are necessary for integration and documentation, respectively.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    from pvlib.iotools.mines import get_mines_horizon\\\\n    import requests\\\\n\\\\n    latitude = 48.8566\\\\n    longitude = 2.3522\\\\n    elevation = 35\\\\n    mock_response = \\\\\\\"0:0\\\\\\\\n1:1\\\\\\\\n2:2\\\\\\\\n3:3\\\\\\\\n\\\\\\\"\\\\n\\\\n    with patch('requests.get') as mock_get:\\\\n        mock_get.return_value.status_code = 200\\\\n        mock_get.return_value.text = mock_response\\\\n\\\\n        horizon = get_mines_horizon(latitude, longitude, elevation)\\\\n        assert len(horizon) == 4\\\\n        assert horizon[0] == 0\\\\n        assert horizon[3] == 3\\\\n\\\\n        mock_get.return_value.status_code = 404\\\\n        with pytest.raises(requests.exceptions.HTTPError):\\\\n            get_mines_horizon(latitude, longitude)\\\\n\\\\n        invalid_response = \\\\\\\"invalid:text\\\\\\\"\\\\n        mock_get.return_value.text = invalid_response\\\\n        with pytest.raises(ValueError):\\\\n            get_mines_horizon(latitude, longitude)\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue where the `get_mines_horizon` function is expected to retrieve horizon data from MINES ParisTech API.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass before applying the patches and fail due to the missing function.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    from pvlib.iotools.mines import get_mines_horizon\\\\n    import requests\\\\n\\\\n    latitude = 0\\\\n    longitude = 0\\\\n    elevation = None\\\\n    mock_response = \\\\\\\"0:0\\\\\\\\n1:1\\\\\\\\n2:2\\\\\\\\n3:3\\\\\\\\n\\\\\\\"\\\\n\\\\n    with patch('requests.get') as mock_get:\\\\n        mock_get.return_value.status_code = 200\\\\n        mock_get.return_value.text = mock_response\\\\n\\\\n        horizon = get_mines_horizon(latitude, longitude, elevation)\\\\n        assert len(horizon) == 4\\\\n        assert horizon[0] == 0\\\\n        assert horizon[3] == 3\\\",\\n      \\\"description\\\": \\\"This test covers the edge cases where latitude and longitude are zero, and elevation is None.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass for these edge cases.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    from pvlib.iotools.mines import get_mines_horizon\\\\n    import requests\\\\n    import pandas as pd\\\\n\\\\n    latitude = 48.8566\\\\n    longitude = 2.3522\\\\n    elevation = 35\\\\n    mock_response = \\\\\\\"0:0\\\\\\\\n1:1\\\\\\\\n2:2\\\\\\\\n3:3\\\\\\\\n\\\\\\\"\\\\n\\\\n    with patch('requests.get') as mock_get:\\\\n        mock_get.return_value.status_code = 200\\\\n        mock_get.return_value.text = mock_response\\\\n\\\\n        horizon = get_mines_horizon(latitude, longitude, elevation)\\\\n        assert len(horizon) == 4\\\\n        assert horizon[0] == 0\\\\n        assert horizon[3] == 3\\\\n\\\\n        # Validate that the DataFrame version of the function works\\\\n        from io import StringIO\\\\n        data = StringIO(mock_response)\\\\n        df = pd.read_csv(data, sep=':', names=['azimuth', 'height'], header=None)\\\\n        horizon_df = get_mines_horizon(latitude, longitude, elevation)\\\\n        pd.testing.assert_frame_equal(horizon_df, df)\\\",\\n      \\\"description\\\": \\\"This test validates that the patches work correctly by checking the functionality of the `get_mines_horizon` function using both the dictionary and DataFrame return types.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass, confirming that the patches successfully implement the desired functionality.\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases were generated to reproduce the original issue, cover edge cases, and validate the patches for the `get_mines_horizon` function.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 368.887884, \"total_tokens\": 27903, \"total_steps\": 9, \"created_at\": 1753370959, \"finished_at\": 1753371327}}"
}