{
  "id": "pvlib__pvlib-python-1469",
  "question": "ModelChain should accept albedo in weather dataframe\n**Is your feature request related to a problem? Please describe.**\r\nAlbedo is treated as a scalar constant in pvlib, but it is of course a function of the weather and changes throughout the year.  Albedo is currently set in the PVSystem or Array and cannot be altered using the ModelChain.  Albedo is provided as a timeseries from many weather data services as well as through NREL's NSRBD and it would be useful to provide this data to the ModelChain.\r\n\r\nAdditionally, treating albedo as property of the Array seems to conflict with the [PVSystem Design Philosophy](https://pvlib-python.readthedocs.io/en/stable/pvsystem.html#design-philosophy), which highlights the separation of the PV system and the exogenous variables, such as the weather.\r\n\r\n**Describe the solution you'd like**\r\nModelChain.run_model() should accept albedo in the weather dataframe, like temperature and ghi.\r\n\r\n**Describe alternatives you've considered**\r\nAn alternative we have implemented is calling ModelChain.run_model() on each row of a dataframe and manually updating the albedo of the array in each tilmestep.  This probably has some side effects that we are unaware of.\r\n\n",
  "answer": "{\"task_id\": \"435311eb-0cee-4c1b-ad6b-aec8333184c1\", \"workflow_run_id\": \"d9cfcd0f-2b3a-4cd1-9211-d3f6400a3af2\", \"data\": {\"id\": \"d9cfcd0f-2b3a-4cd1-9211-d3f6400a3af2\", \"workflow_id\": \"4faf996e-a613-49ba-90e4-a0af9c740cdf\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\\\\nindex 234abc4..125fbc9 100644\\\\n--- a/pvlib/modelchain.py\\\\n+++ b/pvlib/modelchain.py\\\\n@@ -455,7 +455,11 @@ class ModelChain:\\\\n             temp_air = weather['temp_air']\\\\n             wind_speed = weather['wind_speed']\\\\n \\\\n-        self.solar_position = self.location.get_solarposition(times)\\\\n+        # Get albedo from weather if present\\\\n+        albedo = weather['albedo'] if 'albedo' in weather else None\\\\n+\\\\n+        self.albedo = albedo\\\\n+        self.solar_position = self.location.get_solarposition(times)\\\\n         self.extraradiation = self.solar_position['apparent_elevation'].apply(\\\\n             self.solarposition.get_extra_radiation,\\\\n             method=self.solarposition_method,\\\\n@@ -468,7 +472,7 @@ class ModelChain:\\\\n             self.aoi = self.pvsystem.get_aoi(self)\\\\n \\\\n         # Use dynamic albedo if available\\\\n-        total_irrad = self.pvsystem.get_irradiance(\\\\n+        total_irrad = self.pvsystem.get_irradiance(\\\\n             solar_position=self.solar_position,\\\\n             dhi=weather['dhi'],\\\\n             dni=weather['dni'],\\\\n@@ -477,6 +481,7 @@ class ModelChain:\\\\n             reflective_class=self.solarposition_method,\\\\n             dni_extra=self.extraradiation,\\\\n             temp_air=temp_air,\\\\n+            albedo=albedo,\\\\n             wind_speed=wind_speed\\\\n         )\\\\n \\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly integrates albedo from the weather dataframe into the ModelChain, ensuring it is passed to the PVSystem. It maintains good code quality and adheres to best practices by checking for the presence of albedo and defaulting to None if absent. The risk of introducing new bugs is minimal as it only adds a conditional check and passes the albedo parameter.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\\\\nindex 45efc75..f6b3e24 100644\\\\n--- a/pvlib/pvsystem.py\\\\n+++ b/pvlib/pvsystem.py\\\\n@@ -305,11 +305,14 @@ class PVSystem:\\\\n         air_mass: Series\\\\n         dni_extra: float\\\\n         temp_air: Series\\\\n+        albedo: float or Series, default None\\\\n+            Albedo is the fraction of solar radiation reflected from the ground.\\\\n+            If not provided, use the array's albedo.\\\\n \\\\n         Returns\\\\n         -------\\\\n         total_irradiance: DataFrame\\\\n         '''\\\\n+        albedo = albedo if albedo is not None else self.albedo\\\\n \\\\n         poa_sky_diffuse = self.get_irradiance.sky_diffuse(\\\\n             solar_position=solar_position,\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the PVSystem class to accept albedo as a parameter, defaulting to the system's albedo if not provided. It ensures flexibility in handling albedo but relies on the assumption that the caller will manage the albedo input correctly. The risk is slightly higher due to potential misuse if not integrated with the ModelChain properly.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\\\\nindex 8945abc..9cde564 100644\\\\n--- a/pvlib/irradiance.py\\\\n+++ b/pvlib/irradiance.py\\\\n@@ -155,7 +155,8 @@ def get_total_irradiance(surface_tilt, surface_azimuth, \\\\n                          solar_zenith=None, solar_azimuth=None,\\\\n                          dni=None, ghi=None, dhi=None,\\\\n                          dni_extra=None, aoi=None,\\\\n-                         air_mass=None, albedo=0.2):\\\\n+                         air_mass=None, albedo=None):\\\\n+\\\\n     albedo = albedo if albedo is not None else 0.2\\\\n     # calculate components\\\\n     if dni is not None or ghi is not None:\\\\n@@ -161,6 +162,8 @@ def get_total_irradiance(surface_tilt, surface_azimuth, \\\\n        \\\\n     # albedo should support both scalar and Series\\\\n     total_irrad = {}  # storing the results\\\\n     albedo = np.asarray(albedo)\\\\n+    # Check if albedo is a Series\\\\n+    if isinstance(albedo, pd.Series):\\\\n     if albedo.ndim == 0:  # scalar or datetime\\\\n         total_irrad['poa_albedo'] = 0 if np.all(albedo == 0) else albedo * (\\\\n             ghi * (1 - np.exp((-ghi/ghi[0])))\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the get_total_irradiance function to handle albedo as a Series, allowing for dynamic albedo values. While it provides flexibility, it introduces complexity and potential for errors if not handled correctly throughout the system. The patch assumes that albedo will be managed correctly by the caller, increasing the risk of misuse.\\\",\\n      \\\"recommendation\\\": \\\"Use with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most comprehensive and correctly integrates albedo into the ModelChain with minimal risk. Patch 2 provides flexibility in the PVSystem but requires careful integration with the ModelChain. Patch 3 allows for dynamic albedo handling in irradiance calculations but introduces complexity and potential for misuse.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Initialize ModelChain with weather data and albedo included\\\\n    model_chain = ModelChain(location, system)\\\\n    weather_data = {\\\\n        'times': pd.date_range('2022-01-01', periods=24, freq='H'),\\\\n        'temp_air': np.random.uniform(10, 40, 24),\\\\n        'wind_speed': np.random.uniform(0, 10, 24),\\\\n        'dhi': np.random.uniform(100, 800, 24),\\\\n        'dni': np.random.uniform(500, 1000, 24),\\\\n        'albedo': np.random.uniform(0.1, 0.5, 24)\\\\n    }\\\\n    weather_df = pd.DataFrame(weather_data)\\\\n    model_chain.run_model(weather=weather_df)\\\\n    # Check if albedo is correctly passed through the ModelChain to PVSystem\\\\n    assert 'albedo' in model_chain.pvsystem\\\\n    assert np.all(model_chain.pvsystem['albedo'] == weather_data['albedo'])\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by checking if ModelChain accepts albedo in the weather dataframe and passes it to PVSystem.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_albedo_missing\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_albedo_missing():\\\\n    # Initialize ModelChain without albedo in the weather dataframe\\\\n    model_chain = ModelChain(location, system)\\\\n    weather_data = {\\\\n        'times': pd.date_range('2022-01-01', periods=24, freq='H'),\\\\n        'temp_air': np.random.uniform(10, 40, 24),\\\\n        'wind_speed': np.random.uniform(0, 10, 24),\\\\n        'dhi': np.random.uniform(100, 800, 24),\\\\n        'dni': np.random.uniform(500, 1000, 24)\\\\n    }\\\\n    weather_df = pd.DataFrame(weather_data)\\\\n    model_chain.run_model(weather=weather_df)\\\\n    # Check if albedo defaults to None in PVSystem when not provided in weather dataframe\\\\n    assert 'albedo' in model_chain.pvsystem\\\\n    assert model_chain.pvsystem['albedo'] is None\\\",\\n      \\\"description\\\": \\\"This test covers the edge case where albedo is missing in the weather dataframe, ensuring it defaults to None in the PVSystem.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    # Initialize ModelChain with weather data and albedo included\\\\n    model_chain = ModelChain(location, system)\\\\n    weather_data = {\\\\n        'times': pd.date_range('2022-01-01', periods=24, freq='H'),\\\\n        'temp_air': np.random.uniform(10, 40, 24),\\\\n        'wind_speed': np.random.uniform(0, 10, 24),\\\\n        'dhi': np.random.uniform(100, 800, 24),\\\\n        'dni': np.random.uniform(500, 1000, 24),\\\\n        'albedo': np.random.uniform(0.1, 0.5, 24)\\\\n    }\\\\n    weather_df = pd.DataFrame(weather_data)\\\\n    model_chain.run_model(weather=weather_df)\\\\n    # Check if albedo is correctly passed through the ModelChain to PVSystem\\\\n    assert 'albedo' in model_chain.pvsystem\\\\n    assert np.all(model_chain.pvsystem['albedo'] == weather_data['albedo'])\\\",\\n      \\\"description\\\": \\\"This test validates that the patch successfully integrates albedo from the weather dataframe into the ModelChain system.\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases have been generated to reproduce the original issue, cover edge cases, and validate the patch ensuring proper integration of albedo in the ModelChain.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 357.922625, \"total_tokens\": 22753, \"total_steps\": 9, \"created_at\": 1753377714, \"finished_at\": 1753378072}}"
}