[{"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1614", "base_commit": "aa5a0d92e640ee5f3fa9a8ba3ba058a7b594ca44", "patch": "diff --git a/astroid/brain/brain_builtin_inference.py b/astroid/brain/brain_builtin_inference.py\n--- a/astroid/brain/brain_builtin_inference.py\n+++ b/astroid/brain/brain_builtin_inference.py\n@@ -946,7 +946,7 @@ def _infer_str_format_call(\n \n     try:\n         formatted_string = format_template.format(*pos_values, **keyword_values)\n-    except IndexError:\n+    except (IndexError, KeyError):\n         # If there is an IndexError there are too few arguments to interpolate\n         return iter([util.Uninferable])\n \n", "test_patch": "diff --git a/tests/unittest_brain_builtin.py b/tests/unittest_brain_builtin.py\n--- a/tests/unittest_brain_builtin.py\n+++ b/tests/unittest_brain_builtin.py\n@@ -93,6 +93,9 @@ def test_string_format(self, format_string: str) -> None:\n             \"My name is {}, I'm {}\".format(Unknown, 12)\n             \"\"\",\n             \"\"\"\"I am {}\".format()\"\"\",\n+            \"\"\"\n+            \"My name is {fname}, I'm {age}\".format(fsname = \"Daniel\", age = 12)\n+            \"\"\",\n         ],\n     )\n     def test_string_format_uninferable(self, format_string: str) -> None:\n", "problem_statement": "Crash when inferring `str.format` call involving unpacking kwargs\nWhen parsing the following file:\r\n\r\n<!--\r\n If sharing the code is not an option, please state so,\r\n but providing only the stacktrace would still be helpful.\r\n -->\r\n\r\n```python\r\nclass A:\r\n    def render(self, audit_log_entry: AuditLogEntry):\r\n        return \"joined team {team_slug}\".format(**audit_log_entry.data)\r\n\r\n\r\n\r\n```\r\n\r\npylint crashed with a ``AstroidError`` and with the following stacktrace:\r\n```\r\nTraceback (most recent call last):\r\n  File \"/Users/.../astroid/astroid/inference_tip.py\", line 38, in _inference_tip_cached\r\n    result = _cache[func, node]\r\nKeyError: (<function _infer_str_format_call at 0x1064a96c0>, <Call l.3 at 0x106c452d0>)\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/Users/.../pylint/pylint/lint/pylinter.py\", line 731, in _check_file\r\n    check_astroid_module(ast_node)\r\n  File \"/Users/.../pylint/pylint/lint/pylinter.py\", line 950, in check_astroid_module\r\n    retval = self._check_astroid_module(\r\n  File \"/Users/.../pylint/pylint/lint/pylinter.py\", line 1000, in _check_astroid_module\r\n    walker.walk(node)\r\n  File \"/Users/.../pylint/pylint/utils/ast_walker.py\", line 93, in walk\r\n    self.walk(child)\r\n  File \"/Users/.../pylint/pylint/utils/ast_walker.py\", line 93, in walk\r\n    self.walk(child)\r\n  File \"/Users/.../pylint/pylint/utils/ast_walker.py\", line 90, in walk\r\n    callback(astroid)\r\n  File \"/Users/.../pylint/pylint/checkers/classes/special_methods_checker.py\", line 170, in visit_functiondef\r\n    inferred = _safe_infer_call_result(node, node)\r\n  File \"/Users/.../pylint/pylint/checkers/classes/special_methods_checker.py\", line 31, in _safe_infer_call_result\r\n    value = next(inferit)\r\n  File \"/Users/.../astroid/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 1752, in infer_call_result\r\n    yield from returnnode.value.infer(context)\r\n  File \"/Users/.../astroid/astroid/nodes/node_ng.py\", line 159, in infer\r\n    results = list(self._explicit_inference(self, context, **kwargs))\r\n  File \"/Users/.../astroid/astroid/inference_tip.py\", line 45, in _inference_tip_cached\r\n    result = _cache[func, node] = list(func(*args, **kwargs))\r\n  File \"/Users/.../astroid/astroid/brain/brain_builtin_inference.py\", line 948, in _infer_str_format_call\r\n    formatted_string = format_template.format(*pos_values, **keyword_values)\r\nKeyError: 'team_slug'\r\n\r\nThe above exception was the direct cause of the following exception:\r\n\r\nTraceback (most recent call last):\r\n  File \"/Users/.../pylint/pylint/lint/pylinter.py\", line 688, in _check_files\r\n    self._check_file(get_ast, check_astroid_module, file)\r\n  File \"/Users/.../pylint/pylint/lint/pylinter.py\", line 733, in _check_file\r\n    raise astroid.AstroidError from e\r\nastroid.exceptions.AstroidError\r\n```\r\n***\r\ncc @DanielNoord in #1602 \r\nfound by pylint primer \ud83d\ude80 \n", "hints_text": "Thanks! Going to add `KeyError` to the caught exceptions.", "created_at": "2022-06-11T10:33:20Z", "version": "2.12", "FAIL_TO_PASS": ["tests/unittest_brain_builtin.py::TestStringNodes::test_string_format_uninferable[\\n"], "PASS_TO_PASS": ["tests/unittest_brain_builtin.py::BuiltinsTest::test_infer_property", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[empty-indexes]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[numbered-indexes]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[named-indexes]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[numbered-indexes-from-positional]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[named-indexes-from-keyword]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format[mixed-indexes-from-mixed]", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format_uninferable[\"I", "tests/unittest_brain_builtin.py::TestStringNodes::test_string_format_with_specs"], "environment_setup_commit": "52f6d2d7722db383af035be929f18af5e9fe8cd5"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1478", "base_commit": "3f397ed44075b01f4ba535750356859195a51b2d", "patch": "diff --git a/pvlib/clearsky.py b/pvlib/clearsky.py\n--- a/pvlib/clearsky.py\n+++ b/pvlib/clearsky.py\n@@ -960,8 +960,8 @@ def bird(zenith, airmass_relative, aod380, aod500, precipitable_water,\n         Extraterrestrial radiation [W/m^2], defaults to 1364[W/m^2]\n     asymmetry : numeric\n         Asymmetry factor, defaults to 0.85\n-    albedo : numeric\n-        Albedo, defaults to 0.2\n+    albedo : numeric, default 0.2\n+        Ground surface albedo. [unitless]\n \n     Returns\n     -------\ndiff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\n--- a/pvlib/irradiance.py\n+++ b/pvlib/irradiance.py\n@@ -304,7 +304,7 @@ def beam_component(surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n def get_total_irradiance(surface_tilt, surface_azimuth,\n                          solar_zenith, solar_azimuth,\n                          dni, ghi, dhi, dni_extra=None, airmass=None,\n-                         albedo=.25, surface_type=None,\n+                         albedo=0.25, surface_type=None,\n                          model='isotropic',\n                          model_perez='allsitescomposite1990'):\n     r\"\"\"\n@@ -344,7 +344,7 @@ def get_total_irradiance(surface_tilt, surface_azimuth,\n     airmass : None or numeric, default None\n         Relative airmass (not adjusted for pressure). [unitless]\n     albedo : numeric, default 0.25\n-        Surface albedo. [unitless]\n+        Ground surface albedo. [unitless]\n     surface_type : None or str, default None\n         Surface type. See :py:func:`~pvlib.irradiance.get_ground_diffuse` for\n         the list of accepted values.\n@@ -1872,7 +1872,7 @@ def gti_dirint(poa_global, aoi, solar_zenith, solar_azimuth, times,\n         applied.\n \n     albedo : numeric, default 0.25\n-        Surface albedo\n+        Ground surface albedo. [unitless]\n \n     model : String, default 'perez'\n         Irradiance model.  See :py:func:`get_sky_diffuse` for allowed values.\ndiff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -268,7 +268,7 @@ class ModelChainResult:\n     _per_array_fields = {'total_irrad', 'aoi', 'aoi_modifier',\n                          'spectral_modifier', 'cell_temperature',\n                          'effective_irradiance', 'dc', 'diode_params',\n-                         'dc_ohmic_losses', 'weather'}\n+                         'dc_ohmic_losses', 'weather', 'albedo'}\n \n     # system-level information\n     solar_position: Optional[pd.DataFrame] = field(default=None)\n@@ -366,6 +366,10 @@ class ModelChainResult:\n     \"\"\"DatetimeIndex containing a copy of the index of the input weather data.\n     \"\"\"\n \n+    albedo: Optional[PerArray[pd.Series]] = None\n+    \"\"\"Series (or tuple of Series, one for each array) containing albedo.\n+    \"\"\"\n+\n     def _result_type(self, value):\n         \"\"\"Coerce `value` to the correct type according to\n         ``self._singleton_tuples``.\"\"\"\n@@ -1339,6 +1343,17 @@ def _prep_inputs_solar_pos(self, weather):\n             **kwargs)\n         return self\n \n+    def _prep_inputs_albedo(self, weather):\n+        \"\"\"\n+        Get albedo from weather\n+        \"\"\"\n+        try:\n+            self.results.albedo = _tuple_from_dfs(weather, 'albedo')\n+        except KeyError:\n+            self.results.albedo = tuple([\n+                a.albedo for a in self.system.arrays])\n+        return self\n+\n     def _prep_inputs_airmass(self):\n         \"\"\"\n         Assign airmass\n@@ -1471,11 +1486,17 @@ def prepare_inputs(self, weather):\n \n         Parameters\n         ----------\n-        weather : DataFrame, or tuple or list of DataFrame\n+        weather : DataFrame, or tuple or list of DataFrames\n             Required column names include ``'dni'``, ``'ghi'``, ``'dhi'``.\n-            Optional column names are ``'wind_speed'``, ``'temp_air'``; if not\n+            Optional column names are ``'wind_speed'``, ``'temp_air'``,\n+            ``'albedo'``.\n+\n+            If optional columns ``'wind_speed'``, ``'temp_air'`` are not\n             provided, air temperature of 20 C and wind speed\n-            of 0 m/s will be added to the DataFrame.\n+            of 0 m/s will be added to the ``weather`` DataFrame.\n+\n+            If optional column ``'albedo'`` is provided, albedo values in the\n+            ModelChain's PVSystem.arrays are ignored.\n \n             If `weather` is a tuple or list, it must be of the same length and\n             order as the Arrays of the ModelChain's PVSystem.\n@@ -1494,7 +1515,7 @@ def prepare_inputs(self, weather):\n         Notes\n         -----\n         Assigns attributes to ``results``: ``times``, ``weather``,\n-        ``solar_position``, ``airmass``, ``total_irrad``, ``aoi``\n+        ``solar_position``, ``airmass``, ``total_irrad``, ``aoi``, ``albedo``.\n \n         See also\n         --------\n@@ -1507,6 +1528,7 @@ def prepare_inputs(self, weather):\n \n         self._prep_inputs_solar_pos(weather)\n         self._prep_inputs_airmass()\n+        self._prep_inputs_albedo(weather)\n \n         # PVSystem.get_irradiance and SingleAxisTracker.get_irradiance\n         # and PVSystem.get_aoi and SingleAxisTracker.get_aoi\n@@ -1531,6 +1553,7 @@ def prepare_inputs(self, weather):\n             _tuple_from_dfs(self.results.weather, 'dni'),\n             _tuple_from_dfs(self.results.weather, 'ghi'),\n             _tuple_from_dfs(self.results.weather, 'dhi'),\n+            albedo=self.results.albedo,\n             airmass=self.results.airmass['airmass_relative'],\n             model=self.transposition_model\n         )\n@@ -1724,16 +1747,32 @@ def run_model(self, weather):\n         Parameters\n         ----------\n         weather : DataFrame, or tuple or list of DataFrame\n-            Irradiance column names must include ``'dni'``, ``'ghi'``, and\n-            ``'dhi'``. If optional columns ``'temp_air'`` and ``'wind_speed'``\n+            Column names must include:\n+\n+            - ``'dni'``\n+            - ``'ghi'``\n+            - ``'dhi'``\n+\n+            Optional columns are:\n+\n+            - ``'temp_air'``\n+            - ``'cell_temperature'``\n+            - ``'module_temperature'``\n+            - ``'wind_speed'``\n+            - ``'albedo'``\n+\n+            If optional columns ``'temp_air'`` and ``'wind_speed'``\n             are not provided, air temperature of 20 C and wind speed of 0 m/s\n             are added to the DataFrame. If optional column\n             ``'cell_temperature'`` is provided, these values are used instead\n-            of `temperature_model`. If optional column `module_temperature`\n-            is provided, `temperature_model` must be ``'sapm'``.\n+            of `temperature_model`. If optional column ``'module_temperature'``\n+            is provided, ``temperature_model`` must be ``'sapm'``.\n \n-            If list or tuple, must be of the same length and order as the\n-            Arrays of the ModelChain's PVSystem.\n+            If optional column ``'albedo'`` is provided, ``'albedo'`` may not\n+            be present on the ModelChain's PVSystem.Arrays.\n+\n+            If weather is a list or tuple, it must be of the same length and\n+            order as the Arrays of the ModelChain's PVSystem.\n \n         Returns\n         -------\ndiff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\n--- a/pvlib/pvsystem.py\n+++ b/pvlib/pvsystem.py\n@@ -134,7 +134,7 @@ class PVSystem:\n         a single array is created from the other parameters (e.g.\n         `surface_tilt`, `surface_azimuth`). Must contain at least one Array,\n         if length of arrays is 0 a ValueError is raised. If `arrays` is\n-        specified the following parameters are ignored:\n+        specified the following PVSystem parameters are ignored:\n \n         - `surface_tilt`\n         - `surface_azimuth`\n@@ -157,13 +157,14 @@ class PVSystem:\n         North=0, East=90, South=180, West=270.\n \n     albedo : None or float, default None\n-        The ground albedo. If ``None``, will attempt to use\n-        ``surface_type`` and ``irradiance.SURFACE_ALBEDOS``\n-        to lookup albedo.\n+        Ground surface albedo. If ``None``, then ``surface_type`` is used\n+        to look up a value in ``irradiance.SURFACE_ALBEDOS``.\n+        If ``surface_type`` is also None then a ground surface albedo\n+        of 0.25 is used.\n \n     surface_type : None or string, default None\n-        The ground surface type. See ``irradiance.SURFACE_ALBEDOS``\n-        for valid values.\n+        The ground surface type. See ``irradiance.SURFACE_ALBEDOS`` for\n+        valid values.\n \n     module : None or string, default None\n         The model name of the modules.\n@@ -333,30 +334,33 @@ def get_aoi(self, solar_zenith, solar_azimuth):\n \n     @_unwrap_single_value\n     def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n-                       **kwargs):\n+                       dni_extra=None, airmass=None, albedo=None,\n+                       model='haydavies', **kwargs):\n         \"\"\"\n         Uses the :py:func:`irradiance.get_total_irradiance` function to\n-        calculate the plane of array irradiance components on a tilted\n-        surface defined by ``self.surface_tilt``,\n-        ``self.surface_azimuth``, and ``self.albedo``.\n+        calculate the plane of array irradiance components on the tilted\n+        surfaces defined by each array's ``surface_tilt`` and\n+        ``surface_azimuth``.\n \n         Parameters\n         ----------\n-        solar_zenith : float or Series.\n+        solar_zenith : float or Series\n             Solar zenith angle.\n-        solar_azimuth : float or Series.\n+        solar_azimuth : float or Series\n             Solar azimuth angle.\n         dni : float or Series or tuple of float or Series\n-            Direct Normal Irradiance\n+            Direct Normal Irradiance. [W/m2]\n         ghi : float or Series or tuple of float or Series\n-            Global horizontal irradiance\n+            Global horizontal irradiance. [W/m2]\n         dhi : float or Series or tuple of float or Series\n-            Diffuse horizontal irradiance\n-        dni_extra : None, float or Series, default None\n-            Extraterrestrial direct normal irradiance\n+            Diffuse horizontal irradiance. [W/m2]\n+        dni_extra : None, float, Series or tuple of float or Series,\\\n+            default None\n+            Extraterrestrial direct normal irradiance. [W/m2]\n         airmass : None, float or Series, default None\n-            Airmass\n+            Airmass. [unitless]\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n         model : String, default 'haydavies'\n             Irradiance model.\n \n@@ -376,17 +380,24 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         poa_irradiance : DataFrame or tuple of DataFrame\n             Column names are: ``'poa_global', 'poa_direct', 'poa_diffuse',\n             'poa_sky_diffuse', 'poa_ground_diffuse'``.\n+\n+        See also\n+        --------\n+        pvlib.irradiance.get_total_irradiance\n         \"\"\"\n         dni = self._validate_per_array(dni, system_wide=True)\n         ghi = self._validate_per_array(ghi, system_wide=True)\n         dhi = self._validate_per_array(dhi, system_wide=True)\n+\n+        albedo = self._validate_per_array(albedo, system_wide=True)\n+\n         return tuple(\n             array.get_irradiance(solar_zenith, solar_azimuth,\n                                  dni, ghi, dhi,\n-                                 dni_extra, airmass, model,\n-                                 **kwargs)\n-            for array, dni, ghi, dhi in zip(\n-                self.arrays, dni, ghi, dhi\n+                                 dni_extra=dni_extra, airmass=airmass,\n+                                 albedo=albedo, model=model, **kwargs)\n+            for array, dni, ghi, dhi, albedo in zip(\n+                self.arrays, dni, ghi, dhi, albedo\n             )\n         )\n \n@@ -1258,14 +1269,14 @@ class Array:\n         If not provided, a FixedMount with zero tilt is used.\n \n     albedo : None or float, default None\n-        The ground albedo. If ``None``, will attempt to use\n-        ``surface_type`` to look up an albedo value in\n-        ``irradiance.SURFACE_ALBEDOS``. If a surface albedo\n-        cannot be found then 0.25 is used.\n+        Ground surface albedo. If ``None``, then ``surface_type`` is used\n+        to look up a value in ``irradiance.SURFACE_ALBEDOS``.\n+        If ``surface_type`` is also None then a ground surface albedo\n+        of 0.25 is used.\n \n     surface_type : None or string, default None\n-        The ground surface type. See ``irradiance.SURFACE_ALBEDOS``\n-        for valid values.\n+        The ground surface type. See ``irradiance.SURFACE_ALBEDOS`` for valid\n+        values.\n \n     module : None or string, default None\n         The model name of the modules.\n@@ -1425,15 +1436,14 @@ def get_aoi(self, solar_zenith, solar_azimuth):\n                               solar_zenith, solar_azimuth)\n \n     def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n-                       **kwargs):\n+                       dni_extra=None, airmass=None, albedo=None,\n+                       model='haydavies', **kwargs):\n         \"\"\"\n         Get plane of array irradiance components.\n \n         Uses the :py:func:`pvlib.irradiance.get_total_irradiance` function to\n         calculate the plane of array irradiance components for a surface\n-        defined by ``self.surface_tilt`` and ``self.surface_azimuth`` with\n-        albedo ``self.albedo``.\n+        defined by ``self.surface_tilt`` and ``self.surface_azimuth``.\n \n         Parameters\n         ----------\n@@ -1442,15 +1452,17 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         solar_azimuth : float or Series.\n             Solar azimuth angle.\n         dni : float or Series\n-            Direct Normal Irradiance\n-        ghi : float or Series\n+            Direct normal irradiance. [W/m2]\n+        ghi : float or Series. [W/m2]\n             Global horizontal irradiance\n         dhi : float or Series\n-            Diffuse horizontal irradiance\n+            Diffuse horizontal irradiance. [W/m2]\n         dni_extra : None, float or Series, default None\n-            Extraterrestrial direct normal irradiance\n+            Extraterrestrial direct normal irradiance. [W/m2]\n         airmass : None, float or Series, default None\n-            Airmass\n+            Airmass. [unitless]\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n         model : String, default 'haydavies'\n             Irradiance model.\n \n@@ -1463,7 +1475,14 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         poa_irradiance : DataFrame\n             Column names are: ``'poa_global', 'poa_direct', 'poa_diffuse',\n             'poa_sky_diffuse', 'poa_ground_diffuse'``.\n+\n+        See also\n+        --------\n+        :py:func:`pvlib.irradiance.get_total_irradiance`\n         \"\"\"\n+        if albedo is None:\n+            albedo = self.albedo\n+\n         # not needed for all models, but this is easier\n         if dni_extra is None:\n             dni_extra = irradiance.get_extra_radiation(solar_zenith.index)\n@@ -1478,8 +1497,8 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n                                                dni, ghi, dhi,\n                                                dni_extra=dni_extra,\n                                                airmass=airmass,\n+                                               albedo=albedo,\n                                                model=model,\n-                                               albedo=self.albedo,\n                                                **kwargs)\n \n     def get_iam(self, aoi, iam_model='physical'):\n@@ -3293,7 +3312,7 @@ def dc_ohms_from_percent(vmp_ref, imp_ref, dc_ohmic_percent,\n \n     See Also\n     --------\n-    :py:func:`~pvlib.pvsystem.dc_ohmic_losses`\n+    pvlib.pvsystem.dc_ohmic_losses\n \n     References\n     ----------\n@@ -3328,7 +3347,7 @@ def dc_ohmic_losses(resistance, current):\n \n     See Also\n     --------\n-    :py:func:`~pvlib.pvsystem.dc_ohms_from_percent`\n+    pvlib.pvsystem.dc_ohms_from_percent\n \n     References\n     ----------\ndiff --git a/pvlib/tracking.py b/pvlib/tracking.py\n--- a/pvlib/tracking.py\n+++ b/pvlib/tracking.py\n@@ -188,7 +188,8 @@ def get_aoi(self, surface_tilt, surface_azimuth, solar_zenith,\n     @_unwrap_single_value\n     def get_irradiance(self, surface_tilt, surface_azimuth,\n                        solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n+                       albedo=None, dni_extra=None, airmass=None,\n+                       model='haydavies',\n                        **kwargs):\n         \"\"\"\n         Uses the :func:`irradiance.get_total_irradiance` function to\n@@ -215,6 +216,8 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n             Global horizontal irradiance\n         dhi : float or Series\n             Diffuse horizontal irradiance\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n         dni_extra : float or Series, default None\n             Extraterrestrial direct normal irradiance\n         airmass : float or Series, default None\n@@ -245,6 +248,13 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n         ghi = self._validate_per_array(ghi, system_wide=True)\n         dhi = self._validate_per_array(dhi, system_wide=True)\n \n+        if albedo is None:\n+            # assign default albedo here because SingleAxisTracker\n+            # initializes albedo to None\n+            albedo = 0.25\n+\n+        albedo = self._validate_per_array(albedo, system_wide=True)\n+\n         return tuple(\n             irradiance.get_total_irradiance(\n                 surface_tilt,\n@@ -255,10 +265,10 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n                 dni_extra=dni_extra,\n                 airmass=airmass,\n                 model=model,\n-                albedo=self.arrays[0].albedo,\n+                albedo=albedo,\n                 **kwargs)\n-            for array, dni, ghi, dhi in zip(\n-                self.arrays, dni, ghi, dhi\n+            for array, dni, ghi, dhi, albedo in zip(\n+                self.arrays, dni, ghi, dhi, albedo\n             )\n         )\n \n", "test_patch": "diff --git a/pvlib/tests/test_clearsky.py b/pvlib/tests/test_clearsky.py\n--- a/pvlib/tests/test_clearsky.py\n+++ b/pvlib/tests/test_clearsky.py\n@@ -756,6 +756,30 @@ def test_bird():\n     assert np.allclose(\n         testdata['Dif Hz'].where(dusk, 0.), diffuse_horz[1:48], rtol=1e-3\n     )\n+    # repeat test with albedo as a Series\n+    alb_series = pd.Series(0.2, index=times)\n+    irrads = clearsky.bird(\n+        zenith, airmass, aod_380nm, aod_500nm, h2o_cm, o3_cm, press_mB * 100.,\n+        etr, b_a, alb_series\n+    )\n+    Eb, Ebh, Gh, Dh = (irrads[_] for _ in field_names)\n+    direct_beam = pd.Series(np.where(dawn, Eb, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Direct Beam'].where(dusk, 0.), direct_beam[1:48], rtol=1e-3\n+    )\n+    direct_horz = pd.Series(np.where(dawn, Ebh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Direct Hz'].where(dusk, 0.), direct_horz[1:48], rtol=1e-3\n+    )\n+    global_horz = pd.Series(np.where(dawn, Gh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Global Hz'].where(dusk, 0.), global_horz[1:48], rtol=1e-3\n+    )\n+    diffuse_horz = pd.Series(np.where(dawn, Dh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Dif Hz'].where(dusk, 0.), diffuse_horz[1:48], rtol=1e-3\n+    )\n+\n     # test keyword parameters\n     irrads2 = clearsky.bird(\n         zenith, airmass, aod_380nm, aod_500nm, h2o_cm, dni_extra=etr\ndiff --git a/pvlib/tests/test_irradiance.py b/pvlib/tests/test_irradiance.py\n--- a/pvlib/tests/test_irradiance.py\n+++ b/pvlib/tests/test_irradiance.py\n@@ -120,29 +120,38 @@ def test_get_extra_radiation_invalid():\n         irradiance.get_extra_radiation(300, method='invalid')\n \n \n-def test_grounddiffuse_simple_float():\n+def test_get_ground_diffuse_simple_float():\n     result = irradiance.get_ground_diffuse(40, 900)\n     assert_allclose(result, 26.32000014911496)\n \n \n-def test_grounddiffuse_simple_series(irrad_data):\n+def test_get_ground_diffuse_simple_series(irrad_data):\n     ground_irrad = irradiance.get_ground_diffuse(40, irrad_data['ghi'])\n     assert ground_irrad.name == 'diffuse_ground'\n \n \n-def test_grounddiffuse_albedo_0(irrad_data):\n+def test_get_ground_diffuse_albedo_0(irrad_data):\n     ground_irrad = irradiance.get_ground_diffuse(\n         40, irrad_data['ghi'], albedo=0)\n     assert 0 == ground_irrad.all()\n \n \n+def test_get_ground_diffuse_albedo_series(times):\n+    albedo = pd.Series(0.2, index=times)\n+    ground_irrad = irradiance.get_ground_diffuse(\n+        45, pd.Series(1000, index=times), albedo)\n+    expected = albedo * 0.5 * (1 - np.sqrt(2) / 2.) * 1000\n+    expected.name = 'diffuse_ground'\n+    assert_series_equal(ground_irrad, expected)\n+\n+\n def test_grounddiffuse_albedo_invalid_surface(irrad_data):\n     with pytest.raises(KeyError):\n         irradiance.get_ground_diffuse(\n             40, irrad_data['ghi'], surface_type='invalid')\n \n \n-def test_grounddiffuse_albedo_surface(irrad_data):\n+def test_get_ground_diffuse_albedo_surface(irrad_data):\n     result = irradiance.get_ground_diffuse(40, irrad_data['ghi'],\n                                            surface_type='sand')\n     assert_allclose(result, [0, 3.731058, 48.778813, 12.035025], atol=1e-4)\n@@ -387,6 +396,25 @@ def test_get_total_irradiance(irrad_data, ephem_data, dni_et,\n                                           'poa_ground_diffuse']\n \n \n+@pytest.mark.parametrize('model', ['isotropic', 'klucher',\n+                                   'haydavies', 'reindl', 'king', 'perez'])\n+def test_get_total_irradiance_albedo(\n+        irrad_data, ephem_data, dni_et, relative_airmass, model):\n+    albedo = pd.Series(0.2, index=ephem_data.index)\n+    total = irradiance.get_total_irradiance(\n+        32, 180,\n+        ephem_data['apparent_zenith'], ephem_data['azimuth'],\n+        dni=irrad_data['dni'], ghi=irrad_data['ghi'],\n+        dhi=irrad_data['dhi'],\n+        dni_extra=dni_et, airmass=relative_airmass,\n+        model=model,\n+        albedo=albedo)\n+\n+    assert total.columns.tolist() == ['poa_global', 'poa_direct',\n+                                      'poa_diffuse', 'poa_sky_diffuse',\n+                                      'poa_ground_diffuse']\n+\n+\n @pytest.mark.parametrize('model', ['isotropic', 'klucher',\n                                    'haydavies', 'reindl', 'king', 'perez'])\n def test_get_total_irradiance_scalars(model):\n@@ -698,6 +726,14 @@ def test_gti_dirint():\n \n     assert_frame_equal(output, expected)\n \n+    # test with albedo as a Series\n+    albedo = pd.Series(0.05, index=times)\n+    output = irradiance.gti_dirint(\n+        poa_global, aoi, zenith, azimuth, times, surface_tilt, surface_azimuth,\n+        albedo=albedo)\n+\n+    assert_frame_equal(output, expected)\n+\n     # test temp_dew input\n     temp_dew = np.array([70, 80, 20])\n     output = irradiance.gti_dirint(\ndiff --git a/pvlib/tests/test_modelchain.py b/pvlib/tests/test_modelchain.py\n--- a/pvlib/tests/test_modelchain.py\n+++ b/pvlib/tests/test_modelchain.py\n@@ -495,6 +495,26 @@ def test_prepare_inputs_multi_weather(\n     mc.prepare_inputs(input_type((weather, weather)))\n     num_arrays = sapm_dc_snl_ac_system_Array.num_arrays\n     assert len(mc.results.total_irrad) == num_arrays\n+    # check that albedo is transfered to mc.results from mc.system.arrays\n+    assert mc.results.albedo == (0.2, 0.2)\n+\n+\n+@pytest.mark.parametrize(\"input_type\", [tuple, list])\n+def test_prepare_inputs_albedo_in_weather(\n+        sapm_dc_snl_ac_system_Array, location, input_type):\n+    times = pd.date_range(start='20160101 1200-0700',\n+                          end='20160101 1800-0700', freq='6H')\n+    mc = ModelChain(sapm_dc_snl_ac_system_Array, location)\n+    weather = pd.DataFrame({'ghi': 1, 'dhi': 1, 'dni': 1, 'albedo': 0.5},\n+                           index=times)\n+    # weather as a single DataFrame\n+    mc.prepare_inputs(weather)\n+    num_arrays = sapm_dc_snl_ac_system_Array.num_arrays\n+    assert len(mc.results.albedo) == num_arrays\n+    # repeat with tuple of weather\n+    mc.prepare_inputs(input_type((weather, weather)))\n+    num_arrays = sapm_dc_snl_ac_system_Array.num_arrays\n+    assert len(mc.results.albedo) == num_arrays\n \n \n def test_prepare_inputs_no_irradiance(sapm_dc_snl_ac_system, location):\ndiff --git a/pvlib/tests/test_pvsystem.py b/pvlib/tests/test_pvsystem.py\n--- a/pvlib/tests/test_pvsystem.py\n+++ b/pvlib/tests/test_pvsystem.py\n@@ -18,6 +18,7 @@\n from pvlib.pvsystem import FixedMount\n from pvlib import temperature\n from pvlib._deprecation import pvlibDeprecationWarning\n+from pvlib.tools import cosd\n \n \n @pytest.mark.parametrize('iam_model,model_params', [\n@@ -1673,51 +1674,70 @@ def test_PVSystem_multiple_array_get_aoi():\n     assert aoi_one > 0\n \n \n-def test_PVSystem_get_irradiance():\n-    system = pvsystem.PVSystem(surface_tilt=32, surface_azimuth=135)\n+@pytest.fixture\n+def solar_pos():\n     times = pd.date_range(start='20160101 1200-0700',\n                           end='20160101 1800-0700', freq='6H')\n     location = Location(latitude=32, longitude=-111)\n-    solar_position = location.get_solarposition(times)\n+    return location.get_solarposition(times)\n+\n+\n+def test_PVSystem_get_irradiance(solar_pos):\n+    system = pvsystem.PVSystem(surface_tilt=32, surface_azimuth=135)\n     irrads = pd.DataFrame({'dni':[900,0], 'ghi':[600,0], 'dhi':[100,0]},\n-                          index=times)\n+                          index=solar_pos.index)\n \n-    irradiance = system.get_irradiance(solar_position['apparent_zenith'],\n-                                       solar_position['azimuth'],\n+    irradiance = system.get_irradiance(solar_pos['apparent_zenith'],\n+                                       solar_pos['azimuth'],\n                                        irrads['dni'],\n                                        irrads['ghi'],\n                                        irrads['dhi'])\n \n     expected = pd.DataFrame(data=np.array(\n-        [[ 883.65494055,  745.86141676,  137.79352379,  126.397131  ,\n-              11.39639279],\n-           [   0.        ,   -0.        ,    0.        ,    0.        ,    0.        ]]),\n+        [[883.65494055, 745.86141676, 137.79352379, 126.397131, 11.39639279],\n+         [0., -0., 0., 0., 0.]]),\n                             columns=['poa_global', 'poa_direct',\n                                      'poa_diffuse', 'poa_sky_diffuse',\n                                      'poa_ground_diffuse'],\n-                            index=times)\n+                            index=solar_pos.index)\n+    assert_frame_equal(irradiance, expected, check_less_precise=2)\n+\n \n+def test_PVSystem_get_irradiance_albedo(solar_pos):\n+    system = pvsystem.PVSystem(surface_tilt=32, surface_azimuth=135)\n+    irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0],\n+                           'albedo': [0.5, 0.5]},\n+                          index=solar_pos.index)\n+    # albedo as a Series\n+    irradiance = system.get_irradiance(solar_pos['apparent_zenith'],\n+                                       solar_pos['azimuth'],\n+                                       irrads['dni'],\n+                                       irrads['ghi'],\n+                                       irrads['dhi'],\n+                                       albedo=irrads['albedo'])\n+    expected = pd.DataFrame(data=np.array(\n+        [[895.05134334, 745.86141676, 149.18992658, 126.397131, 22.79279558],\n+         [0., -0., 0., 0., 0.]]),\n+        columns=['poa_global', 'poa_direct', 'poa_diffuse', 'poa_sky_diffuse',\n+                 'poa_ground_diffuse'],\n+        index=solar_pos.index)\n     assert_frame_equal(irradiance, expected, check_less_precise=2)\n \n \n-def test_PVSystem_get_irradiance_model(mocker):\n+def test_PVSystem_get_irradiance_model(mocker, solar_pos):\n     spy_perez = mocker.spy(irradiance, 'perez')\n     spy_haydavies = mocker.spy(irradiance, 'haydavies')\n     system = pvsystem.PVSystem(surface_tilt=32, surface_azimuth=135)\n-    times = pd.date_range(start='20160101 1200-0700',\n-                          end='20160101 1800-0700', freq='6H')\n-    location = Location(latitude=32, longitude=-111)\n-    solar_position = location.get_solarposition(times)\n     irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0]},\n-                          index=times)\n-    system.get_irradiance(solar_position['apparent_zenith'],\n-                          solar_position['azimuth'],\n+                          index=solar_pos.index)\n+    system.get_irradiance(solar_pos['apparent_zenith'],\n+                          solar_pos['azimuth'],\n                           irrads['dni'],\n                           irrads['ghi'],\n                           irrads['dhi'])\n     spy_haydavies.assert_called_once()\n-    system.get_irradiance(solar_position['apparent_zenith'],\n-                          solar_position['azimuth'],\n+    system.get_irradiance(solar_pos['apparent_zenith'],\n+                          solar_pos['azimuth'],\n                           irrads['dni'],\n                           irrads['ghi'],\n                           irrads['dhi'],\n@@ -1725,31 +1745,28 @@ def test_PVSystem_get_irradiance_model(mocker):\n     spy_perez.assert_called_once()\n \n \n-def test_PVSystem_multi_array_get_irradiance():\n+def test_PVSystem_multi_array_get_irradiance(solar_pos):\n     array_one = pvsystem.Array(pvsystem.FixedMount(surface_tilt=32,\n                                                    surface_azimuth=135))\n     array_two = pvsystem.Array(pvsystem.FixedMount(surface_tilt=5,\n                                                    surface_azimuth=150))\n     system = pvsystem.PVSystem(arrays=[array_one, array_two])\n-    location = Location(latitude=32, longitude=-111)\n-    times = pd.date_range(start='20160101 1200-0700',\n-                          end='20160101 1800-0700', freq='6H')\n-    solar_position = location.get_solarposition(times)\n+\n     irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0]},\n-                          index=times)\n+                          index=solar_pos.index)\n     array_one_expected = array_one.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         irrads['dni'], irrads['ghi'], irrads['dhi']\n     )\n     array_two_expected = array_two.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         irrads['dni'], irrads['ghi'], irrads['dhi']\n     )\n     array_one_irrad, array_two_irrad = system.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         irrads['dni'], irrads['ghi'], irrads['dhi']\n     )\n     assert_frame_equal(\n@@ -1760,7 +1777,7 @@ def test_PVSystem_multi_array_get_irradiance():\n     )\n \n \n-def test_PVSystem_multi_array_get_irradiance_multi_irrad():\n+def test_PVSystem_multi_array_get_irradiance_multi_irrad(solar_pos):\n     \"\"\"Test a system with two identical arrays but different irradiance.\n \n     Because only the irradiance is different we expect the same output\n@@ -1771,39 +1788,36 @@ def test_PVSystem_multi_array_get_irradiance_multi_irrad():\n     array_one = pvsystem.Array(pvsystem.FixedMount(0, 180))\n     array_two = pvsystem.Array(pvsystem.FixedMount(0, 180))\n     system = pvsystem.PVSystem(arrays=[array_one, array_two])\n-    location = Location(latitude=32, longitude=-111)\n-    times = pd.date_range(start='20160101 1200-0700',\n-                          end='20160101 1800-0700', freq='6H')\n-    solar_position = location.get_solarposition(times)\n+\n     irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0]},\n-                          index=times)\n+                          index=solar_pos.index)\n     irrads_two = pd.DataFrame(\n         {'dni': [0, 900], 'ghi': [0, 600], 'dhi': [0, 100]},\n-        index=times\n+        index=solar_pos.index\n     )\n     array_irrad = system.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         (irrads['dhi'], irrads['dhi']),\n         (irrads['ghi'], irrads['ghi']),\n         (irrads['dni'], irrads['dni'])\n     )\n     assert_frame_equal(array_irrad[0], array_irrad[1])\n     array_irrad = system.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         (irrads['dhi'], irrads_two['dhi']),\n         (irrads['ghi'], irrads_two['ghi']),\n         (irrads['dni'], irrads_two['dni'])\n     )\n     array_one_expected = array_one.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         irrads['dhi'], irrads['ghi'], irrads['dni']\n     )\n     array_two_expected = array_two.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         irrads_two['dhi'], irrads_two['ghi'], irrads_two['dni']\n     )\n     assert not array_irrad[0].equals(array_irrad[1])\n@@ -1812,15 +1826,15 @@ def test_PVSystem_multi_array_get_irradiance_multi_irrad():\n     with pytest.raises(ValueError,\n                        match=\"Length mismatch for per-array parameter\"):\n         system.get_irradiance(\n-            solar_position['apparent_zenith'],\n-            solar_position['azimuth'],\n+            solar_pos['apparent_zenith'],\n+            solar_pos['azimuth'],\n             (irrads['dhi'], irrads_two['dhi'], irrads['dhi']),\n             (irrads['ghi'], irrads_two['ghi']),\n             irrads['dni']\n         )\n     array_irrad = system.get_irradiance(\n-        solar_position['apparent_zenith'],\n-        solar_position['azimuth'],\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n         (irrads['dhi'], irrads_two['dhi']),\n         irrads['ghi'],\n         irrads['dni']\n@@ -1829,6 +1843,44 @@ def test_PVSystem_multi_array_get_irradiance_multi_irrad():\n     assert not array_irrad[0].equals(array_irrad[1])\n \n \n+def test_Array_get_irradiance(solar_pos):\n+    array = pvsystem.Array(pvsystem.FixedMount(surface_tilt=32,\n+                                               surface_azimuth=135))\n+    irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0]},\n+                          index=solar_pos.index)\n+    # defaults for kwargs\n+    modeled = array.get_irradiance(\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n+        irrads['dni'], irrads['ghi'], irrads['dhi']\n+    )\n+    expected = pd.DataFrame(\n+        data=np.array(\n+            [[883.65494055, 745.86141676, 137.79352379, 126.397131,\n+              11.39639279],\n+             [0., -0., 0., 0., 0.]]),\n+        columns=['poa_global', 'poa_direct', 'poa_diffuse', 'poa_sky_diffuse',\n+                 'poa_ground_diffuse'],\n+        index=solar_pos.index\n+    )\n+    assert_frame_equal(modeled, expected, check_less_precise=5)\n+    # with specified kwargs, use isotropic sky diffuse because it's easier\n+    modeled = array.get_irradiance(\n+        solar_pos['apparent_zenith'],\n+        solar_pos['azimuth'],\n+        irrads['dni'], irrads['ghi'], irrads['dhi'],\n+        albedo=0.5, model='isotropic'\n+    )\n+    sky_diffuse = irradiance.isotropic(array.mount.surface_tilt, irrads['dhi'])\n+    ground_diff = irradiance.get_ground_diffuse(\n+        array.mount.surface_tilt, irrads['ghi'], 0.5, surface_type=None)\n+    aoi = irradiance.aoi(array.mount.surface_tilt, array.mount.surface_azimuth,\n+                         solar_pos['apparent_zenith'], solar_pos['azimuth'])\n+    direct = irrads['dni'] * cosd(aoi)\n+    expected = sky_diffuse + ground_diff + direct\n+    assert_series_equal(expected, expected, check_less_precise=5)\n+\n+\n @fail_on_pvlib_version('0.10')\n @pytest.mark.parametrize('attr', ['module_parameters', 'module', 'module_type',\n                                   'temperature_model_parameters', 'albedo',\ndiff --git a/pvlib/tests/test_tracking.py b/pvlib/tests/test_tracking.py\n--- a/pvlib/tests/test_tracking.py\n+++ b/pvlib/tests/test_tracking.py\n@@ -393,6 +393,25 @@ def test_get_irradiance():\n \n     assert_frame_equal(irradiance, expected, check_less_precise=2)\n \n+    # test with albedo as a Series\n+    irrads['albedo'] = [0.5, 0.5]\n+    with np.errstate(invalid='ignore'):\n+        irradiance = system.get_irradiance(tracker_data['surface_tilt'],\n+                                           tracker_data['surface_azimuth'],\n+                                           solar_zenith,\n+                                           solar_azimuth,\n+                                           irrads['dni'],\n+                                           irrads['ghi'],\n+                                           irrads['dhi'],\n+                                           albedo=irrads['albedo'])\n+\n+    expected = pd.Series(data=[21.05514984, nan], index=times,\n+                         name='poa_ground_diffuse')\n+\n+    assert_series_equal(irradiance['poa_ground_diffuse'], expected,\n+                        check_less_precise=2)\n+\n+\n \n def test_SingleAxisTracker___repr__():\n     with pytest.warns(pvlibDeprecationWarning):\n", "problem_statement": "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", "hints_text": "I think I agree that it would make more sense to pass around ground albedo with the weather data instead of treating it as a characteristic of the array.  \r\n\r\n> Albedo is treated as a scalar constant in pvlib\r\n\r\n> An 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. \r\n\r\nIt is true that the docs for `Array` and `PVSystem` say that `albedo` is a float, but I think it also works to set albedo to a time series that matches the weather you pass to `ModelChain.run_model()`.  At least that saves you from looping w/ scalar albedo values.  \nI agree that albedo can change with e.g., precipitation or season, but it is also a property of the bare ground surface, and it is that perspective that put albedo with the system parameters. One use case for not having albedo in the weather data would be to evaluate the effect of different ground cover on array output.\r\n\r\nI am only away of the NSRDB offering albedo with weather data; are there other sources?\r\n\r\nI'm +1 on allowing `albedo` to be a Series. I'm neutral on bundling albedo with weather data, but I don't see a better option.  We only have two data structures that supply `ModelChain`: the `weather` DataFrame, and the `PVSystem` instance. I don't think it is practical to create a third just for `albedo`, and it isn't any more work to add or modify `albedo` to `weather` than it is to extract `albedo` from downloaded weather data and add it to `PVSystem`.\r\n\r\n\nTo clarify my above message, I think it *already* works to set `PVSystem.albedo` or `Array.albedo` to a Series, despite the docs saying it must be float.  \r\n\r\n> are there other sources?\r\n\r\nA non-exhaustive list of examples: [SolarAnywhere](https://www.solaranywhere.com/support/data-fields/albedo/), [SolarGIS](https://solargis.com/docs/getting-started/data-parameters), [MERRA2](https://disc.gsfc.nasa.gov/datasets/M2TMNXRAD_5.12.4/summary)\nDoes anyone know if including time-specific albedo this has been shown to be even more important with bifacials?\r\n\r\n(I would think yes.)\n> Does anyone know if including time-specific albedo this has been shown to be even more important with bifacials?\r\n> \r\n> (I would think yes.)\r\n\r\nYes, it is more important than for single-sided modules. There are ground surfaces where the albedo depends on the solar elevation and hence time of day.\r\n\r\nOne caution about albedo from satellite-derived irradiance: those values are at least km^2 scale, and are observed from space, whereas a PV model is assuming that the albedo is localized (m^2) and has been determined from the irradiance reaching the ground. [SolarAnywhere ](https://www.solaranywhere.com/support/data-fields/albedo/)provides an informative list of caveats.\r\n\r\nThe good news is that the uncertainty in albedo is typically secondary to uncertainty in other data such as GHI, when considering uncertainty in energy production.\r\n\nWhich is the better course of action?\r\n\r\n1. Leave `albedo` on `PVsystem` or `Array`. Edit the docstrings and add tests to make it explicit that `PVSystem.albedo` or `Array.albedo` can be a Series. Advantages: avoids deprecating and removing `PVSystem.albedo`. Downside: users obtaining albedo from weather data sources have an extra step to perform when using `ModelChain` methods.\r\n2. Add `albedo` as an optional column in `weather`, and have `ModelChain` methods use `weather['albedo']` instead of `PVSystem.albedo` when `weather['albedo']` is present. Advantages: convenient for ModelChain users, and avoids deprecating `PVsystem.albedo`. Disadvatanges: potential for confusion when a user also assigns `PVSystem.albedo`. \r\n3. Move `albedo` from `PVSystem` to `weather`. Same advantages as 2, but requires deprecation, and adds an extra step for users who aren't getting albedo with weather data, and would probably lead to adding code to `ModelChain` similar to the code that infers albedo from `PVSystem.surface_type`.\n@cwhanse I lean toward option 1. However, it seems that for option 1, a user might have to duplicate computations already done in the `ModelChain` in the preliminary step of using the weather to compute the existing `PVSystem.albedo` (as some formulaic combination of ground conditions and weather). Do you know if this is really an issue?\nThanks for the discussion around this feature.  I favor Option 2, noting that PVSystem.albedo is already an optional argument.  Option 1 is prone to bugs and cases need to be handled checking for mismatches in series/dataframe lengths or perhaps same series lengths but different indices.  I would discourage Option 3, as it seems there is both a basis for and utility in retaining albedo as a property of PVSystem.\nFor option 2, I would say raise an exception (or at very least a warning) if two albedos are specified. This could definitely be the source of a subtle computational bug when users don't realize a weather file they pulled is doing something they didn't intend.\n+1 for option 2, and I agree with @campanelli-sunpower that an exception should be raised if two albedos are specified\n+1 for option 2, but I like keeping the old method for the case where you just want to set either a single annual albedo or monthly albedos (tho not sure if this option currently exists). I agree also raise exception if both are specified - we can always change it later.", "created_at": "2022-06-21T18:33:24Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_albedo_in_weather[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_albedo_in_weather[list]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance_albedo", "pvlib/tests/test_pvsystem.py::test_Array_get_irradiance", "pvlib/tests/test_tracking.py::test_get_irradiance"], "PASS_TO_PASS": ["pvlib/tests/test_clearsky.py::test_ineichen_series", "pvlib/tests/test_clearsky.py::test_ineichen_series_perez_enhancement", "pvlib/tests/test_clearsky.py::test_ineichen_scalar_input", "pvlib/tests/test_clearsky.py::test_ineichen_nans", "pvlib/tests/test_clearsky.py::test_ineichen_arrays", "pvlib/tests/test_clearsky.py::test_ineichen_dni_extra", "pvlib/tests/test_clearsky.py::test_ineichen_altitude", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_leapyear", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_nointerp", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_months", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_months_leapyear", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_nointerp_months", "pvlib/tests/test_clearsky.py::test_haurwitz", "pvlib/tests/test_clearsky.py::test_simplified_solis_scalar_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_scalar_neg_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_series_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_dni_extra", "pvlib/tests/test_clearsky.py::test_simplified_solis_pressure", "pvlib/tests/test_clearsky.py::test_simplified_solis_aod700", "pvlib/tests/test_clearsky.py::test_simplified_solis_precipitable_water", "pvlib/tests/test_clearsky.py::test_simplified_solis_small_scalar_pw", "pvlib/tests/test_clearsky.py::test_simplified_solis_return_arrays", "pvlib/tests/test_clearsky.py::test_simplified_solis_nans_arrays", "pvlib/tests/test_clearsky.py::test_simplified_solis_nans_series", "pvlib/tests/test_clearsky.py::test_linke_turbidity_corners", "pvlib/tests/test_clearsky.py::test_degrees_to_index_1", "pvlib/tests/test_clearsky.py::test_detect_clearsky", "pvlib/tests/test_clearsky.py::test_detect_clearsky_defaults", "pvlib/tests/test_clearsky.py::test_detect_clearsky_components", "pvlib/tests/test_clearsky.py::test_detect_clearsky_iterations", "pvlib/tests/test_clearsky.py::test_detect_clearsky_kwargs", "pvlib/tests/test_clearsky.py::test_detect_clearsky_window", "pvlib/tests/test_clearsky.py::test_detect_clearsky_time_interval", "pvlib/tests/test_clearsky.py::test_detect_clearsky_arrays", "pvlib/tests/test_clearsky.py::test_detect_clearsky_irregular_times", "pvlib/tests/test_clearsky.py::test_detect_clearsky_missing_index", "pvlib/tests/test_clearsky.py::test__line_length_windowed", "pvlib/tests/test_clearsky.py::test__max_diff_windowed", "pvlib/tests/test_clearsky.py::test__calc_stats", "pvlib/tests/test_clearsky.py::test_bird", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_epoch_year", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_nrel_numba", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_invalid", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_simple_float", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_simple_series", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_0", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_series", "pvlib/tests/test_irradiance.py::test_grounddiffuse_albedo_invalid_surface", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_surface", "pvlib/tests/test_irradiance.py::test_isotropic_float", "pvlib/tests/test_irradiance.py::test_isotropic_series", "pvlib/tests/test_irradiance.py::test_klucher_series_float", "pvlib/tests/test_irradiance.py::test_klucher_series", "pvlib/tests/test_irradiance.py::test_haydavies", "pvlib/tests/test_irradiance.py::test_reindl", "pvlib/tests/test_irradiance.py::test_king", "pvlib/tests/test_irradiance.py::test_perez", "pvlib/tests/test_irradiance.py::test_perez_components", "pvlib/tests/test_irradiance.py::test_perez_negative_horizon", "pvlib/tests/test_irradiance.py::test_perez_arrays", "pvlib/tests/test_irradiance.py::test_perez_scalar", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[isotropic]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[klucher]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[haydavies]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[reindl]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[king]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[perez]", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_model_invalid", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_missing_dni_extra", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_missing_airmass", "pvlib/tests/test_irradiance.py::test_campbell_norman", "pvlib/tests/test_irradiance.py::test_get_total_irradiance", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[isotropic]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[klucher]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[haydavies]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[reindl]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[king]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo[perez]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[isotropic]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[klucher]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[haydavies]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[reindl]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[king]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[perez]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_missing_dni_extra", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_missing_airmass", "pvlib/tests/test_irradiance.py::test_poa_components", "pvlib/tests/test_irradiance.py::test_disc_value[93193-expected0]", "pvlib/tests/test_irradiance.py::test_disc_value[None-expected1]", "pvlib/tests/test_irradiance.py::test_disc_value[101325-expected2]", "pvlib/tests/test_irradiance.py::test_disc_overirradiance", "pvlib/tests/test_irradiance.py::test_disc_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_dirint_value", "pvlib/tests/test_irradiance.py::test_dirint_nans", "pvlib/tests/test_irradiance.py::test_dirint_tdew", "pvlib/tests/test_irradiance.py::test_dirint_no_delta_kt", "pvlib/tests/test_irradiance.py::test_dirint_coeffs", "pvlib/tests/test_irradiance.py::test_dirint_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_gti_dirint", "pvlib/tests/test_irradiance.py::test_erbs", "pvlib/tests/test_irradiance.py::test_erbs_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_erbs_all_scalar", "pvlib/tests/test_irradiance.py::test_dirindex", "pvlib/tests/test_irradiance.py::test_dirindex_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_dni", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[0-0-0-0-0-1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[30-180-30-180-0-1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[30-180-150-0-180--1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[90-0-30-60-75.5224878-0.25]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[90-0-30-170-119.4987042--0.4924038]", "pvlib/tests/test_irradiance.py::test_aoi_projection_precision", "pvlib/tests/test_irradiance.py::test_kt_kt_prime_factor", "pvlib/tests/test_irradiance.py::test_clearsky_index", "pvlib/tests/test_irradiance.py::test_clearness_index", "pvlib/tests/test_irradiance.py::test_clearness_index_zenith_independent", "pvlib/tests/test_modelchain.py::test_ModelChain_creation", "pvlib/tests/test_modelchain.py::test_with_sapm", "pvlib/tests/test_modelchain.py::test_with_pvwatts", "pvlib/tests/test_modelchain.py::test_run_model_with_irradiance", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_invalid_inverter_params_arrays[adr]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_no_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_arrays_one_missing_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_times_error_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_times_arrays", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dhi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[ghi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dni]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_perez", "pvlib/tests/test_modelchain.py::test_run_model_gueymard_perez", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_sapm_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_pvsyst_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_faiman_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_fuentes_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_noct_sam_temp", "pvlib/tests/test_modelchain.py::test_run_model_tracker", "pvlib/tests/test_modelchain.py::test_run_model_tracker_list", "pvlib/tests/test_modelchain.py::test__assign_total_irrad", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_different_indices", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_missing_column", "pvlib/tests/test_modelchain.py::test__prepare_temperature", "pvlib/tests/test_modelchain.py::test__prepare_temperature_len1_weather_tuple", "pvlib/tests/test_modelchain.py::test__prepare_temperature_arrays_weather", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params0-sapm_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params1-pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params2-faiman_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params3-fuentes_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params4-noct_sam_temp]", "pvlib/tests/test_modelchain.py::test_run_model_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_tracking", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_poa_global_differs", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_minimal_input", "pvlib/tests/test_modelchain.py::test_run_model_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_weather_single_array", "pvlib/tests/test_modelchain.py::test_infer_dc_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[desoto]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[singlediode]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvwatts_dc]", "pvlib/tests/test_modelchain.py::test_infer_dc_model_incomplete", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[cec]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[desoto]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec_native]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[sapm_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[faiman_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[fuentes_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[noct_sam_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model_invalid", "pvlib/tests/test_modelchain.py::test_temperature_model_inconsistent", "pvlib/tests/test_modelchain.py::test_dc_model_user_func", "pvlib/tests/test_modelchain.py::test_pvwatts_dc_multiple_strings", "pvlib/tests/test_modelchain.py::test_ac_models[sandia]", "pvlib/tests/test_modelchain.py::test_ac_models[adr]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts]", "pvlib/tests/test_modelchain.py::test_ac_models[sandia_multi]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts_multi]", "pvlib/tests/test_modelchain.py::test_ac_model_user_func", "pvlib/tests/test_modelchain.py::test_ac_model_not_a_model", "pvlib/tests/test_modelchain.py::test_infer_ac_model_invalid_params", "pvlib/tests/test_modelchain.py::test_aoi_models[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_model_no_loss", "pvlib/tests/test_modelchain.py::test_aoi_model_user_func", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[ashrae]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[physical]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model_invalid", "pvlib/tests/test_modelchain.py::test_spectral_models[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_dc_ohmic_model_ohms_from_percent", "pvlib/tests/test_modelchain.py::test_dc_ohmic_model_no_dc_ohmic_loss", "pvlib/tests/test_modelchain.py::test_dc_ohmic_ext_def", "pvlib/tests/test_modelchain.py::test_dc_ohmic_not_a_model", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts_arrays", "pvlib/tests/test_modelchain.py::test_losses_models_ext_def", "pvlib/tests/test_modelchain.py::test_losses_models_no_loss", "pvlib/tests/test_modelchain.py::test_invalid_dc_model_params", "pvlib/tests/test_modelchain.py::test_invalid_models[dc_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[ac_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[aoi_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[spectral_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[temperature_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[losses_model]", "pvlib/tests/test_modelchain.py::test_bad_get_orientation", "pvlib/tests/test_modelchain.py::test_with_sapm_pvsystem_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_no_extra_kwargs", "pvlib/tests/test_modelchain.py::test_ModelChain_attributes_deprecated_10", "pvlib/tests/test_modelchain.py::test_basic_chain_alt_az", "pvlib/tests/test_modelchain.py::test_basic_chain_altitude_pressure", "pvlib/tests/test_modelchain.py::test_complete_irradiance_clean_run", "pvlib/tests/test_modelchain.py::test_complete_irradiance", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_unknown_attribute", "pvlib/tests/test_modelchain.py::test_inconsistent_array_params", "pvlib/tests/test_modelchain.py::test_modelchain__common_keys", "pvlib/tests/test_modelchain.py::test__irrad_for_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[ashrae-model_params0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[physical-model_params1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[martin_ruiz-model_params2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_iam", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_interp", "pvlib/tests/test_pvsystem.py::test__normalize_sam_product_names", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_invalid", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_raise_no_parameters", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_cecmod", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_cecinverter", "pvlib/tests/test_pvsystem.py::test_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[1.5-1.00028714375]", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[airmass1-expected1]", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[airmass2-expected2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_spectral_loss", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_spectral_loss", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters0-multisi-None]", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters1-multisi-None]", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters2-None-coefficients2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_first_solar_spectral_loss", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input0-1140.0510967821876]", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input1-expected1]", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input2-expected2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_effective_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[20-poa_diffuse0-aoi0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[poa_direct1-poa_diffuse1-aoi1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[poa_direct2-poa_diffuse2-20]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_celltemp_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_celltemp_different_arrays", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvsyst_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_faiman_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp_error", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_cell_temperature_invalid", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_module_height", "pvlib/tests/test_pvsystem.py::test_Array__infer_temperature_model_params", "pvlib/tests/test_pvsystem.py::test_Array__infer_cell_type", "pvlib/tests/test_pvsystem.py::test_calcparams_desoto", "pvlib/tests/test_pvsystem.py::test_calcparams_cec", "pvlib/tests/test_pvsystem.py::test_calcparams_cec_extra_params_propagation", "pvlib/tests/test_pvsystem.py::test_calcparams_pvsyst", "pvlib/tests/test_pvsystem.py::test_PVSystem_calcparams_desoto", "pvlib/tests/test_pvsystem.py::test_PVSystem_calcparams_pvsyst", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_desoto]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_cec]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_desoto-1-celltemp0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_desoto-irrad1-1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_cec-1-celltemp2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_cec-irrad3-1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_pvsyst-1-celltemp4]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_pvsyst-irrad5-1]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i0]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i1]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i2]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i3]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i4]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i5]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i6]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i7]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i8]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i9]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i10]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_PVSystem_i_from_v", "pvlib/tests/test_pvsystem.py::test_i_from_v_size", "pvlib/tests/test_pvsystem.py::test_v_from_i_size", "pvlib/tests/test_pvsystem.py::test_mpp_floats", "pvlib/tests/test_pvsystem.py::test_mpp_array", "pvlib/tests/test_pvsystem.py::test_mpp_series", "pvlib/tests/test_pvsystem.py::test_singlediode_series", "pvlib/tests/test_pvsystem.py::test_singlediode_array", "pvlib/tests/test_pvsystem.py::test_singlediode_floats", "pvlib/tests/test_pvsystem.py::test_singlediode_floats_ivcurve", "pvlib/tests/test_pvsystem.py::test_singlediode_series_ivcurve", "pvlib/tests/test_pvsystem.py::test_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_sandia", "pvlib/tests/test_pvsystem.py::test_PVSystem_snlinverter", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_sandia_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[sandia]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[adr]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[pvwatts]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_adr", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_adr_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_invalid", "pvlib/tests/test_pvsystem.py::test_PVSystem_creation", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_creation", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_aoi", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_get_aoi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance_model", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_irradiance_multi_irrad", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module_parameters]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module_type]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[temperature_model_parameters]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[albedo]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[surface_tilt]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[surface_azimuth]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[racking_model]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[modules_per_string]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[strings_per_inverter]", "pvlib/tests/test_pvsystem.py::test_PVSystem___repr__", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array___repr__", "pvlib/tests/test_pvsystem.py::test_Array___repr__", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_scalars", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_arrays", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_series", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_default", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_arrays", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_series", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_dc", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_dc_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_pvwatts_dc", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_pvwatts_dc_value_error", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_losses", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_ac", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_ac_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_num_arrays", "pvlib/tests/test_pvsystem.py::test_PVSystem_at_least_one_array", "pvlib/tests/test_pvsystem.py::test_combine_loss_factors", "pvlib/tests/test_pvsystem.py::test_no_extra_kwargs", "pvlib/tests/test_pvsystem.py::test_AbstractMount_constructor", "pvlib/tests/test_pvsystem.py::test_FixedMount_constructor", "pvlib/tests/test_pvsystem.py::test_FixedMount_get_orientation", "pvlib/tests/test_pvsystem.py::test_SingleAxisTrackerMount_constructor", "pvlib/tests/test_pvsystem.py::test_SingleAxisTrackerMount_get_orientation", "pvlib/tests/test_pvsystem.py::test_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_PVSystem_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_dc_ohmic_losses", "pvlib/tests/test_pvsystem.py::test_Array_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[sapm-keys0]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[fuentes-keys1]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[noct_sam-keys2]", "pvlib/tests/test_tracking.py::test_solar_noon", "pvlib/tests/test_tracking.py::test_scalars", "pvlib/tests/test_tracking.py::test_arrays", "pvlib/tests/test_tracking.py::test_nans", "pvlib/tests/test_tracking.py::test_arrays_multi", "pvlib/tests/test_tracking.py::test_azimuth_north_south", "pvlib/tests/test_tracking.py::test_max_angle", "pvlib/tests/test_tracking.py::test_backtrack", "pvlib/tests/test_tracking.py::test_axis_tilt", "pvlib/tests/test_tracking.py::test_axis_azimuth", "pvlib/tests/test_tracking.py::test_horizon_flat", "pvlib/tests/test_tracking.py::test_horizon_tilted", "pvlib/tests/test_tracking.py::test_low_sun_angles", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_tracking", "pvlib/tests/test_tracking.py::test_get_aoi", "pvlib/tests/test_tracking.py::test_SingleAxisTracker___repr__", "pvlib/tests/test_tracking.py::test_calc_axis_tilt", "pvlib/tests/test_tracking.py::test_slope_aware_backtracking", "pvlib/tests/test_tracking.py::test_singleaxis_aoi_gh1221", "pvlib/tests/test_tracking.py::test_calc_surface_orientation_types", "pvlib/tests/test_tracking.py::test_calc_surface_orientation_kwargs", "pvlib/tests/test_tracking.py::test_calc_surface_orientation_special"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1720", "base_commit": "a8be738418dee0a2b93c241fbd5e0bc82f4b8680", "patch": "diff --git a/pydicom/config.py b/pydicom/config.py\n--- a/pydicom/config.py\n+++ b/pydicom/config.py\n@@ -268,6 +268,18 @@ def disable_value_validation() -> Generator:\n         settings._writing_validation_mode = writing_mode\n \n \n+@contextmanager\n+def strict_reading() -> Generator:\n+    \"\"\"Context manager to temporarily enably strict value validation\n+    for reading.\"\"\"\n+    original_reading_mode = settings._reading_validation_mode\n+    try:\n+        settings.reading_validation_mode = RAISE\n+        yield\n+    finally:\n+        settings._reading_validation_mode = original_reading_mode\n+\n+\n convert_wrong_length_to_UN = False\n \"\"\"Convert a field VR to \"UN\" and return bytes if bytes length is invalid.\n Default ``False``.\ndiff --git a/pydicom/dataset.py b/pydicom/dataset.py\n--- a/pydicom/dataset.py\n+++ b/pydicom/dataset.py\n@@ -16,6 +16,7 @@\n \"\"\"\n import copy\n from bisect import bisect_left\n+from contextlib import nullcontext\n import io\n from importlib.util import find_spec as have_package\n import inspect  # for __dir__\n@@ -2490,18 +2491,23 @@ def to_json_dict(\n             :class:`Dataset` representation based on the DICOM JSON Model.\n         \"\"\"\n         json_dataset = {}\n-        for key in self.keys():\n-            json_key = '{:08X}'.format(key)\n-            try:\n-                data_element = self[key]\n-                json_dataset[json_key] = data_element.to_json_dict(\n-                    bulk_data_element_handler=bulk_data_element_handler,\n-                    bulk_data_threshold=bulk_data_threshold\n-                )\n-            except Exception as exc:\n-                logger.error(f\"Error while processing tag {json_key}\")\n-                if not suppress_invalid_tags:\n-                    raise exc\n+        context = (\n+            config.strict_reading() if suppress_invalid_tags\n+            else nullcontext()\n+        )\n+        with context:\n+            for key in self.keys():\n+                json_key = '{:08X}'.format(key)\n+                try:\n+                    data_element = self[key]\n+                    json_dataset[json_key] = data_element.to_json_dict(\n+                        bulk_data_element_handler=bulk_data_element_handler,\n+                        bulk_data_threshold=bulk_data_threshold\n+                    )\n+                except Exception as exc:\n+                    logger.error(f\"Error while processing tag {json_key}\")\n+                    if not suppress_invalid_tags:\n+                        raise exc\n \n         return json_dataset\n \ndiff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -1248,6 +1248,48 @@ def DS(\n     return DSfloat(val, auto_format, validation_mode)\n \n \n+class ISfloat(float):\n+    \"\"\"Store value for an element with VR **IS** as :class:`float`.\n+\n+    Stores original integer string for exact rewriting of the string\n+    originally read or stored.\n+\n+    Note: By the DICOM standard, IS can only be an :class:`int`,\n+    however, it is not uncommon to see float IS values.  This class\n+    is used if the config settings allow non-strict reading.\n+\n+    Generally, use :class:`~pydicom.valuerep.IS` to create IS values,\n+    this is returned instead if the value cannot be represented as an\n+    :class:`int`.  See :class:`~pydicom.valuerep.IS` for details of the\n+    parameters and return values.\n+    \"\"\"\n+    def __new__(  # type: ignore[misc]\n+            cls: Type[\"ISfloat\"], val: Union[str, float, Decimal],\n+            validation_mode: int = None\n+    ) -> float:\n+        return super().__new__(cls, val)\n+\n+    def __init__(self, val: Union[str, float, Decimal],\n+                 validation_mode: int = None) -> None:\n+        # If a string passed, then store it\n+        if isinstance(val, str):\n+            self.original_string = val.strip()\n+        elif isinstance(val, (IS, ISfloat)) and hasattr(val, 'original_string'):\n+            self.original_string = val.original_string\n+        if validation_mode:\n+            msg = (\n+                f'Value \"{str(self)}\" is not valid for elements with a VR '\n+                'of IS'\n+            )\n+            if validation_mode == config.WARN:\n+                warnings.warn(msg)\n+            elif validation_mode == config.RAISE:\n+                msg += (\n+                    \"\\nSet reading_validation_mode to WARN or IGNORE to bypass\"\n+                )\n+                raise TypeError(msg)\n+\n+\n class IS(int):\n     \"\"\"Store value for an element with VR **IS** as :class:`int`.\n \n@@ -1258,7 +1300,7 @@ class IS(int):\n     def __new__(  # type: ignore[misc]\n             cls: Type[\"IS\"], val: Union[None, str, int, float, Decimal],\n             validation_mode: int = None\n-    ) -> Optional[Union[str, \"IS\"]]:\n+    ) -> Optional[Union[str, \"IS\", \"ISfloat\"]]:\n         \"\"\"Create instance if new integer string\"\"\"\n         if val is None:\n             return val\n@@ -1272,16 +1314,16 @@ def __new__(  # type: ignore[misc]\n             validate_value(\"IS\", val, validation_mode)\n \n         try:\n-            newval = super().__new__(cls, val)\n+            newval: Union[IS, ISfloat] = super().__new__(cls, val)\n         except ValueError:\n             # accept float strings when no integer loss, e.g. \"1.0\"\n             newval = super().__new__(cls, float(val))\n \n-        # check if a float or Decimal passed in, then could have lost info,\n-        # and will raise error. E.g. IS(Decimal('1')) is ok, but not IS(1.23)\n-        #   IS('1.23') will raise ValueError\n+        # If a float or Decimal was passed in, check for non-integer,\n+        # i.e. could lose info if converted to int\n+        # If so, create an ISfloat instead (if allowed by settings)\n         if isinstance(val, (float, Decimal, str)) and newval != float(val):\n-            raise TypeError(\"Could not convert value to integer without loss\")\n+            newval = ISfloat(val, validation_mode)\n \n         # Checks in case underlying int is >32 bits, DICOM does not allow this\n         if (not -2**31 <= newval < 2**31 and\n", "test_patch": "diff --git a/pydicom/tests/test_json.py b/pydicom/tests/test_json.py\n--- a/pydicom/tests/test_json.py\n+++ b/pydicom/tests/test_json.py\n@@ -5,6 +5,7 @@\n \n import pytest\n \n+from pydicom import config\n from pydicom import dcmread\n from pydicom.data import get_testdata_file\n from pydicom.dataelem import DataElement, RawDataElement\n@@ -293,13 +294,10 @@ def test_suppress_invalid_tags_with_failed_dataelement(self):\n         # we have to add a RawDataElement as creating a DataElement would\n         # already raise an exception\n         ds[0x00082128] = RawDataElement(\n-            Tag(0x00082128), 'IS', 4, b'5.25', 0, True, True)\n-\n-        with pytest.raises(TypeError):\n-            ds.to_json_dict()\n+            Tag(0x00082128), 'IS', 4, b'5.25', 0, True, True\n+        )\n \n         ds_json = ds.to_json_dict(suppress_invalid_tags=True)\n-\n         assert \"00082128\" not in ds_json\n \n \ndiff --git a/pydicom/tests/test_valuerep.py b/pydicom/tests/test_valuerep.py\n--- a/pydicom/tests/test_valuerep.py\n+++ b/pydicom/tests/test_valuerep.py\n@@ -6,6 +6,7 @@\n from datetime import datetime, date, time, timedelta, timezone\n from decimal import Decimal\n from itertools import chain\n+from io import BytesIO\n import pickle\n import math\n import sys\n@@ -19,9 +20,10 @@\n from pydicom.data import get_testdata_file\n from pydicom.dataset import Dataset\n from pydicom._dicom_dict import DicomDictionary, RepeatersDictionary\n+from pydicom.filereader import read_dataset\n from pydicom.tag import Tag\n from pydicom.valuerep import (\n-    DS, IS, DSfloat, DSdecimal, PersonName, VR, STANDARD_VR,\n+    DS, IS, DSfloat, DSdecimal, ISfloat, PersonName, VR, STANDARD_VR,\n     AMBIGUOUS_VR, STR_VR, BYTES_VR, FLOAT_VR, INT_VR, LIST_VR\n )\n from pydicom.values import convert_value\n@@ -889,11 +891,31 @@ def test_valid_value(self, disable_value_validation):\n         assert 42 == IS(\"42.0\")\n         assert 42 == IS(42.0)\n \n+    def test_float_value(self):\n+        \"\"\"Read binary value of IS that is actually a float\"\"\"\n+        # from issue #1661\n+        # Create BytesIO with single data element for Exposure (0018,1152)\n+        #   length 4, value \"14.5\"\n+        bin_elem = b\"\\x18\\x00\\x52\\x11\\x04\\x00\\x00\\x0014.5\"\n+        with BytesIO(bin_elem) as bio:\n+            ds = read_dataset(bio, True, True)\n+        assert isinstance(ds.Exposure, ISfloat)\n+        assert ds.Exposure == 14.5\n+\n+        # Strict checking raises an error\n+        with pytest.raises(ValueError):\n+            _ = IS(\"14.5\", validation_mode=config.RAISE)\n+        with pytest.raises(TypeError):\n+            _ = IS(14.5, validation_mode=config.RAISE)\n+\n+    def test_float_init(self):\n+        \"\"\"New ISfloat created from another behaves correctly\"\"\"\n+        is1 = IS(\"14.5\", validation_mode=config.IGNORE)\n+        is2 = IS(is1)\n+        assert is1 == is2\n+        assert is2.original_string == is1.original_string\n+\n     def test_invalid_value(self, disable_value_validation):\n-        with pytest.raises(TypeError, match=\"Could not convert value\"):\n-            IS(0.9)\n-        with pytest.raises(TypeError, match=\"Could not convert value\"):\n-            IS(\"0.9\")\n         with pytest.raises(ValueError, match=\"could not convert string\"):\n             IS(\"foo\")\n \n", "problem_statement": "Strict adherence to VR during parsing is detrimental due to commonplace vendor interpretations\n**Describe the bug**\r\nDICOM Files from GE modalities, which when parsed, raise a TypeError caused by \"violating\" the VR imposed by the DICOM standard; however, real world modalities have and continue to generate such files for good cause.\r\n\r\nFor example the following is raised\r\n\r\n`TypeError('Could not convert value to integer without loss')`\r\n\r\nby a real world DICOM file which has a value\r\n\r\n`(0018,1152) IS [14.5]                                   #   4, 1 Exposure`\r\n\r\nwhere IS is a Value Representation defined as\r\n\r\n> IS - Integer String\r\n\r\n> A string of characters representing an Integer in base-10 (decimal), shall contain only the characters 0 - 9, with an optional leading \"+\" or \"-\". It may be padded with leading and/or trailing spaces. Embedded spaces are not allowed.\r\n\r\n> The integer, n, represented shall be in the range: -231<= n <= (231-1).\r\n\r\n[See DICOM Part 5 Section 6.2](https://dicom.nema.org/dicom/2013/output/chtml/part05/sect_6.2.html)\r\n\r\nwhich means `14.5` is an invalid value due to the fractional portion .5 which definitely would lead to a loss in precision if converted to a pure integer value (of 14). \r\n\r\nAfter discussion with a senior engineer for the vendor, the following dialogue was obtained which quotes an article by David Clune, a well-respected, long-time member of the DICOM committee and community:\r\n\r\n> The tag pair in question is meant to contain the mAs value used for the exposure, which is not constrained to integer values, but for some reason the DICOM standard defines it as such.\r\n\r\n> An interesting article from someone responsible for maintaining the DICOM documentation explains the conundrum quite well:  \r\n\r\nhttp://dclunie.blogspot.com/2008/11/dicom-exposure-attribute-fiasco.html\r\n\r\n> Of note are two excerpts from that article:\r\n\r\n> \"The original ACR-NEMA standard specified ASCII numeric data elements for Exposure, Exposure Time and X-Ray Tube Current that could be decimal values; for no apparent reason DICOM 3.0 in 1993 constrained these to be integers, which for some modalities and subjects are too small to be sufficiently precise\"\r\n\r\n> and\r\n\r\n> \"The authors of DICOM, in attempting to maintain some semblance of backward compatibility with ACR-NEMA and at the same time apply more precise constraints, re-defined all ACR-NEMA data elements of VR AN as either IS or DS, the former being the AN integer numbers (with new size constraints), and the latter being the AN fixed point and floating point numbers. In the process of categorizing the old data elements into either IS or DS, not only were the obvious integers (like counts of images and other things) made into integers, but it appears that also any \"real world\" attribute that in somebody's expert opinion did not need greater precision than a whole integer, was so constrained as well.\"\r\n\r\n> I have inspected a few random DICOM files generated by various modalities and the value is stored accurately, even though it is a violation of the explicit value representation. Additionally, I have worked with (and support) various PACS platforms, and this is the first time this has been raised as an issue. So technically, you are correct that encoding that value as decimal violates the explicit VR, but it appears to be common practice to do so. \r\n\r\n**Expected behavior**\r\nTo deal with the reality of history with respect to the current standard, my opinion, as a long-standing DICOM PACS implementer at Medstrat, is that there is nothing to gain and everything to lose by raising a `TypeError` here. For cases where an integer VR, such as `IS`, could be read as a floating point number instead, then it should be allowed to be so, for at least a limited whitelist of tags.\r\n\r\nArguments against which come to mind are of the ilk that do not heed \"Although practicality beats purity\" as can be read if you \r\n\r\n[`>>> import this`](https://peps.python.org/pep-0020/)\r\n\r\n> Special cases aren't special enough to break the rules.\r\n> Although practicality beats purity.\r\n\r\n**Steps To Reproduce**\r\n\r\n`(0018,1152) IS [14.5]                                   #   4, 1 Exposure`\r\n\r\nSet any DICOM file to have the above for `Exposure` and then do this:\r\n\r\n```\r\n>>> from pydicom import config\r\n>>> pydicom.__version__\r\n'2.3.0'\r\n>>> config.settings.reading_validation_mode = config.IGNORE\r\n>>> ds = pydicom.dcmread('1.2.840.113619.2.107.20220429121335.1.1.dcm')\r\n>>> ds\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/dataset.py\", line 2306, in __str__\r\n    return self._pretty_str()\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/dataset.py\", line 2020, in _pretty_str\r\n    for elem in self:\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/dataset.py\", line 1240, in __iter__\r\n    yield self[tag]\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/dataset.py\", line 939, in __getitem__\r\n    self[tag] = DataElement_from_raw(elem, character_set, self)\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/dataelem.py\", line 859, in DataElement_from_raw\r\n    value = convert_value(vr, raw, encoding)\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/values.py\", line 771, in convert_value\r\n    return converter(byte_string, is_little_endian, num_format)\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/values.py\", line 348, in convert_IS_string\r\n    return MultiString(num_string, valtype=pydicom.valuerep.IS)\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/valuerep.py\", line 1213, in MultiString\r\n    return valtype(splitup[0])\r\n  File \"/usr/local/lib/python3.7/site-packages/pydicom/valuerep.py\", line 1131, in __new__\r\n    raise TypeError(\"Could not convert value to integer without loss\")\r\nTypeError: Could not convert value to integer without loss\r\n```\r\n\r\n**Your environment**\r\n\r\n```bash\r\nmodule       | version\r\n------       | -------\r\nplatform     | Darwin-21.5.0-x86_64-i386-64bit\r\nPython       | 3.7.5 (v3.7.5:5c02a39a0b, Oct 14 2019, 18:49:57)  [Clang 6.0 (clang-600.0.57)]\r\npydicom      | 2.2.2\r\ngdcm         | _module not found_\r\njpeg_ls      | _module not found_\r\nnumpy        | _module not found_\r\nPIL          | _module not found_\r\npylibjpeg    | _module not found_\r\nopenjpeg     | _module not found_\r\nlibjpeg      | _module not found_\r\n```\r\n\n", "hints_text": "Thank you for that thorough explanation (and of course we all know and respect David Clunie \ud83d\ude04 )!\r\nI understand the specific problem with `Exposure Time` (for a similar reason, there exists the tag `Exposure in \u03bcAs` additionally to `Exposure`, but no such thing exists for `Exposure Time`), and I am aware that this is not the only case where float values are written into `IS`. \r\n\r\nI think a similar issue came up before, and my preference would be to make this behavior dependent on the validation mode. The default validation mode for reading is to issue a warning, while for writing new data it is to raise an exception. This would not resolve the problem, though. The real problem is in the implementation of the tag class: the class representing `IS` values is derived from `int`. In my opinion, it would make sense to change that, though I'm not sure if we can do this consistently in a backwards-compatible way.\r\n\r\n@darcymason - I'm quite sure that we have discussed this before, I will try to find the respective issue. Anyway, what are your current thoughts here? \r\n\r\nCC @dclunie\r\n\nAh ok, @gsmethells has already [commented](https://github.com/pydicom/pydicom/issues/1643#issuecomment-1180789608) in the respective issue #1643.\n> I'm quite sure that we have discussed this before\r\n\r\nYes, it has come up several times.\r\n\r\nA new thought has come to me (maybe not thought through properly, or maybe someone has mentioned this before and I'm just thinking it is new): maybe we could create an ISFloat class, operating similarly to the DSFloat idea.  Then, just have an allow_IS_float config flag, perhaps even true by default (return an ISFloat only if it is not an exact int).  Python allows mixed math anyway, so I don't see any real problem in returning a non-int if further math is done.   Only problem might be code that did an `isinstance` check against `IS` but that should be very rare.\n> maybe we could create an ISFloat class, operating similarly to the DSFloat idea\r\n\r\nMy preference would be to change the `IS` class to support both `int` and `float`. We cannot always use float, as that could decrease the precision of integers, but in the case that float numbers are written in the tag, I would prefer to return a float. Not sure what problems this would bring, though... \r\nAdditionally, I think we can still couple the behavior to the validation mode, but I'm not completely sure here.\n> My preference would be to change the `IS` class to support both `int` and `float`.\r\n\r\nIs that actually possible to do, without recreating all the class methods for `int` (or `float`) for math operations?\n> Is that actually possible to do, without recreating all the class methods for int (or float) for math operations?\r\n\r\nI guess not - that would be the downside of that approach. Also an `isinstance(int)` would fail, of course. It is probably better to use `ISFloat` as you proposed and dynamically decide which class to use.\n> It is probably better to use ISFloat as you proposed and dynamically decide which class to use.\r\n\r\nActually that is what you have proposed - sorry, I misread that, I understood that you wanted to configure which class to use. Yes, I like your proposal!\n> Is that actually possible to do, without recreating all the class methods for int (or float) for math operations?\r\n\r\nActually, I feel like we did return a different class from `__new__` somewhere, or at least talked about it.  It turns out that it is possible to [return a different class from `__new__`](https://stackoverflow.com/questions/20221858/python-new-method-returning-something-other-than-class-instance).  I'm not sure it is advisable, though, that is really not being explicit to the user.\n> Actually, I feel like we did return a different class from `__new__` somewhere\r\n\r\nYes, we actually use this to return either a string or an int/float from `IS`/`DS`. This is also a common pattern in Python (they use it for example in `pathlib`), so I think it would not be unexpected.\nThank you for taking the time to discuss ideas and consider this. I understand wanting to adhere tightly to the standard (we do so for edge cases the majority of the time ourselves for our ortho PACS). I also understand the desire to listen to the \"import this\" zen of \"Simple is better than complex\" and, yes, I understand all too well how a large system can grow complicated.\r\n\r\nMy usual pause comes when my design desires run up against DICOM files in the field where there is a common violation of the standard. We maintain petabytes of DICOM images and this issue is common. I would be happy if reading/writing `dataset`s from/to files via pydicom continued to support maintaining existing values, regardless of whether those values violate the standard, (it does now, but is that an intentional design decision? A `dataset` only raises a `TypeError` when the value is directly attempted to be read out, whether via `__iter__` or otherwise) and also provided a (new/existing?) preference for reading `int` VR tag values as `float`.\r\n\r\nMy two cents.\n> reading/writing `dataset`s from/to files via pydicom continued to support maintaining existing values, regardless of whether those values violate the standard\r\n\r\nAny other position would become a deal breaker for our use of the library if it were ever otherwise, simply due to the realities of supporting customers. Especially when it comes to the need to update tags (e.g., a misspelled PatientName) while keeping other unrelated tags the same (i.e., no side-effects).\nSetting 2.4 milestone to at least consider this for the release.", "created_at": "2022-10-26T19:54:28Z", "version": "2.3", "FAIL_TO_PASS": ["pydicom/tests/test_json.py::TestPersonName::test_json_pn_from_file", "pydicom/tests/test_json.py::TestPersonName::test_pn_components_to_json", "pydicom/tests/test_json.py::TestPersonName::test_pn_components_from_json", "pydicom/tests/test_json.py::TestPersonName::test_empty_value", "pydicom/tests/test_json.py::TestPersonName::test_multi_value_to_json", "pydicom/tests/test_json.py::TestPersonName::test_dataelem_from_json", "pydicom/tests/test_json.py::TestAT::test_to_json", "pydicom/tests/test_json.py::TestAT::test_from_json", "pydicom/tests/test_json.py::TestAT::test_invalid_value_in_json", "pydicom/tests/test_json.py::TestAT::test_invalid_tag_in_json", "pydicom/tests/test_json.py::TestDataSetToJson::test_json_from_dicom_file", "pydicom/tests/test_json.py::TestDataSetToJson::test_roundtrip", "pydicom/tests/test_json.py::TestDataSetToJson::test_dataset_dumphandler", "pydicom/tests/test_json.py::TestDataSetToJson::test_dataelement_dumphandler", "pydicom/tests/test_json.py::TestDataSetToJson::test_sort_order", "pydicom/tests/test_json.py::TestDataSetToJson::test_suppress_invalid_tags", "pydicom/tests/test_json.py::TestDataSetToJson::test_suppress_invalid_tags_with_failed_dataelement", "pydicom/tests/test_json.py::TestSequence::test_nested_sequences", "pydicom/tests/test_json.py::TestBinary::test_inline_binary", "pydicom/tests/test_json.py::TestBinary::test_invalid_inline_binary", "pydicom/tests/test_json.py::TestBinary::test_valid_bulkdata_uri", "pydicom/tests/test_json.py::TestBinary::test_invalid_bulkdata_uri", "pydicom/tests/test_json.py::TestBinary::test_bulk_data_reader_is_called", "pydicom/tests/test_json.py::TestBinary::test_bulk_data_reader_is_called_2", "pydicom/tests/test_json.py::TestBinary::test_bulk_data_reader_is_called_within_SQ", "pydicom/tests/test_json.py::TestNumeric::test_numeric_values", "pydicom/tests/test_json.py::TestNumeric::test_numeric_types", "pydicom/tests/test_valuerep.py::TestTM::test_pickling", "pydicom/tests/test_valuerep.py::TestTM::test_pickling_tm_from_time", "pydicom/tests/test_valuerep.py::TestTM::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestTM::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestTM::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestTM::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestTM::test_comparison", "pydicom/tests/test_valuerep.py::TestTM::test_time_behavior", "pydicom/tests/test_valuerep.py::TestDT::test_pickling", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_with_timezone", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_dt_from_datetime", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_dt_from_datetime_with_timezone", "pydicom/tests/test_valuerep.py::TestDT::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestDT::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestDT::test_comparison", "pydicom/tests/test_valuerep.py::TestDT::test_datetime_behavior", "pydicom/tests/test_valuerep.py::TestDA::test_pickling", "pydicom/tests/test_valuerep.py::TestDA::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDA::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestDA::test_comparison", "pydicom/tests/test_valuerep.py::TestDA::test_date_behavior", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[3.14159265358979]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[-1234.456e78]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E-5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E+5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[+1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[42", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[nan]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[3.141592653589793]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1,000]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[127.0.0.1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1.e]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.0-1.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.0-0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.0--0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.123-0.123]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.321--0.321]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1e-05-1e-05]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[3.141592653589793-3.14159265358979]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-3.141592653589793--3.1415926535898]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[5.385940192876374e-07-5.3859401929e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-5.385940192876374e-07--5.385940193e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[12342534378.125532-12342534378.1255]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[64070869985876.78-64070869985876.8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.7976931348623157e+308-1.797693135e+308]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_wrong_type", "pydicom/tests/test_valuerep.py::TestDS::test_empty_value", "pydicom/tests/test_valuerep.py::TestDS::test_float_values", "pydicom/tests/test_valuerep.py::TestDSfloat::test_pickling[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_pickling[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_new_empty[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_new_empty[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str_value[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str_value[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_repr[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_repr[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSdecimal[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSdecimal[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_from_invalid_DS", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_length", "pydicom/tests/test_valuerep.py::TestDSfloat::test_handle_missing_leading_zero", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat_auto_format", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-nan]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan2]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestDSfloat::test_hash", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_pickling", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_float_value", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_new_empty", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_str_value", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSfloat", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSdecimal", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_repr", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_string_too_long", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_string_too_long_raises", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_from_invalid_DS", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val4]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val5]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val6]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val7]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSdecimal_auto_format", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_hash", "pydicom/tests/test_valuerep.py::TestIS::test_empty_value", "pydicom/tests/test_valuerep.py::TestIS::test_str_value", "pydicom/tests/test_valuerep.py::TestIS::test_valid_value", "pydicom/tests/test_valuerep.py::TestIS::test_float_value", "pydicom/tests/test_valuerep.py::TestIS::test_float_init", "pydicom/tests/test_valuerep.py::TestIS::test_invalid_value", "pydicom/tests/test_valuerep.py::TestIS::test_pickling", "pydicom/tests/test_valuerep.py::TestIS::test_longint", "pydicom/tests/test_valuerep.py::TestIS::test_overflow", "pydicom/tests/test_valuerep.py::TestIS::test_str", "pydicom/tests/test_valuerep.py::TestIS::test_repr", "pydicom/tests/test_valuerep.py::TestIS::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestIS::test_hash", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_default", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_enforce_valid_value", "pydicom/tests/test_valuerep.py::TestDecimalString::test_DS_decimal_set", "pydicom/tests/test_valuerep.py::TestDecimalString::test_valid_decimal_strings", "pydicom/tests/test_valuerep.py::TestDecimalString::test_invalid_decimal_strings", "pydicom/tests/test_valuerep.py::TestPersonName::test_last_first", "pydicom/tests/test_valuerep.py::TestPersonName::test_no_components", "pydicom/tests/test_valuerep.py::TestPersonName::test_copy", "pydicom/tests/test_valuerep.py::TestPersonName::test_three_component", "pydicom/tests/test_valuerep.py::TestPersonName::test_formatting", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_kr", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_comp_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_caret_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_not_equal", "pydicom/tests/test_valuerep.py::TestPersonName::test_encoding_carried", "pydicom/tests/test_valuerep.py::TestPersonName::test_hash", "pydicom/tests/test_valuerep.py::TestPersonName::test_next", "pydicom/tests/test_valuerep.py::TestPersonName::test_iterator", "pydicom/tests/test_valuerep.py::TestPersonName::test_contains", "pydicom/tests/test_valuerep.py::TestPersonName::test_length", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_veterinary", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator_from_bytes", "pydicom/tests/test_valuerep.py::TestDateTime::test_date", "pydicom/tests/test_valuerep.py::TestDateTime::test_date_time", "pydicom/tests/test_valuerep.py::TestDateTime::test_time", "pydicom/tests/test_valuerep.py::test_person_name_unicode_warns", "pydicom/tests/test_valuerep.py::test_set_value[AE-str-vm00-vmN0-Receiver]", "pydicom/tests/test_valuerep.py::test_set_value[AS-str-vm01-vmN1-PatientAge]", "pydicom/tests/test_valuerep.py::test_set_value[AT-int-vm02-vmN2-OffendingElement]", "pydicom/tests/test_valuerep.py::test_set_value[CS-str-vm03-vmN3-QualityControlSubject]", "pydicom/tests/test_valuerep.py::test_set_value[DA-str-vm04-vmN4-PatientBirthDate]", "pydicom/tests/test_valuerep.py::test_set_value[DS-str-vm05-vmN5-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DS-int-vm06-vmN6-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DS-float-vm07-vmN7-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DT-str-vm08-vmN8-AcquisitionDateTime]", "pydicom/tests/test_valuerep.py::test_set_value[FD-float-vm09-vmN9-RealWorldValueLUTData]", "pydicom/tests/test_valuerep.py::test_set_value[FL-float-vm010-vmN10-VectorAccuracy]", "pydicom/tests/test_valuerep.py::test_set_value[IS-str-vm011-vmN11-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[IS-int-vm012-vmN12-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[IS-float-vm013-vmN13-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[LO-str-vm014-vmN14-DataSetSubtype]", "pydicom/tests/test_valuerep.py::test_set_value[LT-str-vm015-vmN15-ExtendedCodeMeaning]", "pydicom/tests/test_valuerep.py::test_set_value[OB-bytes-vm016-vmN16-FillPattern]", "pydicom/tests/test_valuerep.py::test_set_value[OD-bytes-vm017-vmN17-DoubleFloatPixelData]", "pydicom/tests/test_valuerep.py::test_set_value[OF-bytes-vm018-vmN18-UValueData]", "pydicom/tests/test_valuerep.py::test_set_value[OL-bytes-vm019-vmN19-TrackPointIndexList]", "pydicom/tests/test_valuerep.py::test_set_value[OV-bytes-vm020-vmN20-SelectorOVValue]", "pydicom/tests/test_valuerep.py::test_set_value[OW-bytes-vm021-vmN21-TrianglePointIndexList]", "pydicom/tests/test_valuerep.py::test_set_value[PN-str-vm022-vmN22-PatientName]", "pydicom/tests/test_valuerep.py::test_set_value[SH-str-vm023-vmN23-CodeValue]", "pydicom/tests/test_valuerep.py::test_set_value[SL-int-vm024-vmN24-RationalNumeratorValue]", "pydicom/tests/test_valuerep.py::test_set_value[SQ-list-vm025-vmN25-BeamSequence]", "pydicom/tests/test_valuerep.py::test_set_value[SS-int-vm026-vmN26-SelectorSSValue]", "pydicom/tests/test_valuerep.py::test_set_value[ST-str-vm027-vmN27-InstitutionAddress]", "pydicom/tests/test_valuerep.py::test_set_value[SV-int-vm028-vmN28-SelectorSVValue]", "pydicom/tests/test_valuerep.py::test_set_value[TM-str-vm029-vmN29-StudyTime]", "pydicom/tests/test_valuerep.py::test_set_value[UC-str-vm030-vmN30-LongCodeValue]", "pydicom/tests/test_valuerep.py::test_set_value[UI-str-vm031-vmN31-SOPClassUID]", "pydicom/tests/test_valuerep.py::test_set_value[UL-int-vm032-vmN32-SimpleFrameList]", "pydicom/tests/test_valuerep.py::test_set_value[UN-bytes-vm033-vmN33-SelectorUNValue]", "pydicom/tests/test_valuerep.py::test_set_value[UR-str-vm034-vmN34-CodingSchemeURL]", "pydicom/tests/test_valuerep.py::test_set_value[US-int-vm035-vmN35-SourceAcquisitionBeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[UT-str-vm036-vmN36-StrainAdditionalInformation]", "pydicom/tests/test_valuerep.py::test_set_value[UV-int-vm037-vmN37-SelectorUVValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AE-str-vm00-vmN0-Receiver]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AS-str-vm01-vmN1-PatientAge]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AT-int-vm02-vmN2-OffendingElement]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[CS-str-vm03-vmN3-QualityControlSubject]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DA-str-vm04-vmN4-PatientBirthDate]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-str-vm05-vmN5-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-int-vm06-vmN6-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-float-vm07-vmN7-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DT-str-vm08-vmN8-AcquisitionDateTime]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[FD-float-vm09-vmN9-RealWorldValueLUTData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[FL-float-vm010-vmN10-VectorAccuracy]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-str-vm011-vmN11-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-int-vm012-vmN12-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-float-vm013-vmN13-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[LO-str-vm014-vmN14-DataSetSubtype]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[LT-str-vm015-vmN15-ExtendedCodeMeaning]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OB-bytes-vm016-vmN16-FillPattern]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OD-bytes-vm017-vmN17-DoubleFloatPixelData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OF-bytes-vm018-vmN18-UValueData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OL-bytes-vm019-vmN19-TrackPointIndexList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OV-bytes-vm020-vmN20-SelectorOVValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OW-bytes-vm021-vmN21-TrianglePointIndexList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[PN-str-vm022-vmN22-PatientName]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SH-str-vm023-vmN23-CodeValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SL-int-vm024-vmN24-RationalNumeratorValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SQ-list-vm025-vmN25-BeamSequence]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SS-int-vm026-vmN26-SelectorSSValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[ST-str-vm027-vmN27-InstitutionAddress]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SV-int-vm028-vmN28-SelectorSVValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[TM-str-vm029-vmN29-StudyTime]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UC-str-vm030-vmN30-LongCodeValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UI-str-vm031-vmN31-SOPClassUID]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UL-int-vm032-vmN32-SimpleFrameList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UN-bytes-vm033-vmN33-SelectorUNValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UR-str-vm034-vmN34-CodingSchemeURL]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[US-int-vm035-vmN35-SourceAcquisitionBeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UT-str-vm036-vmN36-StrainAdditionalInformation]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UV-int-vm037-vmN37-SelectorUVValue]", "pydicom/tests/test_valuerep.py::TestVR::test_behavior", "pydicom/tests/test_valuerep.py::TestVR::test_all_present"], "PASS_TO_PASS": [], "environment_setup_commit": "a8be738418dee0a2b93c241fbd5e0bc82f4b8680"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1458", "base_commit": "8da0b9b215ebfad5756051c891def88e426787e7", "patch": "diff --git a/pydicom/pixel_data_handlers/numpy_handler.py b/pydicom/pixel_data_handlers/numpy_handler.py\n--- a/pydicom/pixel_data_handlers/numpy_handler.py\n+++ b/pydicom/pixel_data_handlers/numpy_handler.py\n@@ -43,7 +43,9 @@\n | (0028,0100) | BitsAllocated             | 1    | 1, 8, 16, 32, | Required |\n |             |                           |      | 64            |          |\n +-------------+---------------------------+------+---------------+----------+\n-| (0028,0103) | PixelRepresentation       | 1    | 0, 1          | Required |\n+| (0028,0101) | BitsStored                | 1    | 1, 8, 12, 16  | Optional |\n++-------------+---------------------------+------+---------------+----------+\n+| (0028,0103) | PixelRepresentation       | 1C   | 0, 1          | Optional |\n +-------------+---------------------------+------+---------------+----------+\n \n \"\"\"\n@@ -284,16 +286,28 @@ def get_pixeldata(ds: \"Dataset\", read_only: bool = False) -> \"np.ndarray\":\n             \"the dataset\"\n         )\n \n+    # Attributes required by both Floating Point Image Pixel Module Attributes\n+    # and Image Pixel Description Macro Attributes\n     required_elements = [\n-        'BitsAllocated', 'Rows', 'Columns', 'PixelRepresentation',\n+        'BitsAllocated', 'Rows', 'Columns',\n         'SamplesPerPixel', 'PhotometricInterpretation'\n     ]\n+    if px_keyword[0] == 'PixelData':\n+        # Attributess required by Image Pixel Description Macro Attributes\n+        required_elements.extend(['PixelRepresentation', 'BitsStored'])\n     missing = [elem for elem in required_elements if elem not in ds]\n     if missing:\n         raise AttributeError(\n             \"Unable to convert the pixel data as the following required \"\n             \"elements are missing from the dataset: \" + \", \".join(missing)\n         )\n+    if ds.SamplesPerPixel > 1:\n+        if not hasattr(ds, 'PlanarConfiguration'):\n+            raise AttributeError(\n+                \"Unable to convert the pixel data as the following \"\n+                \"conditionally required element is missing from the dataset: \"\n+                \"PlanarConfiguration\"\n+            )\n \n     # May be Pixel Data, Float Pixel Data or Double Float Pixel Data\n     pixel_data = getattr(ds, px_keyword[0])\n", "test_patch": "diff --git a/pydicom/tests/test_numpy_pixel_data.py b/pydicom/tests/test_numpy_pixel_data.py\n--- a/pydicom/tests/test_numpy_pixel_data.py\n+++ b/pydicom/tests/test_numpy_pixel_data.py\n@@ -26,6 +26,8 @@\n * PlanarConfiguration\n \"\"\"\n \n+from copy import deepcopy\n+\n import pytest\n \n from pydicom import config\n@@ -1068,6 +1070,7 @@ def test_endianness_not_set(self):\n         ds.Rows = 10\n         ds.Columns = 10\n         ds.BitsAllocated = 16\n+        ds.BitsStored = 16\n         ds.PixelRepresentation = 0\n         ds.SamplesPerPixel = 1\n         ds.PhotometricInterpretation = 'MONOCHROME2'\n@@ -1105,16 +1108,60 @@ def test_no_pixel_data_raises(self):\n         with pytest.raises(AttributeError, match=msg):\n             get_pixeldata(ds)\n \n-    def test_missing_required_elem(self):\n+    def test_missing_required_elem_pixel_data_monochrome(self):\n         \"\"\"Tet get_pixeldata raises if dataset missing required element.\"\"\"\n-        ds = dcmread(EXPL_16_1_1F)\n-        del ds.BitsAllocated\n+        required_attrs = (\n+            'BitsAllocated',\n+            'BitsStored',\n+            'Rows',\n+            'Columns',\n+            'SamplesPerPixel',\n+            'PhotometricInterpretation',\n+            'PixelRepresentation',\n+        )\n+        for attr in required_attrs:\n+            ds = dcmread(EXPL_16_1_1F)\n+            delattr(ds, attr)\n+            msg = (\n+                r\"Unable to convert the pixel data as the following required \"\n+                r\"elements are missing from the dataset: {}\".format(attr)\n+            )\n+            with pytest.raises(AttributeError, match=msg):\n+                get_pixeldata(ds)\n+\n+    def test_missing_required_elem_pixel_data_color(self):\n+        \"\"\"Tet get_pixeldata raises if dataset missing required element.\"\"\"\n+        ds = dcmread(EXPL_8_3_1F)\n+        del ds.Rows\n+        del ds.Columns\n+        msg = (\n+            r\"Unable to convert the pixel data as the following required \"\n+            r\"elements are missing from the dataset: Rows, Columns\"\n+        )\n+        with pytest.raises(AttributeError, match=msg):\n+            get_pixeldata(ds)\n+\n+    def test_missing_conditionally_required_elem_pixel_data_color(self):\n+        \"\"\"Tet get_pixeldata raises if dataset missing required element.\"\"\"\n+        ds = dcmread(EXPL_8_3_1F)\n+        del ds.PlanarConfiguration\n+        msg = (\n+            r\"Unable to convert the pixel data as the following conditionally \"\n+            r\"required element is missing from the dataset: \"\n+            r\"PlanarConfiguration\"\n+        )\n+        with pytest.raises(AttributeError, match=msg):\n+            get_pixeldata(ds)\n+\n+    def test_missing_required_elem_float_pixel_data_monochrome(self):\n+        \"\"\"Tet get_pixeldata raises if dataset missing required element.\"\"\"\n+        ds = dcmread(IMPL_32_1_1F)\n+        ds.FloatPixelData = ds.PixelData\n+        del ds.PixelData\n         del ds.Rows\n-        del ds.SamplesPerPixel\n         msg = (\n             r\"Unable to convert the pixel data as the following required \"\n-            r\"elements are missing from the dataset: BitsAllocated, Rows, \"\n-            r\"SamplesPerPixel\"\n+            r\"elements are missing from the dataset: Rows\"\n         )\n         with pytest.raises(AttributeError, match=msg):\n             get_pixeldata(ds)\n", "problem_statement": "Pixel Representation attribute should be optional for pixel data handler\n**Describe the bug**\r\nThe NumPy pixel data handler currently [requires the Pixel Representation attribute](https://github.com/pydicom/pydicom/blob/8da0b9b215ebfad5756051c891def88e426787e7/pydicom/pixel_data_handlers/numpy_handler.py#L46). This is problematic, because in case of Float Pixel Data or Double Float Pixel Data the attribute shall be absent. Compare [Floating Point Image Pixel Module Attributes](http://dicom.nema.org/medical/dicom/current/output/chtml/part03/sect_C.7.6.24.html) versus [Image Pixel Description Macro Attributes](http://dicom.nema.org/medical/dicom/current/output/chtml/part03/sect_C.7.6.3.html#table_C.7-11c)\r\n\r\n**Expected behavior**\r\nI would expect the `Dataset.pixel_array` property to be able to decode a Float Pixel Data or Double Float Pixel Data element without presence of the Pixel Representation element in the metadata.\r\n\r\n**Steps To Reproduce**\r\n```python\r\nimport numpy as np\r\nfrom pydicom.dataset import Dataset, FileMetaDataset\r\n\r\n\r\nds = Dataset()\r\nds.file_meta = FileMetaDataset()\r\nds.file_meta.TransferSyntaxUID = '1.2.840.10008.1.2.1'\r\n\r\nds.BitsAllocated = 32\r\nds.SamplesPerPixel = 1\r\nds.Rows = 5\r\nds.Columns = 5\r\nds.PhotometricInterpretation = 'MONOCHROME2'\r\n\r\npixel_array = np.zeros((ds.Rows, ds.Columns), dtype=np.float32)\r\nds.FloatPixelData = pixel_array.flatten().tobytes()\r\n\r\nnp.array_equal(ds.pixel_array, pixel_array)\r\n```\n", "hints_text": "", "created_at": "2021-08-04T15:22:07Z", "version": "2.2", "FAIL_TO_PASS": ["pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_missing_required_elem_pixel_data_monochrome", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_missing_conditionally_required_elem_pixel_data_color"], "PASS_TO_PASS": ["pydicom/tests/test_numpy_pixel_data.py::test_unsupported_syntaxes", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_environment", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_supported_dataset", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_rgb_jpeg_dcmtk.dcm-data0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/JPEG-lossy.dcm-data1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_rgb_jpeg_gdcm.dcm-data2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small_jpeg_ls_lossless.dcm-data3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/n/fs/.pydicom/data/emri_small_jpeg_2k_lossless.dcm-data4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/JPEG2000.dcm-data5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small_RLE.dcm-data6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NoNumpyHandler::test_pixel_array_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_environment", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_unsupported_syntax_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_dataset_pixel_array_handler_needs_convert", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_dataset_pixel_array_no_pixels", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_rgb_jpeg_dcmtk.dcm-data0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/JPEG-lossy.dcm-data1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_rgb_jpeg_gdcm.dcm-data2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small_jpeg_ls_lossless.dcm-data3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/n/fs/.pydicom/data/emri_small_jpeg_2k_lossless.dcm-data4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/JPEG2000.dcm-data5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_can_access_unsupported_dataset[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small_RLE.dcm-data6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_pixel_array_8bit_un_signed", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_decompress_using_handler[numpy]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_decompress_using_handler[NumPy]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_decompress_using_handler[np]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_decompress_using_handler[np_handler]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_decompress_using_handler[numpy_handler]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_pixel_array_16bit_un_signed", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_pixel_array_32bit_un_signed", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_1sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_1sample_2frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_3sample_1frame_odd_size", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_3sample_1frame_ybr422", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_3sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_8bit_3sample_2frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/liver_1frame.dcm-data0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/liver.dcm-data1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/OBXXXX1A.dcm-data2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_rgb_small_odd.dcm-data3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/SC_ybr_full_422_uncompressed.dcm-data4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/OBXXXX1A_2frame.dcm-data5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb.dcm-data6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb_2frame.dcm-data7]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small.dcm-data8]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/emri_small.dcm-data9]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb_16bit.dcm-data10]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb_16bit_2frame.dcm-data11]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose_1frame.dcm-data12]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose.dcm-data13]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb_32bit.dcm-data14]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/n/fs/.pydicom/data/SC_rgb_32bit_2frame.dcm-data15]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_1bit_1sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_1bit_1sample_3frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_16bit_1sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_16bit_1sample_1frame_padded", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_16bit_1sample_10frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_16bit_3sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_16bit_3sample_2frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_1sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_1sample_15frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_3sample_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_3sample_2frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_float_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_32bit_float_15frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_64bit_float_1frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_little_64bit_float_15frame", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/liver_1frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/liver_expb_1frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/liver.dcm-/n/fs/.pydicom/data/liver_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/OBXXXX1A.dcm-/n/fs/.pydicom/data/OBXXXX1A_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/OBXXXX1A_2frame.dcm-/n/fs/.pydicom/data/OBXXXX1A_expb_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb.dcm-/n/fs/.pydicom/data/SC_rgb_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb_2frame.dcm-/n/fs/.pydicom/data/SC_rgb_expb_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/MR_small_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/emri_small.dcm-/n/fs/.pydicom/data/emri_small_big_endian.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb_16bit.dcm-/n/fs/.pydicom/data/SC_rgb_expb_16bit.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb_16bit_2frame.dcm-/n/fs/.pydicom/data/SC_rgb_expb_16bit_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose_1frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose_expb_1frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__2.2/pydicom/data/test_files/rtdose_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb_32bit.dcm-/n/fs/.pydicom/data/SC_rgb_expb_32bit.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/n/fs/.pydicom/data/SC_rgb_32bit_2frame.dcm-/n/fs/.pydicom/data/SC_rgb_expb_32bit_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_endianness_not_set", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_read_only", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_no_pixel_data_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_missing_required_elem_pixel_data_color", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_missing_required_elem_float_pixel_data_monochrome", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_unknown_pixel_representation_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_unsupported_syntaxes_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_bad_length_raises", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_missing_padding_warns", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_change_photometric_interpretation", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_array_read_only", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_array_read_only_bit_packed", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_ybr422_excess_padding", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_ybr422_wrong_interpretation", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_float_pixel_data", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_GetPixelData::test_double_float_pixel_data", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[-output0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x00-output1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x01-output2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x02-output3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x04-output4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x08-output5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x10-output6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[@-output8]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x80-output9]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\xaa-output10]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\xf0-output11]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x0f-output12]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\xff-output13]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x00\\x00-output14]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x00\\x01-output15]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x00\\x80-output16]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x00\\xff-output17]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x01\\x80-output18]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\x80\\x80-output19]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_UnpackBits::test_unpack[\\xff\\x80-output20]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[-input0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x00-input1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x01-input2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x02-input3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x04-input4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x08-input5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x10-input6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[@-input8]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x80-input9]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\xaa-input10]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\xf0-input11]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x0f-input12]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\xff-input13]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x00\\x00-input14]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x00\\x01-input15]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x00\\x80-input16]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x00\\xff-input17]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x01\\x80-input18]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\x80\\x80-input19]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack[\\xff\\x80-input20]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_non_binary_input", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_ndarray_input", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_padding", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00@-input0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00\\x10-input2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00\\x08-input3]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00\\x04-input4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00\\x02-input5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x00\\x01-input6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x80-input7]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[@-input8]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x10-input10]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x08-input11]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x04-input12]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x02-input13]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[\\x01-input14]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_pack_partial[-input15]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_PackBits::test_functional"], "environment_setup_commit": "0fa18d2a2179c92efc22200ed6b3689e66cecf92"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-1517", "base_commit": "304a197829f98e7425a46d872ada73176137e5ae", "patch": "diff --git a/src/sqlfluff/core/parser/helpers.py b/src/sqlfluff/core/parser/helpers.py\n--- a/src/sqlfluff/core/parser/helpers.py\n+++ b/src/sqlfluff/core/parser/helpers.py\n@@ -2,6 +2,7 @@\n \n from typing import Tuple, List, Any, Iterator, TYPE_CHECKING\n \n+from sqlfluff.core.errors import SQLParseError\n from sqlfluff.core.string_helpers import curtail_string\n \n if TYPE_CHECKING:\n@@ -26,11 +27,11 @@ def check_still_complete(\n     \"\"\"Check that the segments in are the same as the segments out.\"\"\"\n     initial_str = join_segments_raw(segments_in)\n     current_str = join_segments_raw(matched_segments + unmatched_segments)\n-    if initial_str != current_str:  # pragma: no cover\n-        raise RuntimeError(\n-            \"Dropped elements in sequence matching! {!r} != {!r}\".format(\n-                initial_str, current_str\n-            )\n+\n+    if initial_str != current_str:\n+        raise SQLParseError(\n+            f\"Could not parse: {current_str}\",\n+            segment=unmatched_segments[0],\n         )\n     return True\n \n", "test_patch": "diff --git a/test/dialects/ansi_test.py b/test/dialects/ansi_test.py\n--- a/test/dialects/ansi_test.py\n+++ b/test/dialects/ansi_test.py\n@@ -3,7 +3,7 @@\n import pytest\n import logging\n \n-from sqlfluff.core import FluffConfig, Linter\n+from sqlfluff.core import FluffConfig, Linter, SQLParseError\n from sqlfluff.core.parser import Lexer\n \n \n@@ -214,3 +214,29 @@ def test__dialect__ansi_parse_indented_joins(sql_string, indented_joins, meta_lo\n         idx for idx, raw_seg in enumerate(parsed.tree.iter_raw_seg()) if raw_seg.is_meta\n     )\n     assert res_meta_locs == meta_loc\n+\n+\n+@pytest.mark.parametrize(\n+    \"raw,expected_message\",\n+    [\n+        (\";;\", \"Line 1, Position 1: Found unparsable section: ';;'\"),\n+        (\"select id from tbl;\", \"\"),\n+        (\"select id from tbl;;\", \"Could not parse: ;\"),\n+        (\"select id from tbl;;;;;;\", \"Could not parse: ;;;;;\"),\n+        (\"select id from tbl;select id2 from tbl2;\", \"\"),\n+        (\n+            \"select id from tbl;;select id2 from tbl2;\",\n+            \"Could not parse: ;select id2 from tbl2;\",\n+        ),\n+    ],\n+)\n+def test__dialect__ansi_multiple_semicolons(raw: str, expected_message: str) -> None:\n+    \"\"\"Multiple semicolons should be properly handled.\"\"\"\n+    lnt = Linter()\n+    parsed = lnt.parse_string(raw)\n+\n+    assert len(parsed.violations) == (1 if expected_message else 0)\n+    if expected_message:\n+        violation = parsed.violations[0]\n+        assert isinstance(violation, SQLParseError)\n+        assert violation.desc() == expected_message\n", "problem_statement": "\"Dropped elements in sequence matching\" when doubled semicolon\n## Expected Behaviour\r\nFrankly, I'm not sure whether it (doubled `;`) should be just ignored or rather some specific rule should be triggered.\r\n## Observed Behaviour\r\n```console\r\n(.venv) ?master ~/prod/_inne/sqlfluff> echo \"select id from tbl;;\" | sqlfluff lint -\r\nTraceback (most recent call last):\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/bin/sqlfluff\", line 11, in <module>\r\n    load_entry_point('sqlfluff', 'console_scripts', 'sqlfluff')()\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/lib/python3.9/site-packages/click/core.py\", line 1137, in __call__\r\n    return self.main(*args, **kwargs)\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/lib/python3.9/site-packages/click/core.py\", line 1062, in main\r\n    rv = self.invoke(ctx)\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/lib/python3.9/site-packages/click/core.py\", line 1668, in invoke\r\n    return _process_result(sub_ctx.command.invoke(sub_ctx))\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/lib/python3.9/site-packages/click/core.py\", line 1404, in invoke\r\n    return ctx.invoke(self.callback, **ctx.params)\r\n  File \"/home/adam/prod/_inne/sqlfluff/.venv/lib/python3.9/site-packages/click/core.py\", line 763, in invoke\r\n    return __callback(*args, **kwargs)\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/cli/commands.py\", line 347, in lint\r\n    result = lnt.lint_string_wrapped(sys.stdin.read(), fname=\"stdin\")\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/linter/linter.py\", line 789, in lint_string_wrapped\r\n    linted_path.add(self.lint_string(string, fname=fname, fix=fix))\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/linter/linter.py\", line 668, in lint_string\r\n    parsed = self.parse_string(in_str=in_str, fname=fname, config=config)\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/linter/linter.py\", line 607, in parse_string\r\n    return self.parse_rendered(rendered, recurse=recurse)\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/linter/linter.py\", line 313, in parse_rendered\r\n    parsed, pvs = cls._parse_tokens(\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/linter/linter.py\", line 190, in _parse_tokens\r\n    parsed: Optional[BaseSegment] = parser.parse(\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/parser/parser.py\", line 32, in parse\r\n    parsed = root_segment.parse(parse_context=ctx)\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/parser/segments/base.py\", line 821, in parse\r\n    check_still_complete(segments, m.matched_segments, m.unmatched_segments)\r\n  File \"/home/adam/prod/_inne/sqlfluff/src/sqlfluff/core/parser/helpers.py\", line 30, in check_still_complete\r\n    raise RuntimeError(\r\nRuntimeError: Dropped elements in sequence matching! 'select id from tbl;;' != ';'\r\n\r\n```\r\n## Steps to Reproduce\r\nRun \r\n```console\r\necho \"select id from tbl;;\" | sqlfluff lint -\r\n```\r\n## Dialect\r\ndefault (ansi)\r\n## Version\r\n```\r\nsqlfluff, version 0.6.6\r\nPython 3.9.5\r\n```\r\n## Configuration\r\nNone\r\n\n", "hints_text": "Sounds similar to #1458 where we should handle \"empty\" statement/files better?\nNope, that's the different issue. I doubt that solving one of them would help in other one. I think both issues should stay, just in the case.\nBut what do you think @tunetheweb - should it just ignore these `;;` or raise something like `Found unparsable section:`? \nJust tested and in BigQuery it's an error.\r\nInterestingly Oracle is fine with it.\r\n\r\nI think it should be raised as `Found unparsable section`.", "created_at": "2021-10-06T07:57:35Z", "version": "0.6", "FAIL_TO_PASS": ["test/dialects/ansi_test.py::test__dialect__ansi_multiple_semicolons[select"], "PASS_TO_PASS": ["test/dialects/ansi_test.py::test__dialect__ansi__file_lex[a", "test/dialects/ansi_test.py::test__dialect__ansi__file_lex[b.c-res1]", "test/dialects/ansi_test.py::test__dialect__ansi__file_lex[abc", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectKeywordSegment-select]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[NakedIdentifierSegment-online_sales]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[BareFunctionSegment-current_timestamp]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[FunctionSegment-current_timestamp()]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[NumericLiteralSegment-1000.0]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-online_sales", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[IntervalExpressionSegment-INTERVAL", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-CASE", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-CAST(ROUND(online_sales", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-name", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-MIN", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-DATE_ADD(CURRENT_DATE('America/New_York'),", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-my_array[1]]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-my_array[OFFSET(1)]]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-my_array[5:8]]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-4", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-bits[OFFSET(0)]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-(count_18_24", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-count_18_24", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectStatementSegment-SELECT", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-t.val/t.id]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-CAST(num", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-a.*]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-a.b.*]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-a.b.c.*]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ObjectReferenceSegment-a..c.*]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment--some_variable]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment--", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-concat(left(uaid,", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-c", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-c", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[ExpressionSegment-NULL::INT]", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[SelectClauseElementSegment-NULL::INT", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_parses[TruncateStatementSegment-TRUNCATE", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_not_match[ObjectReferenceSegment-\\n", "test/dialects/ansi_test.py::test__dialect__ansi_specific_segment_not_parse[SELECT", "test/dialects/ansi_test.py::test__dialect__ansi_is_whitespace", "test/dialects/ansi_test.py::test__dialect__ansi_parse_indented_joins[select", "test/dialects/ansi_test.py::test__dialect__ansi_multiple_semicolons[;;-Line"], "environment_setup_commit": "67023b85c41d23d6c6d69812a41b207c4f8a9331"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-718", "base_commit": "3c84edd644fa4db54955e2225a183fa3e0405eb0", "patch": "diff --git a/pvlib/__init__.py b/pvlib/__init__.py\n--- a/pvlib/__init__.py\n+++ b/pvlib/__init__.py\n@@ -7,6 +7,7 @@\n from pvlib import location\n from pvlib import solarposition\n from pvlib import iotools\n+from pvlib import ivtools\n from pvlib import tracking\n from pvlib import pvsystem\n from pvlib import spa\ndiff --git a/pvlib/ivtools.py b/pvlib/ivtools.py\nnew file mode 100644\n--- /dev/null\n+++ b/pvlib/ivtools.py\n@@ -0,0 +1,350 @@\n+# -*- coding: utf-8 -*-\n+\"\"\"\n+Created on Fri Mar 29 10:34:10 2019\n+\n+@author: cwhanse\n+\"\"\"\n+\n+import numpy as np\n+\n+\n+def fit_sdm_cec_sam(celltype, v_mp, i_mp, v_oc, i_sc, alpha_sc, beta_voc,\n+                    gamma_pmp, cells_in_series, temp_ref=25):\n+    \"\"\"\n+    Estimates parameters for the CEC single diode model (SDM) using the SAM\n+    SDK.\n+\n+    Parameters\n+    ----------\n+    celltype : str\n+        Value is one of 'monoSi', 'multiSi', 'polySi', 'cis', 'cigs', 'cdte',\n+        'amorphous'\n+    v_mp : float\n+        Voltage at maximum power point [V]\n+    i_mp : float\n+        Current at maximum power point [A]\n+    v_oc : float\n+        Open circuit voltage [V]\n+    i_sc : float\n+        Short circuit current [A]\n+    alpha_sc : float\n+        Temperature coefficient of short circuit current [A/C]\n+    beta_voc : float\n+        Temperature coefficient of open circuit voltage [V/C]\n+    gamma_pmp : float\n+        Temperature coefficient of power at maximum point point [%/C]\n+    cells_in_series : int\n+        Number of cells in series\n+    temp_ref : float, default 25\n+        Reference temperature condition [C]\n+\n+    Returns\n+    -------\n+    tuple of the following elements:\n+\n+        * I_L_ref : float\n+            The light-generated current (or photocurrent) at reference\n+            conditions [A]\n+\n+        * I_o_ref : float\n+            The dark or diode reverse saturation current at reference\n+            conditions [A]\n+\n+        * R_sh_ref : float\n+            The shunt resistance at reference conditions, in ohms.\n+\n+        * R_s : float\n+            The series resistance at reference conditions, in ohms.\n+\n+        * a_ref : float\n+            The product of the usual diode ideality factor ``n`` (unitless),\n+            number of cells in series ``Ns``, and cell thermal voltage at\n+            reference conditions [V]\n+\n+        * Adjust : float\n+            The adjustment to the temperature coefficient for short circuit\n+            current, in percent.\n+\n+    Raises\n+    ------\n+        ImportError if NREL-PySAM is not installed.\n+\n+        RuntimeError if parameter extraction is not successful.\n+\n+    Notes\n+    -----\n+    Inputs ``v_mp``, ``v_oc``, ``i_mp`` and ``i_sc`` are assumed to be from a\n+    single IV curve at constant irradiance and cell temperature. Irradiance is\n+    not explicitly used by the fitting procedure. The irradiance level at which\n+    the input IV curve is determined and the specified cell temperature\n+    ``temp_ref`` are the reference conditions for the output parameters\n+    ``I_L_ref``, ``I_o_ref``, ``R_sh_ref``, ``R_s``, ``a_ref`` and ``Adjust``.\n+\n+    References\n+    ----------\n+    [1] A. Dobos, \"An Improved Coefficient Calculator for the California\n+    Energy Commission 6 Parameter Photovoltaic Module Model\", Journal of\n+    Solar Energy Engineering, vol 134, 2012.\n+    \"\"\"\n+\n+    try:\n+        from PySAM import PySSC\n+    except ImportError:\n+        raise ImportError(\"Requires NREL's PySAM package at \"\n+                          \"https://pypi.org/project/NREL-PySAM/.\")\n+\n+    datadict = {'tech_model': '6parsolve', 'financial_model': 'none',\n+                'celltype': celltype, 'Vmp': v_mp,\n+                'Imp': i_mp, 'Voc': v_oc, 'Isc': i_sc, 'alpha_isc': alpha_sc,\n+                'beta_voc': beta_voc, 'gamma_pmp': gamma_pmp,\n+                'Nser': cells_in_series, 'Tref': temp_ref}\n+\n+    result = PySSC.ssc_sim_from_dict(datadict)\n+    if result['cmod_success'] == 1:\n+        return tuple([result[k] for k in ['Il', 'Io', 'Rsh', 'Rs', 'a',\n+                      'Adj']])\n+    else:\n+        raise RuntimeError('Parameter estimation failed')\n+\n+\n+def fit_sde_sandia(voltage, current, v_oc=None, i_sc=None, v_mp_i_mp=None,\n+                   vlim=0.2, ilim=0.1):\n+    r\"\"\"\n+    Fits the single diode equation (SDE) to an IV curve.\n+\n+    Parameters\n+    ----------\n+    voltage : ndarray\n+        1D array of `float` type containing voltage at each point on the IV\n+        curve, increasing from 0 to ``v_oc`` inclusive [V]\n+\n+    current : ndarray\n+        1D array of `float` type containing current at each point on the IV\n+        curve, from ``i_sc`` to 0 inclusive [A]\n+\n+    v_oc : float, default None\n+        Open circuit voltage [V]. If not provided, ``v_oc`` is taken as the\n+        last point in the ``voltage`` array.\n+\n+    i_sc : float, default None\n+        Short circuit current [A]. If not provided, ``i_sc`` is taken as the\n+        first point in the ``current`` array.\n+\n+    v_mp_i_mp : tuple of float, default None\n+        Voltage, current at maximum power point in units of [V], [A].\n+        If not provided, the maximum power point is found at the maximum of\n+        ``voltage`` \\times ``current``.\n+\n+    vlim : float, default 0.2\n+        Defines portion of IV curve where the exponential term in the single\n+        diode equation can be neglected, i.e.\n+        ``voltage`` <= ``vlim`` x ``v_oc`` [V]\n+\n+    ilim : float, default 0.1\n+        Defines portion of the IV curve where the exponential term in the\n+        single diode equation is signficant, approximately defined by\n+        ``current`` < (1 - ``ilim``) x ``i_sc`` [A]\n+\n+    Returns\n+    -------\n+    tuple of the following elements:\n+\n+        * photocurrent : float\n+            photocurrent [A]\n+        * saturation_current : float\n+            dark (saturation) current [A]\n+        * resistance_shunt : float\n+            shunt (parallel) resistance, in ohms\n+        * resistance_series : float\n+            series resistance, in ohms\n+        * nNsVth : float\n+            product of thermal voltage ``Vth`` [V], diode ideality factor\n+            ``n``, and number of series cells ``Ns``\n+\n+    Raises\n+    ------\n+    RuntimeError if parameter extraction is not successful.\n+\n+    Notes\n+    -----\n+    Inputs ``voltage``, ``current``, ``v_oc``, ``i_sc`` and ``v_mp_i_mp`` are\n+    assumed to be from a single IV curve at constant irradiance and cell\n+    temperature.\n+\n+    :py:func:`fit_single_diode_sandia` obtains values for the five parameters\n+    for the single diode equation [1]:\n+\n+    .. math::\n+\n+        I = I_{L} - I_{0} (\\exp \\frac{V + I R_{s}}{nNsVth} - 1)\n+        - \\frac{V + I R_{s}}{R_{sh}}\n+\n+    See :py:func:`pvsystem.singlediode` for definition of the parameters.\n+\n+    The extraction method [2] proceeds in six steps.\n+\n+    1. In the single diode equation, replace :math:`R_{sh} = 1/G_{p}` and\n+       re-arrange\n+\n+    .. math::\n+\n+        I = \\frac{I_{L}}{1 + G_{p} R_{s}} - \\frac{G_{p} V}{1 + G_{p} R_{s}}\n+        - \\frac{I_{0}}{1 + G_{p} R_{s}} (\\exp(\\frac{V + I R_{s}}{nNsVth}) - 1)\n+\n+    2. The linear portion of the IV curve is defined as\n+       :math:`V \\le vlim \\times v_oc`. Over this portion of the IV curve,\n+\n+    .. math::\n+\n+        \\frac{I_{0}}{1 + G_{p} R_{s}} (\\exp(\\frac{V + I R_{s}}{nNsVth}) - 1)\n+        \\approx 0\n+\n+    3. Fit the linear portion of the IV curve with a line.\n+\n+    .. math::\n+\n+        I &\\approx \\frac{I_{L}}{1 + G_{p} R_{s}} - \\frac{G_{p} V}{1 + G_{p}\n+        R_{s}} \\\\\n+        &= \\beta_{0} + \\beta_{1} V\n+\n+    4. The exponential portion of the IV curve is defined by\n+       :math:`\\beta_{0} + \\beta_{1} \\times V - I > ilim \\times i_sc`.\n+       Over this portion of the curve, :math:`exp((V + IRs)/nNsVth) >> 1`\n+       so that\n+\n+    .. math::\n+\n+        \\exp(\\frac{V + I R_{s}}{nNsVth}) - 1 \\approx\n+        \\exp(\\frac{V + I R_{s}}{nNsVth})\n+\n+    5. Fit the exponential portion of the IV curve.\n+\n+    .. math::\n+\n+        \\log(\\beta_{0} - \\beta_{1} V - I)\n+        &\\approx \\log(\\frac{I_{0}}{1 + G_{p} R_{s}} + \\frac{V}{nNsVth}\n+        + \\frac{I R_{s}}{nNsVth} \\\\\n+        &= \\beta_{2} + beta_{3} V + \\beta_{4} I\n+\n+    6. Calculate values for ``IL, I0, Rs, Rsh,`` and ``nNsVth`` from the\n+       regression coefficents :math:`\\beta_{0}, \\beta_{1}, \\beta_{3}` and\n+       :math:`\\beta_{4}`.\n+\n+\n+    References\n+    ----------\n+    [1] S.R. Wenham, M.A. Green, M.E. Watt, \"Applied Photovoltaics\" ISBN\n+    0 86758 909 4\n+    [2] C. B. Jones, C. W. Hansen, Single Diode Parameter Extraction from\n+    In-Field Photovoltaic I-V Curves on a Single Board Computer, 46th IEEE\n+    Photovoltaic Specialist Conference, Chicago, IL, 2019\n+    \"\"\"\n+\n+    # If not provided, extract v_oc, i_sc, v_mp and i_mp from the IV curve data\n+    if v_oc is None:\n+        v_oc = voltage[-1]\n+    if i_sc is None:\n+        i_sc = current[0]\n+    if v_mp_i_mp is not None:\n+        v_mp, i_mp = v_mp_i_mp\n+    else:\n+        v_mp, i_mp = _find_mp(voltage, current)\n+\n+    # Find beta0 and beta1 from linear portion of the IV curve\n+    beta0, beta1 = _find_beta0_beta1(voltage, current, vlim, v_oc)\n+\n+    # Find beta3 and beta4 from the exponential portion of the IV curve\n+    beta3, beta4 = _find_beta3_beta4(voltage, current, beta0, beta1, ilim,\n+                                     i_sc)\n+\n+    # calculate single diode parameters from regression coefficients\n+    return _calculate_sde_parameters(beta0, beta1, beta3, beta4, v_mp, i_mp,\n+                                     v_oc)\n+\n+\n+def _find_mp(voltage, current):\n+    \"\"\"\n+    Finds voltage and current at maximum power point.\n+\n+    Parameters\n+    ----------\n+    voltage : ndarray\n+        1D array containing voltage at each point on the IV curve, increasing\n+        from 0 to v_oc inclusive, of `float` type [V]\n+\n+    current : ndarray\n+        1D array containing current at each point on the IV curve, decreasing\n+        from i_sc to 0 inclusive, of `float` type [A]\n+\n+    Returns\n+    -------\n+    v_mp, i_mp : tuple\n+        voltage ``v_mp`` and current ``i_mp`` at the maximum power point [V],\n+        [A]\n+    \"\"\"\n+    p = voltage * current\n+    idx = np.argmax(p)\n+    return voltage[idx], current[idx]\n+\n+\n+def _calc_I0(IL, I, V, Gp, Rs, nNsVth):\n+    return (IL - I - Gp * V - Gp * Rs * I) / np.exp((V + Rs * I) / nNsVth)\n+\n+\n+def _find_beta0_beta1(v, i, vlim, v_oc):\n+    # Get intercept and slope of linear portion of IV curve.\n+    # Start with V =< vlim * v_oc, extend by adding points until slope is\n+    # negative (downward).\n+    beta0 = np.nan\n+    beta1 = np.nan\n+    first_idx = np.searchsorted(v, vlim * v_oc)\n+    for idx in range(first_idx, len(v)):\n+        coef = np.polyfit(v[:idx], i[:idx], deg=1)\n+        if coef[0] < 0:\n+            # intercept term\n+            beta0 = coef[1].item()\n+            # sign change of slope to get positive parameter value\n+            beta1 = -coef[0].item()\n+            break\n+    if any(np.isnan([beta0, beta1])):\n+        raise RuntimeError(\"Parameter extraction failed: beta0={}, beta1={}\"\n+                           .format(beta0, beta1))\n+    else:\n+        return beta0, beta1\n+\n+\n+def _find_beta3_beta4(voltage, current, beta0, beta1, ilim, i_sc):\n+    # Subtract the IV curve from the linear fit.\n+    y = beta0 - beta1 * voltage - current\n+    x = np.array([np.ones_like(voltage), voltage, current]).T\n+    # Select points where y > ilim * i_sc to regress log(y) onto x\n+    idx = (y > ilim * i_sc)\n+    result = np.linalg.lstsq(x[idx], np.log(y[idx]), rcond=None)\n+    coef = result[0]\n+    beta3 = coef[1].item()\n+    beta4 = coef[2].item()\n+    if any(np.isnan([beta3, beta4])):\n+        raise RuntimeError(\"Parameter extraction failed: beta3={}, beta4={}\"\n+                           .format(beta3, beta4))\n+    else:\n+        return beta3, beta4\n+\n+\n+def _calculate_sde_parameters(beta0, beta1, beta3, beta4, v_mp, i_mp, v_oc):\n+    nNsVth = 1.0 / beta3\n+    Rs = beta4 / beta3\n+    Gp = beta1 / (1.0 - Rs * beta1)\n+    Rsh = 1.0 / Gp\n+    IL = (1 + Gp * Rs) * beta0\n+    # calculate I0\n+    I0_vmp = _calc_I0(IL, i_mp, v_mp, Gp, Rs, nNsVth)\n+    I0_voc = _calc_I0(IL, 0, v_oc, Gp, Rs, nNsVth)\n+    if any(np.isnan([I0_vmp, I0_voc])) or ((I0_vmp <= 0) and (I0_voc <= 0)):\n+        raise RuntimeError(\"Parameter extraction failed: I0 is undetermined.\")\n+    elif (I0_vmp > 0) and (I0_voc > 0):\n+        I0 = 0.5 * (I0_vmp + I0_voc)\n+    elif (I0_vmp > 0):\n+        I0 = I0_vmp\n+    else:  # I0_voc > 0\n+        I0 = I0_voc\n+    return (IL, I0, Rsh, Rs, nNsVth)\ndiff --git a/setup.py b/setup.py\n--- a/setup.py\n+++ b/setup.py\n@@ -44,8 +44,8 @@\n TESTS_REQUIRE = ['nose', 'pytest', 'pytest-cov', 'pytest-mock',\n                  'pytest-timeout']\n EXTRAS_REQUIRE = {\n-    'optional': ['ephem', 'cython', 'netcdf4', 'numba', 'pvfactors', 'scipy',\n-                 'siphon', 'tables'],\n+    'optional': ['ephem', 'cython', 'netcdf4', 'nrel-pysam', 'numba',\n+                 'pvfactors', 'scipy', 'siphon', 'tables'],\n     'doc': ['ipython', 'matplotlib', 'sphinx', 'sphinx_rtd_theme'],\n     'test': TESTS_REQUIRE\n }\n", "test_patch": "diff --git a/pvlib/test/conftest.py b/pvlib/test/conftest.py\n--- a/pvlib/test/conftest.py\n+++ b/pvlib/test/conftest.py\n@@ -151,6 +151,15 @@ def has_numba():\n                                         reason='requires pvfactors')\n \n \n+try:\n+    import PySAM  # noqa: F401\n+    has_pysam = True\n+except ImportError:\n+    has_pysam = False\n+\n+requires_pysam = pytest.mark.skipif(not has_pysam, reason=\"requires PySAM\")\n+\n+\n @pytest.fixture(scope=\"session\")\n def sam_data():\n     data = {}\ndiff --git a/pvlib/test/test_ivtools.py b/pvlib/test/test_ivtools.py\nnew file mode 100644\n--- /dev/null\n+++ b/pvlib/test/test_ivtools.py\n@@ -0,0 +1,242 @@\n+# -*- coding: utf-8 -*-\n+\"\"\"\n+Created on Thu May  9 10:51:15 2019\n+\n+@author: cwhanse\n+\"\"\"\n+\n+import numpy as np\n+import pandas as pd\n+import pytest\n+from pvlib import pvsystem\n+from pvlib import ivtools\n+from pvlib.test.conftest import requires_scipy, requires_pysam\n+\n+\n+@pytest.fixture\n+def get_test_iv_params():\n+    return {'IL': 8.0, 'I0': 5e-10, 'Rsh': 1000, 'Rs': 0.2, 'nNsVth': 1.61864}\n+\n+\n+@pytest.fixture\n+def get_cec_params_cansol_cs5p_220p():\n+    return {'input': {'V_mp_ref': 46.6, 'I_mp_ref': 4.73, 'V_oc_ref': 58.3,\n+                      'I_sc_ref': 5.05, 'alpha_sc': 0.0025,\n+                      'beta_voc': -0.19659, 'gamma_pmp': -0.43,\n+                      'cells_in_series': 96},\n+            'output': {'a_ref': 2.3674, 'I_L_ref': 5.056, 'I_o_ref': 1.01e-10,\n+                       'R_sh_ref': 837.51, 'R_s': 1.004, 'Adjust': 2.3}}\n+\n+\n+@requires_scipy\n+def test_fit_sde_sandia(get_test_iv_params, get_bad_iv_curves):\n+    test_params = get_test_iv_params\n+    testcurve = pvsystem.singlediode(photocurrent=test_params['IL'],\n+                                     saturation_current=test_params['I0'],\n+                                     resistance_shunt=test_params['Rsh'],\n+                                     resistance_series=test_params['Rs'],\n+                                     nNsVth=test_params['nNsVth'],\n+                                     ivcurve_pnts=300)\n+    expected = tuple(test_params[k] for k in ['IL', 'I0', 'Rsh', 'Rs',\n+                     'nNsVth'])\n+    result = ivtools.fit_sde_sandia(voltage=testcurve['v'],\n+                                    current=testcurve['i'])\n+    assert np.allclose(result, expected, rtol=5e-5)\n+    result = ivtools.fit_sde_sandia(voltage=testcurve['v'],\n+                                    current=testcurve['i'],\n+                                    v_oc=testcurve['v_oc'],\n+                                    i_sc=testcurve['i_sc'])\n+    assert np.allclose(result, expected, rtol=5e-5)\n+    result = ivtools.fit_sde_sandia(voltage=testcurve['v'],\n+                                    current=testcurve['i'],\n+                                    v_oc=testcurve['v_oc'],\n+                                    i_sc=testcurve['i_sc'],\n+                                    v_mp_i_mp=(testcurve['v_mp'],\n+                                    testcurve['i_mp']))\n+    assert np.allclose(result, expected, rtol=5e-5)\n+    result = ivtools.fit_sde_sandia(voltage=testcurve['v'],\n+                                    current=testcurve['i'], vlim=0.1)\n+    assert np.allclose(result, expected, rtol=5e-5)\n+\n+\n+@requires_scipy\n+def test_fit_sde_sandia_bad_iv(get_bad_iv_curves):\n+    # bad IV curves for coverage of if/then in _calculate_sde_parameters\n+    v1, i1, v2, i2 = get_bad_iv_curves\n+    result = ivtools.fit_sde_sandia(voltage=v1, current=i1)\n+    assert np.allclose(result, (-2.4322856072799985, 8.854688976836396,\n+                                -63.56227601452038, 111.18558915546389,\n+                                -137.9965046659527))\n+    result = ivtools.fit_sde_sandia(voltage=v2, current=i2)\n+    assert np.allclose(result, (2.62405311949227, 1.8657963912925288,\n+                                110.35202827739991, -65.652554411442,\n+                                174.49362093001415))\n+\n+\n+@requires_pysam\n+def test_fit_sdm_cec_sam(get_cec_params_cansol_cs5p_220p):\n+    input_data = get_cec_params_cansol_cs5p_220p['input']\n+    I_L_ref, I_o_ref, R_sh_ref, R_s, a_ref, Adjust = \\\n+        ivtools.fit_sdm_cec_sam(\n+            celltype='polySi', v_mp=input_data['V_mp_ref'],\n+            i_mp=input_data['I_mp_ref'], v_oc=input_data['V_oc_ref'],\n+            i_sc=input_data['I_sc_ref'], alpha_sc=input_data['alpha_sc'],\n+            beta_voc=input_data['beta_voc'],\n+            gamma_pmp=input_data['gamma_pmp'],\n+            cells_in_series=input_data['cells_in_series'])\n+    expected = pd.Series(get_cec_params_cansol_cs5p_220p['output'])\n+    modeled = pd.Series(index=expected.index, data=np.nan)\n+    modeled['a_ref'] = a_ref\n+    modeled['I_L_ref'] = I_L_ref\n+    modeled['I_o_ref'] = I_o_ref\n+    modeled['R_sh_ref'] = R_sh_ref\n+    modeled['R_s'] = R_s\n+    modeled['Adjust'] = Adjust\n+    assert np.allclose(modeled.values, expected.values, rtol=5e-2)\n+    # test for fitting failure\n+    with pytest.raises(RuntimeError):\n+        I_L_ref, I_o_ref, R_sh_ref, R_s, a_ref, Adjust = \\\n+            ivtools.fit_sdm_cec_sam(\n+                celltype='polySi', v_mp=0.45, i_mp=5.25, v_oc=0.55, i_sc=5.5,\n+                alpha_sc=0.00275, beta_voc=0.00275, gamma_pmp=0.0055,\n+                cells_in_series=1, temp_ref=25)\n+\n+\n+@pytest.fixture\n+def get_bad_iv_curves():\n+    # v1, i1 produces a bad value for I0_voc\n+    v1 = np.array([0, 0.338798867469060, 0.677597734938121, 1.01639660240718,\n+                   1.35519546987624, 1.69399433734530, 2.03279320481436,\n+                   2.37159207228342, 2.71039093975248, 3.04918980722154,\n+                   3.38798867469060, 3.72678754215966, 4.06558640962873,\n+                   4.40438527709779, 4.74318414456685, 5.08198301203591,\n+                   5.42078187950497, 5.75958074697403, 6.09837961444309,\n+                   6.43717848191215, 6.77597734938121, 7.11477621685027,\n+                   7.45357508431933, 7.79237395178839, 8.13117281925745,\n+                   8.46997168672651, 8.80877055419557, 9.14756942166463,\n+                   9.48636828913369, 9.82516715660275, 10.1639660240718,\n+                   10.5027648915409, 10.8415637590099, 11.1803626264790,\n+                   11.5191614939481, 11.8579603614171, 12.1967592288862,\n+                   12.5355580963552, 12.8743569638243, 13.2131558312934,\n+                   13.5519546987624, 13.8907535662315, 14.2295524337005,\n+                   14.5683513011696, 14.9071501686387, 15.2459490361077,\n+                   15.5847479035768, 15.9235467710458, 16.2623456385149,\n+                   16.6011445059840, 16.9399433734530, 17.2787422409221,\n+                   17.6175411083911, 17.9563399758602, 18.2951388433293,\n+                   18.6339377107983, 18.9727365782674, 19.3115354457364,\n+                   19.6503343132055, 19.9891331806746, 20.3279320481436,\n+                   20.6667309156127, 21.0055297830817, 21.3443286505508,\n+                   21.6831275180199, 22.0219263854889, 22.3607252529580,\n+                   22.6995241204270, 23.0383229878961, 23.3771218553652,\n+                   23.7159207228342, 24.0547195903033, 24.3935184577724,\n+                   24.7323173252414, 25.0711161927105, 25.4099150601795,\n+                   25.7487139276486, 26.0875127951177, 26.4263116625867,\n+                   26.7651105300558, 27.1039093975248, 27.4427082649939,\n+                   27.7815071324630, 28.1203059999320, 28.4591048674011,\n+                   28.7979037348701, 29.1367026023392, 29.4755014698083,\n+                   29.8143003372773, 30.1530992047464, 30.4918980722154,\n+                   30.8306969396845, 31.1694958071536, 31.5082946746226,\n+                   31.8470935420917, 32.1858924095607, 32.5246912770298,\n+                   32.8634901444989, 33.2022890119679, 33.5410878794370])\n+    i1 = np.array([3.39430882774470, 2.80864492110761, 3.28358165429196,\n+                   3.41191190551673, 3.11975662808148, 3.35436585834612,\n+                   3.23953272899809, 3.60307083325333, 2.80478101508277,\n+                   2.80505102853845, 3.16918996870373, 3.21088388439857,\n+                   3.46332865310431, 3.09224155015883, 3.17541550741062,\n+                   3.32470179290389, 3.33224664316240, 3.07709000050741,\n+                   2.89141245343405, 3.01365768561537, 3.23265176770231,\n+                   3.32253647634228, 2.97900657569736, 3.31959549243966,\n+                   3.03375461550111, 2.97579298978937, 3.25432831375159,\n+                   2.89178382564454, 3.00341909207567, 3.72637492250097,\n+                   3.28379856976360, 2.96516169245835, 3.25658381110230,\n+                   3.41655911533139, 3.02718097944604, 3.11458376760376,\n+                   3.24617304369762, 3.45935502367636, 3.21557333256913,\n+                   3.27611176482650, 2.86954135732485, 3.32416319254657,\n+                   3.15277467598732, 3.08272557013770, 3.15602202666259,\n+                   3.49432799877150, 3.53863997177632, 3.10602611478455,\n+                   3.05373911151821, 3.09876772570781, 2.97417228624287,\n+                   2.84573593699237, 3.16288578405195, 3.06533173612783,\n+                   3.02118336639575, 3.34374977225502, 2.97255164138821,\n+                   3.19286135682863, 3.10999753817133, 3.26925354620079,\n+                   3.11957809501529, 3.20155017481720, 3.31724984405837,\n+                   3.42879043512927, 3.17933067619240, 3.47777362613969,\n+                   3.20708912539777, 3.48205761174907, 3.16804363684327,\n+                   3.14055472378230, 3.13445657434470, 2.91152696252998,\n+                   3.10984113847427, 2.80443349399489, 3.23146278164875,\n+                   2.94521083406108, 3.17388903141715, 3.05930294897030,\n+                   3.18985234673287, 3.27946609274898, 3.33717523113602,\n+                   2.76394303462702, 3.19375132937510, 2.82628616689450,\n+                   2.85238527394143, 2.82975892599489, 2.79196912313914,\n+                   2.72860792049395, 2.75585977414140, 2.44280222448805,\n+                   2.36052347370628, 2.26785071765738, 2.10868255743462,\n+                   2.06165739407987, 1.90047259509385, 1.39925575828709,\n+                   1.24749015957606, 0.867823806536762, 0.432752457749993, 0])\n+    # v2, i2 produces a bad value for I0_vmp\n+    v2 = np.array([0, 0.365686097622586, 0.731372195245173, 1.09705829286776,\n+                   1.46274439049035, 1.82843048811293, 2.19411658573552,\n+                   2.55980268335810, 2.92548878098069, 3.29117487860328,\n+                   3.65686097622586, 4.02254707384845, 4.38823317147104,\n+                   4.75391926909362, 5.11960536671621, 5.48529146433880,\n+                   5.85097756196138, 6.21666365958397, 6.58234975720655,\n+                   6.94803585482914, 7.31372195245173, 7.67940805007431,\n+                   8.04509414769690, 8.41078024531949, 8.77646634294207,\n+                   9.14215244056466, 9.50783853818725, 9.87352463580983,\n+                   10.2392107334324, 10.6048968310550, 10.9705829286776,\n+                   11.3362690263002, 11.7019551239228, 12.0676412215454,\n+                   12.4333273191679, 12.7990134167905, 13.1646995144131,\n+                   13.5303856120357, 13.8960717096583, 14.2617578072809,\n+                   14.6274439049035, 14.9931300025260, 15.3588161001486,\n+                   15.7245021977712, 16.0901882953938, 16.4558743930164,\n+                   16.8215604906390, 17.1872465882616, 17.5529326858841,\n+                   17.9186187835067, 18.2843048811293, 18.6499909787519,\n+                   19.0156770763745, 19.3813631739971, 19.7470492716197,\n+                   20.1127353692422, 20.4784214668648, 20.8441075644874,\n+                   21.2097936621100, 21.5754797597326, 21.9411658573552,\n+                   22.3068519549778, 22.6725380526004, 23.0382241502229,\n+                   23.4039102478455, 23.7695963454681, 24.1352824430907,\n+                   24.5009685407133, 24.8666546383359, 25.2323407359585,\n+                   25.5980268335810, 25.9637129312036, 26.3293990288262,\n+                   26.6950851264488, 27.0607712240714, 27.4264573216940,\n+                   27.7921434193166, 28.1578295169392, 28.5235156145617,\n+                   28.8892017121843, 29.2548878098069, 29.6205739074295,\n+                   29.9862600050521, 30.3519461026747, 30.7176322002973,\n+                   31.0833182979198, 31.4490043955424, 31.8146904931650,\n+                   32.1803765907876, 32.5460626884102, 32.9117487860328,\n+                   33.2774348836554, 33.6431209812779, 34.0088070789005,\n+                   34.3744931765231, 34.7401792741457, 35.1058653717683,\n+                   35.4715514693909, 35.8372375670135, 36.2029236646360])\n+    i2 = np.array([6.49218806928330, 6.49139336899548, 6.17810697175204,\n+                   6.75197816263663, 6.59529074137515, 6.18164578868300,\n+                   6.38709397931910, 6.30685422248427, 6.44640615548925,\n+                   6.88727230397772, 6.42074852785591, 6.46348580823746,\n+                   6.38642309763941, 5.66356277572311, 6.61010381702082,\n+                   6.33288284311125, 6.22475343933610, 6.30651399433833,\n+                   6.44435022944051, 6.43741711131908, 6.03536180208946,\n+                   6.23814639328170, 5.97229140403242, 6.20790000748341,\n+                   6.22933550182341, 6.22992127804882, 6.13400871899299,\n+                   6.83491312449950, 6.07952797245846, 6.35837746415450,\n+                   6.41972128662324, 6.85256717258275, 6.25807797296759,\n+                   6.25124948151766, 6.22229212812413, 6.72249444167406,\n+                   6.41085549981649, 6.75792874870056, 6.22096181559171,\n+                   6.47839564388996, 6.56010208597432, 6.63300966556949,\n+                   6.34617546039339, 6.79812221146153, 6.14486056194136,\n+                   6.14979256889311, 6.16883037644880, 6.57309183229605,\n+                   6.40064681038509, 6.18861448239873, 6.91340138179698,\n+                   5.94164388433788, 6.23638991745862, 6.31898940411710,\n+                   6.45247884556830, 6.58081455524297, 6.64915284801713,\n+                   6.07122119270245, 6.41398258148256, 6.62144271089614,\n+                   6.36377197712687, 6.51487678829345, 6.53418950147730,\n+                   6.18886469125371, 6.26341063475750, 6.83488211680259,\n+                   6.62699397226695, 6.41286837534735, 6.44060085001851,\n+                   6.48114130629288, 6.18607038456406, 6.16923370572396,\n+                   6.64223126283631, 6.07231852289266, 5.79043710204375,\n+                   6.48463886529882, 6.36263392044401, 6.11212476454494,\n+                   6.14573900812925, 6.12568047243240, 6.43836230231577,\n+                   6.02505694060219, 6.13819468942244, 6.22100593815064,\n+                   6.02394682666345, 5.89016573063789, 5.74448527739202,\n+                   5.50415294280017, 5.31883018164157, 4.87476769510305,\n+                   4.74386713755523, 4.60638346931628, 4.06177345572680,\n+                   3.73334482123538, 3.13848311672243, 2.71638862600768,\n+                   2.02963773590165, 1.49291145092070, 0.818343889647352, 0])\n+\n+    return v1, i1, v2, i2\n", "problem_statement": "CEC 6-parameter coefficient generation\nSAM is able to extract the CEC parameters required for calcparams_desoto.  This is done through the 'CEC Performance Model with User Entered Specifications' module model, and coefficients are automatically extracted given nameplate parameters Voc, Isc, Imp, Vmp and TempCoeff.  The method is based on Aron Dobos' \"An Improved Coefficient Calculator for the California Energy Commission 6 Parameter Photovoltaic Module Model \", 2012\r\n\r\nIdeally we should be able to work with the SAM open source code, extract the bit that does the coefficient generation, and put it into a PVLib function that would allow users to run calcparams_desoto with any arbitrary module type.  At the moment we are dependent on PV modules loaded into the SAM or CEC database.\r\n\r\nThank you!\r\n\n", "hints_text": "SAM solution routine is located at https://github.com/NREL/ssc/blob/develop/shared/6par_solve.h , function \"solve_with_sanity_and_heuristics\" . Additional dependencies on other files 6par_newton.h, 6par_search.h, 6par_jacobian.h, 6par_lu.h, 6par_gamma.h, all located in https://github.com/NREL/ssc/tree/develop/shared . \nI'd like to try to take this up sometime soon, maybe by the end of the year? Is anyone else working on it? I haven't seen a PR. Does anyone have any strong opinions? Is seems like there's a SAM implementation, is there a PVLIB-MATLAB version already? thx\nI'm not working on it. If it is practical, it would be great to wrap the SAM method. NREL put a lot of effort into the heuristics for initial values and updating.  We have a function in PVLib for MATLAB but it's a very different algorithm. For consistency with the parameter databases it would be better to use the SAM algorithm.\nDo I read between the lines that the two algorithms do not converge on the same parameters?  Then it would be of great interest to have python implementations of each!  Or perhaps even more important: documents describing the full algorithms along with every heuristic, assumption and constraint they implement...\n@adriesse yes. Each model fitting algorithm I'm aware of, will produce different parameter values from the same data although the predicted IV curves may be similar.\nFWIW I am trying to establish a common/fair metric to compare different model fits in PVfit. In addition to choosing, say, which residual(s) to analyze, the effect of model discrepancy significantly affects such comparisons. PVfit\u2019s orthogonal distance regression (ODR) fits to the single-diode model (SDM) often are worse than other algorithms\u2019 fits in terms of terminal current residuals, but I have observed consistently that this is likely due to model discrepancy. One might claim that ODR is being more \u201chonest\u201d here, but I think it really means that the model should be improved. Note that PVfit\u2019s ODR alternatively optimizes the residual of the sum of currents at the high-voltage diode node in the equivalent-circuit model, and it doesn\u2019t make unfounded assumptions about error-free measurements in all data channels except the terminal current (which also has implications for parameter non-identifiability). I have seen plenty of evidence that ODR fits are \u201cgood\u201d in the absence of significant model discrepancy, such as for some (but not all!) double-diode model (DDM) fits across various material systems. \n> If it is practical, it would be great to wrap the SAM method. \r\n\r\nTwo kinds of practicalities here:\r\n\r\n1. Technical. I don't know how to do it but I am sure it's possible. I am ok with it so long as pvlib remains straightforward to install from source.\r\n2. Legal. We would need assurance from NREL that it's ok for pvlib to distribute this SAM code under the terms of a MIT license rather than SAM's mixed MIT/GPL license.\r\n\nIn the next 6-8 months, the NREL team will be updating our python wrapper for the SAM software development kit so that you can pip install the sam-sdk and call its routines in a much more native pythonic fashion. That might be a quick way to implement the CEC parameter generation to solve both the technical and legal practicalities Will mentions above, as well as the many file dependencies involved with that routine. The SDK is licensed separately from the SAM open source code under an MIT type license, so that distribution with pvlib would be ok, and once we have it set up as a python package, that should make it a non-issue to include with pvlib installation from source. \r\n\r\nTo @adriesse 's point, the publication associated with the method implemented in SAM is here: http://solarenergyengineering.asmedigitalcollection.asme.org/article.aspx?articleid=1458865 \nThanks @janinefreeman that is great news!\n@janinefreeman Does NREL's web site no longer provide free preprints of Aron's paper?\n@thunderfish24  I'm not finding a copy of it in NREL's pubs database, but it looks like you may be able to get it here (didn't create an account to give it a try): https://www.osti.gov/biblio/1043759 \n@thunderfish24 I sent you a preprint.", "created_at": "2019-05-13T14:43:11Z", "version": "0.5", "FAIL_TO_PASS": ["pvlib/test/test_ivtools.py::test_fit_sde_sandia", "pvlib/test/test_ivtools.py::test_fit_sde_sandia_bad_iv"], "PASS_TO_PASS": [], "environment_setup_commit": "84818c6d950142927359ffe308c728a0c080ddce"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-816", "base_commit": "3551f5b5a5f8d4de3ed92e5e479ac8c74a8c893a", "patch": "diff --git a/pydicom/charset.py b/pydicom/charset.py\n--- a/pydicom/charset.py\n+++ b/pydicom/charset.py\n@@ -125,15 +125,24 @@ def decode_string(value, encodings, delimiters):\n     \"\"\"\n     # shortcut for the common case - no escape sequences present\n     if ESC not in value:\n+        first_encoding = encodings[0]\n         try:\n-            return value.decode(encodings[0])\n+            return value.decode(first_encoding)\n+        except LookupError:\n+            if config.enforce_valid_values:\n+                raise\n+            warnings.warn(u\"Unknown encoding '{}' - \"\n+                          u\"using default encoding instead\"\n+                          .format(first_encoding))\n+            first_encoding = default_encoding\n+            return value.decode(first_encoding)\n         except UnicodeError:\n             if config.enforce_valid_values:\n                 raise\n-            warnings.warn(u\"Failed to decode byte string with encoding {} - \"\n+            warnings.warn(u\"Failed to decode byte string with encoding '{}' - \"\n                           u\"using replacement characters in decoded \"\n-                          u\"string\".format(encodings[0]))\n-            return value.decode(encodings[0], errors='replace')\n+                          u\"string\".format(first_encoding))\n+            return value.decode(first_encoding, errors='replace')\n \n     # Each part of the value that starts with an escape sequence is decoded\n     # separately. If it starts with an escape sequence, the\n", "test_patch": "diff --git a/pydicom/tests/test_charset.py b/pydicom/tests/test_charset.py\n--- a/pydicom/tests/test_charset.py\n+++ b/pydicom/tests/test_charset.py\n@@ -100,6 +100,26 @@ def test_standard_file(self):\n         ds.decode()\n         assert u'CompressedSamples^CT1' == ds.PatientName\n \n+    def test_invalid_character_set(self):\n+        \"\"\"charset: replace invalid encoding with default encoding\"\"\"\n+        ds = dcmread(get_testdata_files(\"CT_small.dcm\")[0])\n+        ds.read_encoding = None\n+        ds.SpecificCharacterSet = 'Unsupported'\n+        with pytest.warns(UserWarning,\n+                          match=u\"Unknown encoding 'Unsupported' \"\n+                                u\"- using default encoding instead\"):\n+            ds.decode()\n+            assert u'CompressedSamples^CT1' == ds.PatientName\n+\n+    def test_invalid_character_set_enforce_valid(self):\n+        \"\"\"charset: raise on invalid encoding\"\"\"\n+        config.enforce_valid_values = True\n+        ds = dcmread(get_testdata_files(\"CT_small.dcm\")[0])\n+        ds.read_encoding = None\n+        ds.SpecificCharacterSet = 'Unsupported'\n+        with pytest.raises(LookupError, match='unknown encoding: Unsupported'):\n+            ds.decode()\n+\n     def test_decoding_with_specific_tags(self):\n         \"\"\"Decoding is correctly applied even if  Specific Character Set\n         is not in specific tags...\"\"\"\n@@ -126,8 +146,8 @@ def test_bad_encoded_single_encoding(self):\n         elem = DataElement(0x00100010, 'PN',\n                            b'\\xc4\\xe9\\xef\\xed\\xf5\\xf3\\xe9\\xef\\xf2')\n \n-        with pytest.warns(UserWarning, match='Failed to decode byte string '\n-                                             'with encoding UTF8'):\n+        with pytest.warns(UserWarning, match=\"Failed to decode byte string \"\n+                                             \"with encoding 'UTF8'\"):\n             pydicom.charset.decode(elem, ['ISO_IR 192'])\n             assert u'\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd' == elem.value\n \n@@ -235,9 +255,11 @@ def test_patched_charset(self):\n             # make sure no warning is issued for the correct value\n             assert 1 == len(w)\n \n-        # not patched incorrect encoding raises\n+        # not patched incorrect encoding is replaced by default encoding\n         elem = DataElement(0x00100010, 'PN', b'Buc^J\\xc3\\xa9r\\xc3\\xb4me')\n-        with pytest.raises(LookupError):\n+        with pytest.warns(UserWarning,\n+                          match=u\"Unknown encoding 'ISOIR 192' - \"\n+                                u\"using default encoding instead\"):\n             pydicom.charset.decode(elem, ['ISOIR 192'])\n \n         # Python encoding also can be used directly\n", "problem_statement": "LookupError: unknown encoding: Not Supplied\n#### Description\r\nOutput from `ds = pydicom.read_file(dcmFile)` (an RTSTRUCT dicom file, SOP UID 1.2.840.10008.5.1.4.1.1.481.3) results in some tags throwing a LookupError: \"LookupError: unknown encoding: Not Supplied\"\r\nSpecific tags which cannot be decoded are as follows:\r\n['DeviceSerialNumber',\r\n 'Manufacturer',\r\n 'ManufacturerModelName',\r\n 'PatientID',\r\n 'PatientName',\r\n 'RTROIObservationsSequence',\r\n 'ReferringPhysicianName',\r\n 'SeriesDescription',\r\n 'SoftwareVersions',\r\n 'StructureSetLabel',\r\n 'StructureSetName',\r\n 'StructureSetROISequence',\r\n 'StudyDescription',\r\n 'StudyID']\r\n\r\nI suspect that it's due to the fact that `ds.SpecificCharacterSet = 'Not Supplied'`, but when I try to set `ds.SpecificCharacterSet` to something reasonable (ie ISO_IR_100 or 'iso8859'), it doesn't seem to make any difference.\r\n\r\nReading the same file, with NO modifications, in gdcm does not result in any errors and all fields are readable.\r\n\r\n#### Steps/Code to Reproduce\r\n```py\r\nimport pydicom \r\nds = pydicom.read_file(dcmFile)\r\nprint(ds.PatientName)\r\n```\r\n\r\n#### Expected Results\r\nNo error is thrown and the name of the patient is printed.\r\n\r\n#### Actual Results\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"C:\\Users\\Amanda\\AppData\\Local\\Continuum\\anaconda3\\envs\\itk\\lib\\site-packages\\pydicom\\valuerep.py\", line 706, in __str__\r\n    return '='.join(self.components).__str__()\r\n  File \"C:\\Users\\Amanda\\AppData\\Local\\Continuum\\anaconda3\\envs\\itk\\lib\\site-packages\\pydicom\\valuerep.py\", line 641, in components\r\n    self._components = _decode_personname(groups, self.encodings)\r\n  File \"C:\\Users\\Amanda\\AppData\\Local\\Continuum\\anaconda3\\envs\\itk\\lib\\site-packages\\pydicom\\valuerep.py\", line 564, in _decode_personname\r\n    for comp in components]\r\n  File \"C:\\Users\\Amanda\\AppData\\Local\\Continuum\\anaconda3\\envs\\itk\\lib\\site-packages\\pydicom\\valuerep.py\", line 564, in <listcomp>\r\n    for comp in components]\r\n  File \"C:\\Users\\Amanda\\AppData\\Local\\Continuum\\anaconda3\\envs\\itk\\lib\\site-packages\\pydicom\\charset.py\", line 129, in decode_string\r\n    return value.decode(encodings[0])\r\nLookupError: unknown encoding: Not Supplied\r\n\r\n#### Versions\r\nPlatform: Windows-10-10.0.17763-SP0\r\nPython Version: Python 3.6.4 |Anaconda, Inc.| (default, Mar 12 2018, 20:20:50) [MSC v.1900 64 bit (AMD64)]\r\npydicom Version: pydicom 1.2.2\r\n\n", "hints_text": "You said on the pynetdicom issue you can't upload an anonymised file, but can you open the file in a hex editor and post the raw byte output from the first few (non-identifying) elements? From the start of the file to the end of say (0008,0070) should be enough.\r\n\r\nAlternatively you could truncate the file at the end of the (0008,0070) element and upload that.\r\n\r\nIf you need to know how to interpret the encoded data check out [Part 5, Chapter 7](http://dicom.nema.org/medical/dicom/current/output/chtml/part05/chapter_7.html) of the DICOM Standard. And if the file's been saved in the [DICOM File Format](http://dicom.nema.org/medical/dicom/current/output/chtml/part10/chapter_7.html) there may also be a 128 byte header  following by 'DICM' before the start of the dataset (which we don't need).\n> open the file in a hex editor and post the raw byte output from the first few (non-identifying) elements\r\n\r\nAlternatively:\r\n```\r\nimport pydicom.config\r\npydicom.config.debug(True)\r\nds = dcmread(youfilename)\r\n```\r\nAnd as suggested copy the first non-identifying part of the debug output for posting.\nPlease find the truncated dataset attached as requested. I wasn't allowed to upload a *.dcm file so just wrote it as a *.txt file. The file is readable by pydicom but exhibits the same aforementioned problems, where the \"LookupError: unknown encoding: Not Supplied\" happens only for the Manufacturer tag.\r\n\r\n[truncated.txt](https://github.com/pydicom/pydicom/files/2947883/truncated.txt)\r\n\nOkay, so I've tried deleting SpecificCharacterSet and the error still occurs.  I think pydicom is still holding on to the original values, and needs some code to handle the case when SpecificCharacterSet is deleted or set again after reading the file. We can dig into it a little further.\r\n\r\n\nWorkaround:\r\n```python\r\nfrom pydicom import dcmread\r\n\r\nds = dcmread('path/to/file')\r\ndel ds.SpecificCharacterSet\r\nds.read_encoding = []\r\n```\r\n\nBrilliant!  That worked! \r\n\r\nThank you for the quick fix! I've found pydicom is a lot more user-friendly than gdcm so I'm really glad I don't have to resort to that. \ud83d\ude00 \nNo problem, @mrbean-bremen do you want to handle the underlying issue?\nCertainly - as I may have introduced it... Not sure if I'll find the time today though. ", "created_at": "2019-03-10T15:41:44Z", "version": "1.2", "FAIL_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding", "pydicom/tests/test_charset.py::TestCharset::test_patched_charset"], "PASS_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_encodings", "pydicom/tests/test_charset.py::TestCharset::test_nested_character_sets", "pydicom/tests/test_charset.py::TestCharset::test_inherited_character_set_in_sequence", "pydicom/tests/test_charset.py::TestCharset::test_standard_file", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set_enforce_valid", "pydicom/tests/test_charset.py::TestCharset::test_decoding_with_specific_tags", "pydicom/tests/test_charset.py::TestCharset::test_bad_charset", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_code_extensions_not_allowed", "pydicom/tests/test_charset.py::TestCharset::test_convert_encodings_warnings", "pydicom/tests/test_charset.py::TestCharset::test_convert_python_encodings", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_patched_code_extension_charset", "pydicom/tests/test_charset.py::TestCharset::test_multi_charset_default_value", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_personname", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_text", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_code_extensions[ISO", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrArab-\\u0642\\u0628\\u0627\\u0646\\u064a^\\u0644\\u0646\\u0632\\u0627\\u0631]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFren-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFrenMulti-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGerm-\\xc4neas^R\\xfcdiger]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGreek-\\u0394\\u03b9\\u03bf\\u03bd\\u03c5\\u03c3\\u03b9\\u03bf\\u03c2]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH31-Yamada^Tarou=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH32-\\uff94\\uff8f\\uff80\\uff9e^\\uff80\\uff9b\\uff73=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrHbrw-\\u05e9\\u05e8\\u05d5\\u05df^\\u05d3\\u05d1\\u05d5\\u05e8\\u05d4]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrI2-Hong^Gildong=\\u6d2a^\\u5409\\u6d1e=\\ud64d^\\uae38\\ub3d9]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMulti-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMultiExplicitIR6-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrKoreanMulti-\\uae40\\ud76c\\uc911]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrRuss-\\u041b\\u044e\\u043ace\\u043c\\u0431yp\\u0433]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX1-Wang^XiaoDong=\\u738b^\\u5c0f\\u6771]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX2-Wang^XiaoDong=\\u738b^\\u5c0f\\u4e1c]", "pydicom/tests/test_charset.py::TestCharset::test_changed_character_set"], "environment_setup_commit": "b4b44acbf1ddcaf03df16210aac46cb3a8acd6b9"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4043", "base_commit": "5f639444ddf4afe8b4f0d2c7f0d4ac0b976930de", "patch": "diff --git a/src/sqlfluff/cli/commands.py b/src/sqlfluff/cli/commands.py\n--- a/src/sqlfluff/cli/commands.py\n+++ b/src/sqlfluff/cli/commands.py\n@@ -1116,6 +1116,68 @@ def parse(\n         sys.exit(EXIT_SUCCESS)\n \n \n+@cli.command()\n+@common_options\n+@core_options\n+@click.argument(\"path\", nargs=1, type=click.Path(allow_dash=True))\n+def render(\n+    path: str,\n+    bench: bool,\n+    logger: Optional[logging.Logger] = None,\n+    extra_config_path: Optional[str] = None,\n+    ignore_local_config: bool = False,\n+    **kwargs,\n+) -> None:\n+    \"\"\"Render SQL files and just spit out the result.\n+\n+    PATH is the path to a sql file. This should be either a single file\n+    file ('path/to/file.sql') or a single ('-') character to indicate reading\n+    from *stdin*.\n+    \"\"\"\n+    c = get_config(\n+        extra_config_path, ignore_local_config, require_dialect=False, **kwargs\n+    )\n+    # We don't want anything else to be logged if we want json or yaml output\n+    # unless we're writing to a file.\n+    output_stream = make_output_stream(c, None, None)\n+    lnt, formatter = get_linter_and_formatter(c, output_stream)\n+    verbose = c.get(\"verbose\")\n+\n+    progress_bar_configuration.disable_progress_bar = True\n+\n+    formatter.dispatch_config(lnt)\n+\n+    # Set up logging.\n+    set_logging_level(\n+        verbosity=verbose,\n+        formatter=formatter,\n+        logger=logger,\n+        stderr_output=False,\n+    )\n+\n+    # handle stdin if specified via lone '-'\n+    with PathAndUserErrorHandler(formatter, path):\n+        if \"-\" == path:\n+            raw_sql = sys.stdin.read()\n+            fname = \"stdin\"\n+            file_config = lnt.config\n+        else:\n+            raw_sql, file_config, _ = lnt.load_raw_file_and_config(path, lnt.config)\n+            fname = path\n+\n+    # Get file specific config\n+    file_config.process_raw_file_for_config(raw_sql)\n+    rendered = lnt.render_string(raw_sql, fname, file_config, \"utf8\")\n+\n+    if rendered.templater_violations:\n+        for v in rendered.templater_violations:\n+            click.echo(formatter.format_violation(v))\n+        sys.exit(EXIT_FAIL)\n+    else:\n+        click.echo(rendered.templated_file.templated_str)\n+        sys.exit(EXIT_SUCCESS)\n+\n+\n # This \"__main__\" handler allows invoking SQLFluff using \"python -m\", which\n # simplifies the use of cProfile, e.g.:\n # python -m cProfile -s cumtime -m sqlfluff.cli.commands lint slow_file.sql\ndiff --git a/src/sqlfluff/core/linter/linter.py b/src/sqlfluff/core/linter/linter.py\n--- a/src/sqlfluff/core/linter/linter.py\n+++ b/src/sqlfluff/core/linter/linter.py\n@@ -110,7 +110,7 @@ def rule_tuples(self) -> List[RuleTuple]:\n     # These are the building blocks of the linting process.\n \n     @staticmethod\n-    def _load_raw_file_and_config(\n+    def load_raw_file_and_config(\n         fname: str, root_config: FluffConfig\n     ) -> Tuple[str, FluffConfig, str]:\n         \"\"\"Load a raw file and the associated config.\"\"\"\n@@ -837,7 +837,7 @@ def render_string(\n     def render_file(self, fname: str, root_config: FluffConfig) -> RenderedFile:\n         \"\"\"Load and render a file with relevant config.\"\"\"\n         # Load the raw file.\n-        raw_file, config, encoding = self._load_raw_file_and_config(fname, root_config)\n+        raw_file, config, encoding = self.load_raw_file_and_config(fname, root_config)\n         # Render the file\n         return self.render_string(raw_file, fname, config, encoding)\n \n@@ -1211,7 +1211,7 @@ def parse_path(\n                 self.formatter.dispatch_path(path)\n             # Load the file with the config and yield the result.\n             try:\n-                raw_file, config, encoding = self._load_raw_file_and_config(\n+                raw_file, config, encoding = self.load_raw_file_and_config(\n                     fname, self.config\n                 )\n             except SQLFluffSkipFile as s:\n", "test_patch": "diff --git a/test/cli/commands_test.py b/test/cli/commands_test.py\n--- a/test/cli/commands_test.py\n+++ b/test/cli/commands_test.py\n@@ -31,6 +31,7 @@\n     parse,\n     dialects,\n     get_config,\n+    render,\n )\n from sqlfluff.core.rules import BaseRule, LintFix, LintResult\n from sqlfluff.core.parser.segments.raw import CommentSegment\n@@ -250,6 +251,15 @@ def test__cli__command_lint_stdin(command):\n     invoke_assert_code(args=[lint, (\"--dialect=ansi\",) + command], cli_input=sql)\n \n \n+def test__cli__command_render_stdin():\n+    \"\"\"Check render on a simple script using stdin.\"\"\"\n+    with open(\"test/fixtures/cli/passing_a.sql\") as test_file:\n+        sql = test_file.read()\n+    result = invoke_assert_code(args=[render, (\"--dialect=ansi\", \"-\")], cli_input=sql)\n+    # Check we get back out the same file we input.\n+    assert result.output.startswith(sql)\n+\n+\n @pytest.mark.parametrize(\n     \"command\",\n     [\n@@ -263,6 +273,13 @@ def test__cli__command_lint_stdin(command):\n                 \"L051\",\n             ],\n         ),\n+        # Basic render\n+        (\n+            render,\n+            [\n+                \"test/fixtures/cli/passing_b.sql\",\n+            ],\n+        ),\n         # Original tests from test__cli__command_lint\n         (lint, [\"-n\", \"test/fixtures/cli/passing_a.sql\"]),\n         (lint, [\"-n\", \"-v\", \"test/fixtures/cli/passing_a.sql\"]),\n@@ -474,8 +491,15 @@ def test__cli__command_lint_parse(command):\n         (\n             (\n                 lint,\n-                [\"test/fixtures/cli/unknown_jinja_tag/test.sql\", \"-vvvvvvv\"],\n-                \"y\",\n+                [\"test/fixtures/cli/unknown_jinja_tag/test.sql\"],\n+            ),\n+            1,\n+        ),\n+        # Test render fail\n+        (\n+            (\n+                render,\n+                [\"test/fixtures/cli/fail_many.sql\"],\n             ),\n             1,\n         ),\n@@ -1790,3 +1814,41 @@ def test__cli__multiple_files__fix_multiple_errors_show_errors():\n \n     # Assert that they are sorted in alphabetical order\n     assert unfix_err_log.index(indent_pass_msg) < unfix_err_log.index(multi_fail_msg)\n+\n+\n+def test__cli__render_fail():\n+    \"\"\"Basic how render fails.\"\"\"\n+    expected_render_output = (\n+        \"L:   3 | P:   8 |  TMP | Undefined jinja template \" \"variable: 'something'\"\n+    )\n+\n+    result = invoke_assert_code(\n+        ret_code=1,\n+        args=[\n+            render,\n+            [\n+                \"test/fixtures/cli/fail_many.sql\",\n+            ],\n+        ],\n+    )\n+    # Check whole output. The replace command just accounts for\n+    # cross platform testing.\n+    assert result.output.replace(\"\\\\\", \"/\").startswith(expected_render_output)\n+\n+\n+def test__cli__render_pass():\n+    \"\"\"Basic how render works.\"\"\"\n+    expected_render_output = \"SELECT 56 FROM sch1.tbl2\"\n+\n+    result = invoke_assert_code(\n+        ret_code=0,\n+        args=[\n+            render,\n+            [\n+                \"test/fixtures/templater/jinja_a/jinja.sql\",\n+            ],\n+        ],\n+    )\n+    # Check whole output. The replace command just accounts for\n+    # cross platform testing.\n+    assert result.output.replace(\"\\\\\", \"/\").startswith(expected_render_output)\ndiff --git a/test/core/linter_test.py b/test/core/linter_test.py\n--- a/test/core/linter_test.py\n+++ b/test/core/linter_test.py\n@@ -90,7 +90,7 @@ def test__linter__skip_large_bytes(filesize, raises_skip):\n     # First check the function directly\n     if raises_skip:\n         with pytest.raises(SQLFluffSkipFile) as excinfo:\n-            Linter._load_raw_file_and_config(\n+            Linter.load_raw_file_and_config(\n                 \"test/fixtures/linter/indentation_errors.sql\", config\n             )\n         assert \"Skipping\" in str(excinfo.value)\n", "problem_statement": "add ability to render the compiled sql\n### Search before asking\n\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\n\n\n### Description\n\nIt would be nice to see the compiled sql in which any templates are rendered. I would be happy to work on this but it may be a struggle and would need some guidance.\n\n### Use case\n\n It would help debug linting errors around jinja templates.\r\nIt would also make it easier to copy and use the query in the bigquery ui, for example. We process our queries through Airflow so currently I can start a dag run and look at the rendered template to get this effect. That's not very efficient though :)\r\n\n\n### Dialect\n\nWe use bigquery but this could apply to all dialects.\n\n### Are you willing to work on and submit a PR to address the issue?\n\n- [X] Yes I am willing to submit a PR!\n\n### Code of Conduct\n\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\n\n", "hints_text": "If you're interested in contributing this feature, I suggest starting by looking at the existing `parse` command, implemented [here](https://github.com/sqlfluff/sqlfluff/blob/main/src/sqlfluff/cli/commands.py#L938). I think this new command will be pretty similar except for the output format.\r\n\r\nI think `sqlfluff render` is a good name for it. \ud83d\udc4d\nIf you have questions, feel free to drop them here or ask in the \"contributing\" Slack channel.", "created_at": "2022-11-08T14:56:55Z", "version": "1.3", "FAIL_TO_PASS": ["test/cli/commands_test.py::test__cli__command_directed", "test/cli/commands_test.py::test__cli__command_dialect", "test/cli/commands_test.py::test__cli__command_no_dialect", "test/cli/commands_test.py::test__cli__command_parse_error_dialect_explicit_warning", "test/cli/commands_test.py::test__cli__command_parse_error_dialect_implicit_warning", "test/cli/commands_test.py::test__cli__command_dialect_legacy", "test/cli/commands_test.py::test__cli__command_extra_config_fail", "test/cli/commands_test.py::test__cli__command_lint_stdin[command0]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command1]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command2]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command3]", "test/cli/commands_test.py::test__cli__command_render_stdin", "test/cli/commands_test.py::test__cli__command_lint_parse[command0]", "test/cli/commands_test.py::test__cli__command_lint_parse[command1]", "test/cli/commands_test.py::test__cli__command_lint_parse[command2]", "test/cli/commands_test.py::test__cli__command_lint_parse[command3]", "test/cli/commands_test.py::test__cli__command_lint_parse[command4]", "test/cli/commands_test.py::test__cli__command_lint_parse[command5]", "test/cli/commands_test.py::test__cli__command_lint_parse[command6]", "test/cli/commands_test.py::test__cli__command_lint_parse[command7]", "test/cli/commands_test.py::test__cli__command_lint_parse[command8]", "test/cli/commands_test.py::test__cli__command_lint_parse[command9]", "test/cli/commands_test.py::test__cli__command_lint_parse[command10]", "test/cli/commands_test.py::test__cli__command_lint_parse[command11]", "test/cli/commands_test.py::test__cli__command_lint_parse[command12]", "test/cli/commands_test.py::test__cli__command_lint_parse[command13]", "test/cli/commands_test.py::test__cli__command_lint_parse[command14]", "test/cli/commands_test.py::test__cli__command_lint_parse[command15]", "test/cli/commands_test.py::test__cli__command_lint_parse[command16]", "test/cli/commands_test.py::test__cli__command_lint_parse[command17]", "test/cli/commands_test.py::test__cli__command_lint_parse[command18]", "test/cli/commands_test.py::test__cli__command_lint_parse[command19]", "test/cli/commands_test.py::test__cli__command_lint_parse[command20]", "test/cli/commands_test.py::test__cli__command_lint_parse[command21]", "test/cli/commands_test.py::test__cli__command_lint_parse[command22]", "test/cli/commands_test.py::test__cli__command_lint_parse[command23]", "test/cli/commands_test.py::test__cli__command_lint_parse[command24]", "test/cli/commands_test.py::test__cli__command_lint_parse[command25]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command0-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command1-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command2-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command3-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command4-1]", "test/cli/commands_test.py::test__cli__command_lint_warning_explicit_file_ignored", "test/cli/commands_test.py::test__cli__command_lint_skip_ignore_files", "test/cli/commands_test.py::test__cli__command_lint_ignore_local_config", "test/cli/commands_test.py::test__cli__command_versioning", "test/cli/commands_test.py::test__cli__command_version", "test/cli/commands_test.py::test__cli__command_rules", "test/cli/commands_test.py::test__cli__command_dialects", "test/cli/commands_test.py::test__cli__command__fix[L001-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/whitespace_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L003-test/fixtures/linter/indentation_error_hard.sql]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_templating_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_suppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[0_lint_errors_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[0_lint_errors_1_suppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_parse_error_FIX_EVEN_UNPARSABLE]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[2_files_with_lint_errors_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[command-line-False]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[command-line-True]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[config-file-False]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[config-file-True]", "test/cli/commands_test.py::test__cli__fix_loop_limit_behavior[--", "test/cli/commands_test.py::test__cli__command_fix_stdin[select", "test/cli/commands_test.py::test__cli__command_fix_stdin[", "test/cli/commands_test.py::test__cli__command_fix_stdin[SELECT", "test/cli/commands_test.py::test__cli__command_fix_stdin_logging_to_stderr", "test/cli/commands_test.py::test__cli__command_fix_stdin_safety", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[create", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[select", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-y-0-0]", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-n-1-1]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[None-yaml]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[None-json]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[outfile-yaml]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[outfile-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[select", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[SElect", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command0]", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command1]", "test/cli/commands_test.py::test__cli__command_lint_nocolor", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-human]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-yaml]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-github-annotation-native]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-human]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-yaml]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-github-annotation-native]", "test/cli/commands_test.py::test__cli__command_lint_serialize_github_annotation", "test/cli/commands_test.py::test__cli__command_lint_serialize_github_annotation_native", "test/cli/commands_test.py::test__cli__command_lint_serialize_annotation_level_error_failure_equivalent[github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_annotation_level_error_failure_equivalent[github-annotation-native]", "test/cli/commands_test.py::test___main___help", "test/cli/commands_test.py::test_encoding[utf-8-ascii]", "test/cli/commands_test.py::test_encoding[utf-8-sig-UTF-8-SIG]", "test/cli/commands_test.py::test_encoding[utf-32-UTF-32]", "test/cli/commands_test.py::test_cli_encoding[utf-8-command-line-False]", "test/cli/commands_test.py::test_cli_encoding[utf-8-SIG-command-line-True]", "test/cli/commands_test.py::test_cli_encoding[utf-8-config-file-False]", "test/cli/commands_test.py::test_cli_encoding[utf-8-SIG-config-file-True]", "test/cli/commands_test.py::test_cli_no_disable_noqa_flag", "test/cli/commands_test.py::test_cli_disable_noqa_flag", "test/cli/commands_test.py::test_cli_get_default_config", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_disabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_disabled_progress_bar_deprecated_option", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_paths", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_files", "test/cli/commands_test.py::test__cli__fix_multiple_errors_no_show_errors", "test/cli/commands_test.py::test__cli__fix_multiple_errors_show_errors", "test/cli/commands_test.py::test__cli__multiple_files__fix_multiple_errors_show_errors", "test/cli/commands_test.py::test__cli__render_fail", "test/cli/commands_test.py::test__cli__render_pass", "test/core/linter_test.py::test__linter__path_from_paths__dir", "test/core/linter_test.py::test__linter__path_from_paths__default", "test/core/linter_test.py::test__linter__path_from_paths__exts", "test/core/linter_test.py::test__linter__path_from_paths__file", "test/core/linter_test.py::test__linter__skip_large_bytes[0-False]", "test/core/linter_test.py::test__linter__skip_large_bytes[5-True]", "test/core/linter_test.py::test__linter__skip_large_bytes[2000-False]", "test/core/linter_test.py::test__linter__path_from_paths__not_exist", "test/core/linter_test.py::test__linter__path_from_paths__not_exist_ignore", "test/core/linter_test.py::test__linter__path_from_paths__explicit_ignore", "test/core/linter_test.py::test__linter__path_from_paths__sqlfluffignore_current_directory", "test/core/linter_test.py::test__linter__path_from_paths__dot", "test/core/linter_test.py::test__linter__path_from_paths__ignore[test/fixtures/linter/sqlfluffignore]", "test/core/linter_test.py::test__linter__path_from_paths__ignore[test/fixtures/linter/sqlfluffignore/]", "test/core/linter_test.py::test__linter__path_from_paths__ignore[test/fixtures/linter/sqlfluffignore/.]", "test/core/linter_test.py::test__linter__lint_string_vs_file[test/fixtures/linter/indentation_errors.sql]", "test/core/linter_test.py::test__linter__lint_string_vs_file[test/fixtures/linter/whitespace_errors.sql]", "test/core/linter_test.py::test__linter__get_violations_filter_rules[None-7]", "test/core/linter_test.py::test__linter__get_violations_filter_rules[L010-2]", "test/core/linter_test.py::test__linter__get_violations_filter_rules[rules2-2]", "test/core/linter_test.py::test__linter__linting_result__sum_dicts", "test/core/linter_test.py::test__linter__linting_result__combine_dicts", "test/core/linter_test.py::test__linter__linting_result_check_tuples_by_path[False-list]", "test/core/linter_test.py::test__linter__linting_result_check_tuples_by_path[True-dict]", "test/core/linter_test.py::test__linter__linting_result_get_violations[1]", "test/core/linter_test.py::test__linter__linting_result_get_violations[2]", "test/core/linter_test.py::test__linter__linting_parallel_thread[False]", "test/core/linter_test.py::test__linter__linting_parallel_thread[True]", "test/core/linter_test.py::test_lint_path_parallel_wrapper_exception", "test/core/linter_test.py::test__linter__get_runner_processes[512-1-1]", "test/core/linter_test.py::test__linter__get_runner_processes[512-0-512]", "test/core/linter_test.py::test__linter__get_runner_processes[512--12-500]", "test/core/linter_test.py::test__linter__get_runner_processes[512-5-5]", "test/core/linter_test.py::test__linter__get_runner_processes[1--1-1]", "test/core/linter_test.py::test__linter__linting_unexpected_error_handled_gracefully", "test/core/linter_test.py::test__linter__raises_malformed_noqa", "test/core/linter_test.py::test__linter__empty_file", "test/core/linter_test.py::test__linter__mask_templated_violations[True-check_tuples0]", "test/core/linter_test.py::test__linter__mask_templated_violations[False-check_tuples1]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8.sql-autodetect-False]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8-sig.sql-autodetect-False]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8.sql-utf-8-False]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8-sig.sql-utf-8-True]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8.sql-utf-8-sig-False]", "test/core/linter_test.py::test__linter__encoding[test/fixtures/linter/encoding-utf-8-sig.sql-utf-8-sig-False]", "test/core/linter_test.py::test_parse_noqa[-None]", "test/core/linter_test.py::test_parse_noqa[noqa-expected1]", "test/core/linter_test.py::test_parse_noqa[noqa?-SQLParseError]", "test/core/linter_test.py::test_parse_noqa[noqa:-expected3]", "test/core/linter_test.py::test_parse_noqa[noqa:L001,L002-expected4]", "test/core/linter_test.py::test_parse_noqa[noqa:", "test/core/linter_test.py::test_parse_noqa[Inline", "test/core/linter_test.py::test_parse_noqa_no_dups", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_no_ignore]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_ignore_specific_line]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_ignore_different_specific_line]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_ignore_different_specific_rule]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_ignore_enable_this_range]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_ignore_disable_this_range]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_1_ignore_disable_specific_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_2_ignore_disable_specific_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_3_ignore_disable_specific_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_4_ignore_disable_specific_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_1_ignore_disable_all_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_2_ignore_disable_all_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_3_ignore_disable_all_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violation_line_4_ignore_disable_all_2_3]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[4_violations_two_types_disable_specific_enable_all]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[4_violations_two_types_disable_all_enable_specific]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violations_comment_inline_ignore]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[2_violations_comment_inline_ignore]", "test/core/linter_test.py::test_linted_file_ignore_masked_violations[1_violations_comment_inline_glob_ignore]", "test/core/linter_test.py::test_linter_noqa", "test/core/linter_test.py::test_linter_noqa_with_templating", "test/core/linter_test.py::test_linter_noqa_template_errors", "test/core/linter_test.py::test_linter_noqa_prs", "test/core/linter_test.py::test_linter_noqa_tmp", "test/core/linter_test.py::test_linter_noqa_disable", "test/core/linter_test.py::test_delayed_exception", "test/core/linter_test.py::test__attempt_to_change_templater_warning", "test/core/linter_test.py::test_safe_create_replace_file[utf8_create]", "test/core/linter_test.py::test_safe_create_replace_file[utf8_update]", "test/core/linter_test.py::test_safe_create_replace_file[utf8_special_char]", "test/core/linter_test.py::test_safe_create_replace_file[incorrect_encoding]", "test/core/linter_test.py::test_advanced_api_methods", "test/core/linter_test.py::test_normalise_newlines", "test/core/linter_test.py::test_require_match_parse_grammar"], "PASS_TO_PASS": [], "environment_setup_commit": "dc59c2a5672aacedaf91f0e6129b467eefad331b"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1782", "base_commit": "0bc5a53dedd8aa9e553c732a31003ce020bc2f54", "patch": "diff --git a/pvlib/singlediode.py b/pvlib/singlediode.py\n--- a/pvlib/singlediode.py\n+++ b/pvlib/singlediode.py\n@@ -794,6 +794,13 @@ def _lambertw(photocurrent, saturation_current, resistance_series,\n     # Compute open circuit voltage\n     v_oc = _lambertw_v_from_i(0., **params)\n \n+    # Set small elements <0 in v_oc to 0\n+    if isinstance(v_oc, np.ndarray):\n+        v_oc[(v_oc < 0) & (v_oc > -1e-12)] = 0.\n+    elif isinstance(v_oc, (float, int)):\n+        if v_oc < 0 and v_oc > -1e-12:\n+            v_oc = 0.\n+\n     # Find the voltage, v_mp, where the power is maximized.\n     # Start the golden section search at v_oc * 1.14\n     p_mp, v_mp = _golden_sect_DataFrame(params, 0., v_oc * 1.14, _pwr_optfcn)\n", "test_patch": "diff --git a/pvlib/tests/test_singlediode.py b/pvlib/tests/test_singlediode.py\n--- a/pvlib/tests/test_singlediode.py\n+++ b/pvlib/tests/test_singlediode.py\n@@ -168,6 +168,19 @@ def test_singlediode_precision(method, precise_iv_curves):\n     assert np.allclose(pc['i_xx'], outs['i_xx'], atol=1e-6, rtol=0)\n \n \n+def test_singlediode_lambert_negative_voc():\n+\n+    # Those values result in a negative v_oc out of `_lambertw_v_from_i`\n+    x = np.array([0., 1.480501e-11, 0.178, 8000., 1.797559])\n+    outs = pvsystem.singlediode(*x, method='lambertw')\n+    assert outs['v_oc'] == 0\n+\n+    # Testing for an array\n+    x  = np.array([x, x]).T\n+    outs = pvsystem.singlediode(*x, method='lambertw')\n+    assert np.array_equal(outs['v_oc'], [0, 0])\n+\n+\n @pytest.mark.parametrize('method', ['lambertw'])\n def test_ivcurve_pnts_precision(method, precise_iv_curves):\n     \"\"\"\n", "problem_statement": "_golden_sect_DataFrame changes in 0.9.4\n**Describe the bug**\r\n\r\n`0.9.4` introduced the following changes in the `_golden_sect_DataFrame`: We are checking `upper` and `lower` parameters and raise an error if `lower > upper`.\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/81598e4fa8a9bd8fadaa7544136579c44885b3d1/pvlib/tools.py#L344-L345\r\n\r\n`_golden_sect_DataFrame` is used by `_lambertw`:\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/81598e4fa8a9bd8fadaa7544136579c44885b3d1/pvlib/singlediode.py#L644-L649\r\n\r\nI often have slightly negative `v_oc` values (really close to 0) when running simulations (second number in the array below):\r\n```\r\narray([ 9.46949758e-16, -8.43546518e-15,  2.61042547e-15,  3.82769773e-15,\r\n        1.01292315e-15,  4.81308106e+01,  5.12484772e+01,  5.22675087e+01,\r\n        5.20708941e+01,  5.16481028e+01,  5.12364071e+01,  5.09209060e+01,\r\n        5.09076598e+01,  5.10187680e+01,  5.11328118e+01,  5.13997628e+01,\r\n        5.15121386e+01,  5.05621451e+01,  4.80488068e+01,  7.18224446e-15,\r\n        1.21386700e-14,  6.40136698e-16,  4.36081007e-16,  6.51236255e-15])\r\n```\r\n\r\nIf we have one negative number in a large timeseries, the simulation will crash which seems too strict.\r\n\r\n**Expected behavior**\r\n\r\nThat would be great to either:\r\n* Have this data check be less strict and allow for slightly negative numbers, which are not going to affect the quality of the results.\r\n* On `_lambertw`: Do not allow negative `v_oc` and set negative values to `np.nan`, so that the error is not triggered. It will be up to the upstream code (user) to manage those `np.nan`.\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: >= 0.9.4\r\n - ``pandas.__version__``: 1.5.3\r\n - python: 3.10.11\r\n\nsinglediode error with very low effective_irradiance\n**Describe the bug**\r\n\r\nSince pvlib 0.9.4 release (https://github.com/pvlib/pvlib-python/pull/1606) I get an error while running the single-diode model with some very low effective irradiance values.\r\n\r\n**To Reproduce**\r\n\r\n```python\r\nfrom pvlib import pvsystem\r\n\r\neffective_irradiance=1.341083e-17\r\ntemp_cell=13.7 \r\n\r\ncec_modules = pvsystem.retrieve_sam('CECMod')\r\ncec_module = cec_modules['Trina_Solar_TSM_300DEG5C_07_II_']\r\n\r\nmount = pvsystem.FixedMount()\r\narray = pvsystem.Array(mount=mount,\r\n                       module_parameters=cec_module)\r\n\r\nsystem = pvsystem.PVSystem(arrays=[array])\r\n\r\nparams = system.calcparams_cec(effective_irradiance, \r\n                               temp_cell)\r\n\r\nsystem.singlediode(*params)\r\n```\r\n\r\n```in _golden_sect_DataFrame(params, lower, upper, func, atol)\r\n    303 \"\"\"\r\n    304 Vectorized golden section search for finding maximum of a function of a\r\n    305 single variable.\r\n   (...)\r\n    342 pvlib.singlediode._pwr_optfcn\r\n    343 \"\"\"\r\n    344 if np.any(upper - lower < 0.):\r\n--> 345     raise ValueError('upper >= lower is required')\r\n    347 phim1 = (np.sqrt(5) - 1) / 2\r\n    349 df = params\r\n\r\nValueError: upper >= lower is required\r\n```\r\n\r\n**Expected behavior**\r\nThis complicates the bifacial modeling procedure as `run_model_from_effective_irradiance` can be called with very low irradiance values estimated by pvfactors (at sunrise or sunset for instance). \r\n\r\n**Versions:**\r\n - ``pvlib.__version__``:  0.9.4\r\n - ``pandas.__version__``: 1.5.3\r\n - python: 3.10\r\n\r\n**Additional context**\r\n\r\nv_oc is negative in this case which causes the error. \r\n\r\n```python\r\nfrom pvlib.singlediode import _lambertw_v_from_i\r\nphotocurrent = params[0]\r\nsaturation_current = params[1]\r\nresistance_series = params[2]\r\nresistance_shunt = params[3]\r\nnNsVth = params[4]\r\nv_oc = _lambertw_v_from_i(resistance_shunt, resistance_series, nNsVth, 0.,\r\n                              saturation_current, photocurrent)\r\n```\r\n\n", "hints_text": "See #1673 \n@cedricleroy can you provide the inputs and function call that produced the negative `v_oc` shown above?\n@echedey-ls Thanks! I thought I checked for related issues, but apparently not enough \ud83d\ude04 \r\n\r\n@cwhanse Sure thing:\r\n\r\nRunning [`_lambertw_v_from_i` in `_lambertw`](https://github.com/pvlib/pvlib-python/blob/v0.9.4/pvlib/singlediode.py#L639-L641) with the following data:\r\n\r\n```\r\n    resistance_shunt  resistance_series    nNsVth  current  saturation_current  photocurrent          v_oc\r\n0        8000.000000              0.178  1.797559      0.0        1.480501e-11      0.000000  8.306577e-16\r\n1        8000.000000              0.178  1.797048      0.0        1.456894e-11      0.000000 -7.399531e-15\r\n2        8000.000000              0.178  1.791427      0.0        1.220053e-11      0.000000  2.289847e-15\r\n3        8000.000000              0.178  1.789892      0.0        1.162201e-11      0.000000  3.357630e-15\r\n4        8000.000000              0.178  1.790915      0.0        1.200467e-11      0.000000  8.885291e-16\r\n5        7384.475098              0.178  1.796786      0.0        1.444902e-11      0.237291  4.222001e+01\r\n6        5023.829590              0.178  1.814643      0.0        2.524836e-11      1.458354  4.495480e+01\r\n7        2817.370605              0.178  1.841772      0.0        5.803733e-11      3.774055  4.584869e+01\r\n8        1943.591919              0.178  1.877364      0.0        1.682954e-10      6.225446  4.567622e+01\r\n9        1609.391479              0.178  1.910984      0.0        4.479085e-10      8.887444  4.530535e+01\r\n10       1504.273193              0.178  1.937034      0.0        9.402419e-10     11.248103  4.494422e+01\r\n11       1482.143799              0.178  1.951216      0.0        1.399556e-09     12.272360  4.466746e+01\r\n12       1485.013794              0.178  1.950762      0.0        1.381967e-09     12.114989  4.465584e+01\r\n13       1506.648315              0.178  1.942643      0.0        1.100982e-09     11.167084  4.475331e+01\r\n14       1580.780029              0.178  1.928508      0.0        7.387948e-10      9.350249  4.485334e+01\r\n15       1832.828735              0.178  1.901971      0.0        3.453772e-10      6.842797  4.508751e+01\r\n16       2604.075684              0.178  1.869294      0.0        1.325485e-10      4.191604  4.518609e+01\r\n17       4594.301270              0.178  1.844949      0.0        6.390201e-11      1.771347  4.435276e+01\r\n18       6976.270996              0.178  1.829467      0.0        3.987927e-11      0.409881  4.214808e+01\r\n19       8000.000000              0.178  1.821491      0.0        3.120619e-11      0.000000  6.300214e-15\r\n20       8000.000000              0.178  1.813868      0.0        2.464867e-11      0.000000  1.064796e-14\r\n21       8000.000000              0.178  1.809796      0.0        2.171752e-11      0.000000  5.615234e-16\r\n22       8000.000000              0.178  1.808778      0.0        2.103918e-11      0.000000  3.825272e-16\r\n23       8000.000000              0.178  1.806231      0.0        1.943143e-11      0.000000  5.712599e-15\r\n```\r\n\r\n[data.csv](https://github.com/pvlib/pvlib-python/files/11807543/data.csv)\r\n\r\n\n> If we have one negative number in a large timeseries, the simulation will crash which seems too strict.\r\n\r\nAgree this is not desirable.\r\n\r\nMy thoughts:\r\n\r\n1. We could insert `v_oc = np.maximum(v_oc, 0)` above this [line](https://github.com/pvlib/pvlib-python/blob/e643dc3f835c29b12b13d7375e33885dcb5d07c7/pvlib/singlediode.py#L649). That would preserve nan.\r\n2. I am reluctant to change `_lambertw_v_from_i`. That function's job is to solve the diode equation, which is valid for negative current. I don't think this function should make decisions about its solution. There will always be some degree of imprecision (currently it's around 10-13 or smaller, I think).\r\n3. I am also reluctant to change `_golden_sect_DataFrame` for similar reasons - the function's job should be to find a minimum using the golden section search. Complying with the `lower < upper` requirement is the job of the code that calls this function.\r\n\r\n\n1/ makes sense to me. I agree with the CONS for 2/ and 3/\r\n\r\nHappy to open a PR with 1. if that helps. \n> Happy to open a PR with 1. if that helps.\r\n\r\nThat is welcome.  Because I'm cautious about junk values with larger magnitude being covered up by 0s, maybe \r\n\r\n```\r\nv_oc[(v_oc < 0) & (v_oc > 1e-12)] = 0.\r\n```\r\n\r\n\nThat's unexpected, thanks for reporting. \r\n\r\nI'll note that the negative Voc results from taking the difference of two very large but nearly equal numbers. It's likely limited to the CEC model, where the shunt resistance is inversely proportional to irradiance, which would be about 1e19 at photocurrent of 1e-17 for this case.\nNow this gets strange: the Voc value is positive with pvlib v0.9.3. The function involved `pvlib.singlediode._lambertw_v_from_i` hasn't changed for many releases. In both pvlib v0.9.3 and v0.9.4, in this calculation of Voc, the lambertw term overflows so the Voc value is computed using only python arithmetic operators and numpy.log.\r\n\r\nI'm starting to think the error depends on python and numpy versions.\nThe difference between 0.9.3 and 0.9.4 here may be due to slightly different values returned by `calcparams_cec`.  Compare the output of `print(list(map(str, params)))`; I get slightly different saturation current values for the given example.  Maybe the changed Boltzmann constant in #1617 is the cause?\n+1 to #1617 as the likely culprit. I get the positive/negative Voc values with the same python and numpy versions but different pvlib versions.\nTo illustrate the challenge, [this line](https://github.com/pvlib/pvlib-python/blob/f4d7c6e1c17b3fddba7cc49d39feed2a6fa0f30e/pvlib/singlediode.py#L566) computes the Voc.\r\n\r\nStripping out the indexing the computation is\r\n\r\n```\r\n    V = (IL + I0 - I) / Gsh - \\\r\n        I * Rs - a * lambertwterm\r\n```\r\nWith pvlib v0.9.4, Io is 7.145289906185543e-12. a is not affected, since a value of the Boltzmann contant is inherent in the a_ref value from the database. (IL + I0 - I) / Gsh is 107825185636.40567, I * Rs is 0, and a * lambertwterm is 107825185636.40569\r\n\r\nWith pvlib v0.9.3, Io is 7.145288699667595e-12.  (IL + I0 - I) / Gsh is 107825167429.58397, I * Rs is 0, and a * lambertwterm is 107825167429.58395\r\n\r\nThe difference defining Voc is in the least significant digit.\r\n\nIncreasing the iterations that solve for lambertwterm doesn't fix this issue.\nThis smells to me like the inevitable error from accumulated round-off. \r\n\r\nFWIW, negative Voc can be achieved in 0.9.3 as well -- try the given example but with `effective_irradiance=1.e-18`.  The difference is that before #1606, it led to nans and warnings instead of raising an error. \r\n\r\n\n@pasquierjb I recommend intercepting the effective irradiance and setting values to 0 which are below a minimum on the order of 1e-9 W/m2. That will propagate to shunt resistance = np.inf, which changes the calculation path in pvlib.singlediode and gives Voc=0.\r\n\r\nI'm not sure we'll be able to extend the numerical solution of the single diode equation to be accurate at very low but non-zero values of photocurrent (and/or enormous but finite values of shunt resistance.)\r\n\r\nI note that `pvlib.pvsystem.calcparams_desoto` doesn't like `effective_irradiance=0.` but is OK with `effective_irradiance=np.array([0.])`.  Has to do with trapping and ignoring division by zero warnings and errors.\nHave you tried setting `method='newton'` instead of `'lambertw'`? https://pvlib-python.readthedocs.io/en/stable/reference/generated/pvlib.pvsystem.singlediode.html#pvlib-pvsystem-singlediode\nSetting `method='newton'` gets a solution to this case. `method` isn't available as a parameter of the `PVSystem.singlediode` method so @pasquierjb would need to change his workflow to use it. Something for us to consider adding.\nMy workaround for this issue was to first filter very low `effective_irradiance` values (`<1e-8`), and then filter `photocurrent` and `saturation_current` parameters when `effective_irradiance=0` and made them `=0`. This assures that you won't get negative `v_oc` values.", "created_at": "2023-06-26T14:46:54Z", "version": "0.9", "FAIL_TO_PASS": ["pvlib/tests/test_singlediode.py::test_singlediode_lambert_negative_voc"], "PASS_TO_PASS": ["pvlib/tests/test_singlediode.py::test_method_spr_e20_327[brentq]", "pvlib/tests/test_singlediode.py::test_method_spr_e20_327[newton]", "pvlib/tests/test_singlediode.py::test_newton_fs_495[brentq]", "pvlib/tests/test_singlediode.py::test_newton_fs_495[newton]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[1-lambertw]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[1-brentq]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[1-newton]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[2-lambertw]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[2-brentq]", "pvlib/tests/test_singlediode.py::test_singlediode_precision[2-newton]", "pvlib/tests/test_singlediode.py::test_ivcurve_pnts_precision[1-lambertw]", "pvlib/tests/test_singlediode.py::test_ivcurve_pnts_precision[2-lambertw]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[1-lambertw]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[1-brentq]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[1-newton]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[2-lambertw]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[2-brentq]", "pvlib/tests/test_singlediode.py::test_v_from_i_i_from_v_precision[2-newton]", "pvlib/tests/test_singlediode.py::test_pvsyst_recombination_loss[newton-1000-25-expected0-tol0]", "pvlib/tests/test_singlediode.py::test_pvsyst_recombination_loss[newton-888-55-expected1-tol1]", "pvlib/tests/test_singlediode.py::test_pvsyst_recombination_loss[brentq-1000-25-expected0-tol0]", "pvlib/tests/test_singlediode.py::test_pvsyst_recombination_loss[brentq-888-55-expected1-tol1]", "pvlib/tests/test_singlediode.py::test_pvsyst_breakdown[newton-brk_params0-recomb_params0-1000-25-expected0-tol0]", "pvlib/tests/test_singlediode.py::test_pvsyst_breakdown[newton-brk_params1-recomb_params1-888-55-expected1-tol1]", "pvlib/tests/test_singlediode.py::test_pvsyst_breakdown[brentq-brk_params0-recomb_params0-1000-25-expected0-tol0]", "pvlib/tests/test_singlediode.py::test_pvsyst_breakdown[brentq-brk_params1-recomb_params1-888-55-expected1-tol1]", "pvlib/tests/test_singlediode.py::test_bishop88_kwargs_transfer[newton-method_kwargs0]", "pvlib/tests/test_singlediode.py::test_bishop88_kwargs_transfer[brentq-method_kwargs1]", "pvlib/tests/test_singlediode.py::test_bishop88_kwargs_fails[newton-method_kwargs0]", "pvlib/tests/test_singlediode.py::test_bishop88_kwargs_fails[brentq-method_kwargs1]", "pvlib/tests/test_singlediode.py::test_bishop88_full_output_kwarg[newton]", "pvlib/tests/test_singlediode.py::test_bishop88_full_output_kwarg[brentq]"], "environment_setup_commit": "6072e0982c3c0236f532ddfa48fbf461180d834e"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1333", "base_commit": "d2a5b3c7b1e203fec3c7ca73c30eb1785d3d4d0a", "patch": "diff --git a/astroid/modutils.py b/astroid/modutils.py\n--- a/astroid/modutils.py\n+++ b/astroid/modutils.py\n@@ -297,6 +297,9 @@ def _get_relative_base_path(filename, path_to_check):\n     if os.path.normcase(real_filename).startswith(path_to_check):\n         importable_path = real_filename\n \n+    # if \"var\" in path_to_check:\n+    #     breakpoint()\n+\n     if importable_path:\n         base_path = os.path.splitext(importable_path)[0]\n         relative_base_path = base_path[len(path_to_check) :]\n@@ -307,8 +310,11 @@ def _get_relative_base_path(filename, path_to_check):\n \n def modpath_from_file_with_callback(filename, path=None, is_package_cb=None):\n     filename = os.path.expanduser(_path_from_filename(filename))\n+    paths_to_check = sys.path.copy()\n+    if path:\n+        paths_to_check += path\n     for pathname in itertools.chain(\n-        path or [], map(_cache_normalize_path, sys.path), sys.path\n+        paths_to_check, map(_cache_normalize_path, paths_to_check)\n     ):\n         if not pathname:\n             continue\n", "test_patch": "diff --git a/tests/unittest_modutils.py b/tests/unittest_modutils.py\n--- a/tests/unittest_modutils.py\n+++ b/tests/unittest_modutils.py\n@@ -30,6 +30,7 @@\n import tempfile\n import unittest\n import xml\n+from pathlib import Path\n from xml import etree\n from xml.etree import ElementTree\n \n@@ -189,6 +190,30 @@ def test_load_from_module_symlink_on_symlinked_paths_in_syspath(self) -> None:\n         # this should be equivalent to: import secret\n         self.assertEqual(modutils.modpath_from_file(symlink_secret_path), [\"secret\"])\n \n+    def test_load_packages_without_init(self) -> None:\n+        \"\"\"Test that we correctly find packages with an __init__.py file.\n+\n+        Regression test for issue reported in:\n+        https://github.com/PyCQA/astroid/issues/1327\n+        \"\"\"\n+        tmp_dir = Path(tempfile.gettempdir())\n+        self.addCleanup(os.chdir, os.curdir)\n+        os.chdir(tmp_dir)\n+\n+        self.addCleanup(shutil.rmtree, tmp_dir / \"src\")\n+        os.mkdir(tmp_dir / \"src\")\n+        os.mkdir(tmp_dir / \"src\" / \"package\")\n+        with open(tmp_dir / \"src\" / \"__init__.py\", \"w\", encoding=\"utf-8\"):\n+            pass\n+        with open(tmp_dir / \"src\" / \"package\" / \"file.py\", \"w\", encoding=\"utf-8\"):\n+            pass\n+\n+        # this should be equivalent to: import secret\n+        self.assertEqual(\n+            modutils.modpath_from_file(str(Path(\"src\") / \"package\"), [\".\"]),\n+            [\"src\", \"package\"],\n+        )\n+\n \n class LoadModuleFromPathTest(resources.SysPathSetup, unittest.TestCase):\n     def test_do_not_load_twice(self) -> None:\n", "problem_statement": "astroid 2.9.1 breaks pylint with missing __init__.py: F0010: error while code parsing: Unable to load file __init__.py\n### Steps to reproduce\r\n> Steps provided are for Windows 11, but initial problem found in Ubuntu 20.04\r\n\r\n> Update 2022-01-04: Corrected repro steps and added more environment details\r\n\r\n1. Set up simple repo with following structure (all files can be empty):\r\n```\r\nroot_dir/\r\n|--src/\r\n|----project/ # Notice the missing __init__.py\r\n|------file.py # It can be empty, but I added `import os` at the top\r\n|----__init__.py\r\n```\r\n2. Open a command prompt\r\n3. `cd root_dir`\r\n4. `python -m venv venv`\r\n5. `venv/Scripts/activate`\r\n6. `pip install pylint astroid==2.9.1` # I also repro'd on the latest, 2.9.2\r\n7. `pylint src/project` # Updated from `pylint src`\r\n8. Observe failure:\r\n```\r\nsrc\\project\\__init__.py:1:0: F0010: error while code parsing: Unable to load file src\\project\\__init__.py:\r\n```\r\n\r\n### Current behavior\r\nFails with `src\\project\\__init__.py:1:0: F0010: error while code parsing: Unable to load file src\\project\\__init__.py:`\r\n\r\n### Expected behavior\r\nDoes not fail with error.\r\n> If you replace step 6 with `pip install pylint astroid==2.9.0`, you get no failure with an empty output - since no files have content\r\n\r\n### `python -c \"from astroid import __pkginfo__; print(__pkginfo__.version)\"` output\r\n2.9.1\r\n\r\n`python 3.9.1`\r\n`pylint 2.12.2 `\r\n\r\n\r\n\r\nThis issue has been observed with astroid `2.9.1` and `2.9.2`\n", "hints_text": "I can't seem to reproduce this in my `virtualenv`. This might be specific to `venv`? Needs some further investigation.\n@interifter Which version of `pylint` are you using?\nRight, ``pip install pylint astroid==2.9.0``, will keep the local version if you already have one, so I thought it was ``2.12.2`` but that could be false. In fact it probably isn't 2.12.2. For the record, you're not supposed to set the version of ``astroid`` yourself, pylint does, and bad thing will happen if you try to set the version of an incompatible astroid. We might want to update the issue's template to have this information next.\nMy apologies... I updated the repro steps with a critical missed detail: `pylint src/project`, instead of `pylint src`\r\n\r\nBut I verified that either with, or without, `venv`, the issue is reproduced.\r\n\r\nAlso, I never have specified the `astroid` version, before. \r\n\r\nHowever, this isn't the first time the issue has been observed.\r\nBack in early 2019, a [similar issue](https://stackoverflow.com/questions/48024049/pylint-raises-error-if-directory-doesnt-contain-init-py-file) was observed with either `astroid 2.2.0` or `isort 4.3.5`, which led me to try pinning `astroid==2.9.0`, which worked.\n> @interifter Which version of `pylint` are you using?\r\n\r\n`2.12.2`\r\n\r\nFull env info:\r\n\r\n```\r\nPackage           Version\r\n----------------- -------\r\nastroid           2.9.2\r\ncolorama          0.4.4\r\nisort             5.10.1\r\nlazy-object-proxy 1.7.1\r\nmccabe            0.6.1\r\npip               20.2.3\r\nplatformdirs      2.4.1\r\npylint            2.12.2\r\nsetuptools        49.2.1\r\ntoml              0.10.2\r\ntyping-extensions 4.0.1\r\nwrapt             1.13.3\r\n```\r\n\nI confirm the bug and i'm able to reproduce it with `python 3.9.1`. \r\n```\r\n$> pip freeze\r\nastroid==2.9.2\r\nisort==5.10.1\r\nlazy-object-proxy==1.7.1\r\nmccabe==0.6.1\r\nplatformdirs==2.4.1\r\npylint==2.12.2\r\ntoml==0.10.2\r\ntyping-extensions==4.0.1\r\nwrapt==1.13.3\r\n```\nBisected and this is the faulty commit:\r\nhttps://github.com/PyCQA/astroid/commit/2ee20ccdf62450db611acc4a1a7e42f407ce8a14\nFix in #1333, no time to write tests yet so if somebody has any good ideas: please let me know!", "created_at": "2022-01-08T19:36:45Z", "version": "2.10", "FAIL_TO_PASS": ["tests/unittest_modutils.py::ModPathFromFileTest::test_load_packages_without_init"], "PASS_TO_PASS": ["tests/unittest_modutils.py::ModuleFileTest::test_find_egg_module", "tests/unittest_modutils.py::ModuleFileTest::test_find_zipped_module", "tests/unittest_modutils.py::LoadModuleFromNameTest::test_known_values_load_module_from_name_1", "tests/unittest_modutils.py::LoadModuleFromNameTest::test_known_values_load_module_from_name_2", "tests/unittest_modutils.py::LoadModuleFromNameTest::test_raise_load_module_from_name_1", "tests/unittest_modutils.py::GetModulePartTest::test_get_module_part_exception", "tests/unittest_modutils.py::GetModulePartTest::test_known_values_get_builtin_module_part", "tests/unittest_modutils.py::GetModulePartTest::test_known_values_get_compiled_module_part", "tests/unittest_modutils.py::GetModulePartTest::test_known_values_get_module_part_1", "tests/unittest_modutils.py::GetModulePartTest::test_known_values_get_module_part_2", "tests/unittest_modutils.py::GetModulePartTest::test_known_values_get_module_part_3", "tests/unittest_modutils.py::ModPathFromFileTest::test_import_symlink_both_outside_of_path", "tests/unittest_modutils.py::ModPathFromFileTest::test_import_symlink_with_source_outside_of_path", "tests/unittest_modutils.py::ModPathFromFileTest::test_known_values_modpath_from_file_1", "tests/unittest_modutils.py::ModPathFromFileTest::test_load_from_module_symlink_on_symlinked_paths_in_syspath", "tests/unittest_modutils.py::ModPathFromFileTest::test_raise_modpath_from_file_exception", "tests/unittest_modutils.py::LoadModuleFromPathTest::test_do_not_load_twice", "tests/unittest_modutils.py::FileFromModPathTest::test_builtin", "tests/unittest_modutils.py::FileFromModPathTest::test_site_packages", "tests/unittest_modutils.py::FileFromModPathTest::test_std_lib", "tests/unittest_modutils.py::FileFromModPathTest::test_unexisting", "tests/unittest_modutils.py::FileFromModPathTest::test_unicode_in_package_init", "tests/unittest_modutils.py::GetSourceFileTest::test", "tests/unittest_modutils.py::GetSourceFileTest::test_raise", "tests/unittest_modutils.py::StandardLibModuleTest::test_4", "tests/unittest_modutils.py::StandardLibModuleTest::test_builtin", "tests/unittest_modutils.py::StandardLibModuleTest::test_builtins", "tests/unittest_modutils.py::StandardLibModuleTest::test_custom_path", "tests/unittest_modutils.py::StandardLibModuleTest::test_datetime", "tests/unittest_modutils.py::StandardLibModuleTest::test_failing_edge_cases", "tests/unittest_modutils.py::StandardLibModuleTest::test_nonstandard", "tests/unittest_modutils.py::StandardLibModuleTest::test_unknown", "tests/unittest_modutils.py::IsRelativeTest::test_deep_relative", "tests/unittest_modutils.py::IsRelativeTest::test_deep_relative2", "tests/unittest_modutils.py::IsRelativeTest::test_deep_relative3", "tests/unittest_modutils.py::IsRelativeTest::test_deep_relative4", "tests/unittest_modutils.py::IsRelativeTest::test_is_relative_bad_path", "tests/unittest_modutils.py::IsRelativeTest::test_known_values_is_relative_1", "tests/unittest_modutils.py::IsRelativeTest::test_known_values_is_relative_3", "tests/unittest_modutils.py::IsRelativeTest::test_known_values_is_relative_4", "tests/unittest_modutils.py::IsRelativeTest::test_known_values_is_relative_5", "tests/unittest_modutils.py::GetModuleFilesTest::test_get_all_files", "tests/unittest_modutils.py::GetModuleFilesTest::test_get_module_files_1", "tests/unittest_modutils.py::GetModuleFilesTest::test_load_module_set_attribute", "tests/unittest_modutils.py::ExtensionPackageWhitelistTest::test_is_module_name_part_of_extension_package_whitelist_success", "tests/unittest_modutils.py::ExtensionPackageWhitelistTest::test_is_module_name_part_of_extension_package_whitelist_true"], "environment_setup_commit": "da745538c7236028a22cdf0405f6829fcf6886bc"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-916", "base_commit": "a94cc9996d2c716298c846f2dbba84f9f901e0a7", "patch": "diff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -329,41 +329,60 @@ def from_json(cls, dataset_class, tag, vr, value, value_key,\n                     # Some DICOMweb services get this wrong, so we\n                     # workaround the issue and warn the user\n                     # rather than raising an error.\n-                    logger.error(\n+                    logger.warning(\n                         'value of data element \"{}\" with VR Person Name (PN) '\n                         'is not formatted correctly'.format(tag)\n                     )\n                     elem_value.append(v)\n                 else:\n-                    elem_value.extend(list(v.values()))\n-            if vm == '1':\n+                    if 'Phonetic' in v:\n+                        comps = ['', '', '']\n+                    elif 'Ideographic' in v:\n+                        comps = ['', '']\n+                    else:\n+                        comps = ['']\n+                    if 'Alphabetic' in v:\n+                        comps[0] = v['Alphabetic']\n+                    if 'Ideographic' in v:\n+                        comps[1] = v['Ideographic']\n+                    if 'Phonetic' in v:\n+                        comps[2] = v['Phonetic']\n+                    elem_value.append('='.join(comps))\n+            if len(elem_value) == 1:\n+                elem_value = elem_value[0]\n+            elif not elem_value:\n+                elem_value = empty_value_for_VR(vr)\n+        elif vr == 'AT':\n+            elem_value = []\n+            for v in value:\n                 try:\n-                    elem_value = elem_value[0]\n-                except IndexError:\n-                    elem_value = ''\n+                    elem_value.append(int(v, 16))\n+                except ValueError:\n+                    warnings.warn('Invalid value \"{}\" for AT element - '\n+                                  'ignoring it'.format(v))\n+                value = value[0]\n+            if not elem_value:\n+                elem_value = empty_value_for_VR(vr)\n+            elif len(elem_value) == 1:\n+                elem_value = elem_value[0]\n         else:\n-            if vm == '1':\n-                if value_key == 'InlineBinary':\n-                    elem_value = base64.b64decode(value)\n-                elif value_key == 'BulkDataURI':\n-                    if bulk_data_uri_handler is None:\n-                        logger.warning(\n-                            'no bulk data URI handler provided for retrieval '\n-                            'of value of data element \"{}\"'.format(tag)\n-                        )\n-                        elem_value = b''\n-                    else:\n-                        elem_value = bulk_data_uri_handler(value)\n+            if isinstance(value, list) and len(value) == 1:\n+                value = value[0]\n+            if value_key == 'InlineBinary':\n+                elem_value = base64.b64decode(value)\n+            elif value_key == 'BulkDataURI':\n+                if bulk_data_uri_handler is None:\n+                    logger.warning(\n+                        'no bulk data URI handler provided for retrieval '\n+                        'of value of data element \"{}\"'.format(tag)\n+                    )\n+                    elem_value = empty_value_for_VR(vr, raw=True)\n                 else:\n-                    if value:\n-                        elem_value = value[0]\n-                    else:\n-                        elem_value = value\n+                    elem_value = bulk_data_uri_handler(value)\n             else:\n                 elem_value = value\n         if elem_value is None:\n-            logger.warning('missing value for data element \"{}\"'.format(tag))\n-            elem_value = ''\n+            elem_value = empty_value_for_VR(vr)\n \n         elem_value = jsonrep.convert_to_python_number(elem_value, vr)\n \n@@ -404,11 +423,9 @@ def to_json(self, bulk_data_element_handler,\n             but `bulk_data_element_handler` is ``None`` and hence not callable\n \n         \"\"\"\n-        # TODO: Determine whether more VRs need to be converted to strings\n-        _VRs_TO_QUOTE = ['AT', ]\n         json_element = {'vr': self.VR, }\n         if self.VR in jsonrep.BINARY_VR_VALUES:\n-            if self.value is not None:\n+            if not self.is_empty:\n                 binary_value = self.value\n                 encoded_value = base64.b64encode(binary_value).decode('utf-8')\n                 if len(encoded_value) > bulk_data_threshold:\n@@ -440,35 +457,35 @@ def to_json(self, bulk_data_element_handler,\n             ]\n             json_element['Value'] = value\n         elif self.VR == 'PN':\n-            elem_value = self.value\n-            if elem_value is not None:\n-                if compat.in_py2:\n-                    elem_value = PersonNameUnicode(elem_value, 'UTF8')\n-                if len(elem_value.components) > 2:\n-                    json_element['Value'] = [\n-                        {'Phonetic': elem_value.components[2], },\n-                    ]\n-                elif len(elem_value.components) > 1:\n-                    json_element['Value'] = [\n-                        {'Ideographic': elem_value.components[1], },\n-                    ]\n+            if not self.is_empty:\n+                elem_value = []\n+                if self.VM > 1:\n+                    value = self.value\n                 else:\n-                    json_element['Value'] = [\n-                        {'Alphabetic': elem_value.components[0], },\n-                    ]\n+                    value = [self.value]\n+                for v in value:\n+                    if compat.in_py2:\n+                        v = PersonNameUnicode(v, 'UTF8')\n+                    comps = {'Alphabetic': v.components[0]}\n+                    if len(v.components) > 1:\n+                        comps['Ideographic'] = v.components[1]\n+                    if len(v.components) > 2:\n+                        comps['Phonetic'] = v.components[2]\n+                    elem_value.append(comps)\n+                json_element['Value'] = elem_value\n+        elif self.VR == 'AT':\n+            if not self.is_empty:\n+                value = self.value\n+                if self.VM == 1:\n+                    value = [value]\n+                json_element['Value'] = [format(v, '08X') for v in value]\n         else:\n-            if self.value is not None:\n-                is_multivalue = isinstance(self.value, MultiValue)\n-                if self.VM > 1 or is_multivalue:\n+            if not self.is_empty:\n+                if self.VM > 1:\n                     value = self.value\n                 else:\n                     value = [self.value]\n-                # ensure it's a list and not another iterable\n-                # (e.g. tuple), which would not be JSON serializable\n-                if self.VR in _VRs_TO_QUOTE:\n-                    json_element['Value'] = [str(v) for v in value]\n-                else:\n-                    json_element['Value'] = [v for v in value]\n+                json_element['Value'] = [v for v in value]\n         if hasattr(json_element, 'Value'):\n             json_element['Value'] = jsonrep.convert_to_python_number(\n                 json_element['Value'], self.VR\ndiff --git a/pydicom/jsonrep.py b/pydicom/jsonrep.py\n--- a/pydicom/jsonrep.py\n+++ b/pydicom/jsonrep.py\n@@ -2,6 +2,8 @@\n \"\"\"Methods for converting Datasets and DataElements to/from json\"\"\"\n \n # Order of keys is significant!\n+from pydicom.compat import int_type\n+\n JSON_VALUE_KEYS = ('Value', 'BulkDataURI', 'InlineBinary',)\n \n BINARY_VR_VALUES = ['OW', 'OB', 'OD', 'OF', 'OL', 'UN',\n@@ -30,7 +32,7 @@ def convert_to_python_number(value, vr):\n         return None\n     number_type = None\n     if vr in VRs_TO_BE_INTS:\n-        number_type = int\n+        number_type = int_type\n     if vr in VRs_TO_BE_FLOATS:\n         number_type = float\n     if number_type is not None:\ndiff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -646,9 +646,13 @@ def __init__(self, val, encodings=None, original_string=None):\n             # this is the decoded string - save the original string if\n             # available for easier writing back\n             self.original_string = original_string\n-            self._components = tuple(val.split('='))\n+            components = val.split('=')\n+            # Remove empty elements from the end to avoid trailing '='\n+            while len(components) and not components[-1]:\n+                components.pop()\n+            self._components = tuple(components)\n \n-        # if the encoding is not given, leave it as undefined (None)\n+            # if the encoding is not given, leave it as undefined (None)\n         self.encodings = _verify_encodings(encodings)\n         self._dict = {}\n \n@@ -812,7 +816,7 @@ def formatted(self, format_str):\n \n     def __bool__(self):\n         if self.original_string is None:\n-            return (self._components is not None and\n+            return (bool(self._components) and\n                     (len(self._components) > 1 or bool(self._components[0])))\n         return bool(self.original_string)\n \n", "test_patch": "diff --git a/pydicom/tests/test_json.py b/pydicom/tests/test_json.py\n--- a/pydicom/tests/test_json.py\n+++ b/pydicom/tests/test_json.py\n@@ -1,89 +1,226 @@\n # -*- coding: utf-8 -*-\n # Copyright 2008-2019 pydicom authors. See LICENSE file for details.\n+import json\n+\n import pytest\n+\n+from pydicom import dcmread, compat\n from pydicom.data import get_testdata_files\n from pydicom.dataelem import DataElement\n from pydicom.dataset import Dataset\n-from pydicom import compat\n-\n+from pydicom.tag import Tag, BaseTag\n from pydicom.valuerep import PersonNameUnicode, PersonName3\n \n \n-def test_json_PN():\n-    s = open(get_testdata_files(\"test_PN.json\")[0], \"r\").read()\n-    ds = Dataset.from_json(s)\n-    assert isinstance(ds[0x00080090].value,\n-                      (PersonNameUnicode, PersonName3))\n-    assert isinstance(ds[0x00100010].value,\n-                      (PersonNameUnicode, PersonName3))\n-    inner_seq = ds[0x04000561].value[0][0x04000550]\n-    dataelem = inner_seq[0][0x00100010]\n-    assert isinstance(dataelem.value, (PersonNameUnicode, PersonName3))\n-\n-\n-@pytest.mark.skipif(compat.in_py2,\n-                    reason='JSON conversion not yet working in Python 2')\n-def test_dataelem_from_json():\n-    tag = 0x0080090\n-    vr = \"PN\"\n-    value = [{\"Alphabetic\": \"\"}]\n-    dataelem = DataElement.from_json(Dataset, tag, vr, value, \"Value\")\n-    assert isinstance(dataelem.value, (PersonNameUnicode, PersonName3))\n-\n-\n-def test_json_roundtrip():\n-    ds = Dataset()\n-    ds.add_new(0x00080005, 'CS', 'ISO_IR 100')\n-    ds.add_new(0x00090010, 'LO', 'Creator 1.0')\n-    ds.add_new(0x00091001, 'SH', 'Version1')\n-    ds.add_new(0x00091002, 'OB', b'BinaryContent')\n-    ds.add_new(0x00091003, 'OW', b'\\x0102\\x3040\\x5060')\n-    ds.add_new(0x00091004, 'OF', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07')\n-    ds.add_new(0x00091005, 'OD', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07'\n-                                 b'\\x01\\x01\\x02\\x03\\x04\\x05\\x06\\x07')\n-    ds.add_new(0x00091006, 'OL', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07'\n-                                 b'\\x01\\x01\\x02\\x03')\n-    ds.add_new(0x00091007, 'UI', '1.2.3.4.5.6')\n-    ds.add_new(0x00091008, 'DA', '20200101')\n-    ds.add_new(0x00091009, 'TM', '115500')\n-    ds.add_new(0x0009100a, 'DT', '20200101115500.000000')\n-    ds.add_new(0x0009100b, 'UL', 3000000000)\n-    ds.add_new(0x0009100c, 'SL', -2000000000)\n-    ds.add_new(0x0009100d, 'US', 40000)\n-    ds.add_new(0x0009100e, 'SS', -22222)\n-    ds.add_new(0x0009100f, 'FL', 3.14)\n-    ds.add_new(0x00091010, 'FD', 3.14159265)\n-    ds.add_new(0x00091011, 'CS', 'TEST MODE')\n-    ds.add_new(0x00091012, 'PN', 'CITIZEN^1')\n-    ds.add_new(0x00091013, 'PN', u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046')\n-    ds.add_new(0x00091014, 'IS', '42')\n-    ds.add_new(0x00091015, 'DS', '3.14159265')\n-    ds.add_new(0x00091016, 'AE', b'CONQUESTSRV1')\n-    ds.add_new(0x00091017, 'AS', '055Y')\n-    ds.add_new(0x00091018, 'LT', 50 * u'\u041a\u0430\u043b\u0438\u043d\u043a\u0430,')\n-    ds.add_new(0x00091019, 'UC', 'LONG CODE VALUE')\n-    ds.add_new(0x0009101a, 'UN', b'\\x0102\\x3040\\x5060')\n-    ds.add_new(0x0009101b, 'UR', 'https://example.com')\n-    ds.add_new(0x0009101c, 'AT', b'\\x10\\x00\\x20\\x00\\x10\\x00\\x30\\x00\\x10')\n-    ds.add_new(0x0009101d, 'ST', 100 * u'\u0639\u0644\u064a \u0628\u0627\u0628\u0627')\n-    ds.add_new(0x0009101e, 'SH', u'\u0394\u03b9\u03bf\u03bd\u03c5\u03c3\u03b9\u03bf\u03c2')\n-    ds.add_new(0x00090011, 'LO', 'Creator 2.0')\n-    ds.add_new(0x00091101, 'SH', 'Version2')\n-    ds.add_new(0x00091102, 'US', 2)\n-\n-    jsonmodel = ds.to_json(bulk_data_threshold=100)\n-    ds2 = Dataset.from_json(jsonmodel)\n-\n-    assert ds2.SpecificCharacterSet == ['ISO_IR 100']\n-\n-\n-def test_json_private_DS_VM():\n-    test1_json = get_testdata_files(\"test1.json\")[0]\n-    jsonmodel = open(test1_json, 'r').read()\n-    ds = Dataset.from_json(jsonmodel)\n-    import json\n-    jsonmodel2 = ds.to_json(dump_handler=lambda d: json.dumps(d, indent=2))\n-    ds2 = Dataset.from_json(jsonmodel2)\n-\n-    assert ds.PatientIdentityRemoved == 'YES'\n-    assert ds2.PatientIdentityRemoved == 'YES'\n+class TestPersonName(object):\n+    def test_json_PN_from_file(self):\n+        with open(get_testdata_files(\"test_PN.json\")[0]) as s:\n+            ds = Dataset.from_json(s.read())\n+        assert isinstance(ds[0x00080090].value,\n+                          (PersonNameUnicode, PersonName3))\n+        assert isinstance(ds[0x00100010].value,\n+                          (PersonNameUnicode, PersonName3))\n+        inner_seq = ds[0x04000561].value[0][0x04000550]\n+        dataelem = inner_seq[0][0x00100010]\n+        assert isinstance(dataelem.value, (PersonNameUnicode, PersonName3))\n+\n+    def test_PN_components_to_json(self):\n+        def check_name(tag, components):\n+            # we cannot directly compare the dictionaries, as they are not\n+            # ordered in Python 2\n+            value = ds_json[tag]['Value']\n+            assert 1 == len(value)\n+            value = value[0]\n+            if len(components) == 3:\n+                assert components[2] == value['Phonetic']\n+            else:\n+                assert 'Phonetic' not in value\n+            if len(components) >= 2:\n+                assert components[1] == value['Ideographic']\n+            else:\n+                assert 'Ideographic' not in value\n+            assert components[0] == value['Alphabetic']\n+\n+        ds = Dataset()\n+        ds.add_new(0x00100010, 'PN', u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046')\n+        ds.add_new(0x00091001, 'PN', u'Yamada^Tarou')\n+        ds.add_new(0x00091002, 'PN', u'Yamada^Tarou==')\n+        ds.add_new(0x00091003, 'PN', u'=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046')\n+        ds.add_new(0x00091004, 'PN', u'Yamada^Tarou==\u3084\u307e\u3060^\u305f\u308d\u3046')\n+        ds.add_new(0x00091005, 'PN', u'==\u3084\u307e\u3060^\u305f\u308d\u3046')\n+        ds.add_new(0x00091006, 'PN', u'=\u5c71\u7530^\u592a\u90ce')\n+        ds.add_new(0x00091007, 'PN', u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce')\n+        ds_json = json.loads(ds.to_json())\n+        check_name('00100010', ['Yamada^Tarou', u'\u5c71\u7530^\u592a\u90ce', u'\u3084\u307e\u3060^\u305f\u308d\u3046'])\n+        check_name('00091001', ['Yamada^Tarou'])\n+        check_name('00091002', ['Yamada^Tarou'])\n+        check_name('00091003', ['', u'\u5c71\u7530^\u592a\u90ce', u'\u3084\u307e\u3060^\u305f\u308d\u3046'])\n+        check_name('00091004', ['Yamada^Tarou', '', u'\u3084\u307e\u3060^\u305f\u308d\u3046'])\n+        check_name('00091005', ['', '', u'\u3084\u307e\u3060^\u305f\u308d\u3046'])\n+        check_name('00091006', ['', u'\u5c71\u7530^\u592a\u90ce'])\n+        check_name('00091007', ['Yamada^Tarou', u'\u5c71\u7530^\u592a\u90ce'])\n+\n+    def test_PN_components_from_json(self):\n+        # this is the encoded dataset from the previous test, with some\n+        # empty components omitted\n+        ds_json = (u'{\"00100010\": {\"vr\": \"PN\", \"Value\": [{\"Alphabetic\": '\n+                   u'\"Yamada^Tarou\", \"Ideographic\": \"\u5c71\u7530^\u592a\u90ce\", '\n+                   u'\"Phonetic\": \"\u3084\u307e\u3060^\u305f\u308d\u3046\"}]}, '\n+                   u'\"00091001\": {\"vr\": \"PN\", \"Value\": '\n+                   u'[{\"Alphabetic\": \"Yamada^Tarou\"}]}, '\n+                   u'\"00091002\": {\"vr\": \"PN\", \"Value\": '\n+                   u'[{\"Alphabetic\": \"Yamada^Tarou\", \"Ideographic\": \"\", '\n+                   u'\"Phonetic\": \"\"}]}, '\n+                   u'\"00091003\": {\"vr\": \"PN\", \"Value\": [{'\n+                   u'\"Ideographic\": \"\u5c71\u7530^\u592a\u90ce\", '\n+                   u'\"Phonetic\": \"\u3084\u307e\u3060^\u305f\u308d\u3046\"}]}, '\n+                   u'\"00091004\": {\"vr\": \"PN\", \"Value\": '\n+                   u'[{\"Alphabetic\": \"Yamada^Tarou\", '\n+                   u'\"Phonetic\": \"\u3084\u307e\u3060^\u305f\u308d\u3046\"}]}, '\n+                   u'\"00091005\": {\"vr\": \"PN\", \"Value\": '\n+                   u'[{\"Phonetic\": \"\u3084\u307e\u3060^\u305f\u308d\u3046\"}]}, '\n+                   u'\"00091006\": {\"vr\": \"PN\", \"Value\":'\n+                   u' [{\"Ideographic\": \"\u5c71\u7530^\u592a\u90ce\"}]}, '\n+                   u'\"00091007\": {\"vr\": \"PN\", \"Value\": '\n+                   u'[{\"Alphabetic\": \"Yamada^Tarou\", '\n+                   u'\"Ideographic\": \"\u5c71\u7530^\u592a\u90ce\"}]}}')\n+        if compat.in_py2:\n+            ds_json = ds_json.encode('UTF8')\n+\n+        ds = Dataset.from_json(ds_json)\n+        assert u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046' == ds.PatientName\n+        assert u'Yamada^Tarou' == ds[0x00091001].value\n+        assert u'Yamada^Tarou' == ds[0x00091002].value\n+        assert u'=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046' == ds[0x00091003].value\n+        assert u'Yamada^Tarou==\u3084\u307e\u3060^\u305f\u308d\u3046' == ds[0x00091004].value\n+        assert u'==\u3084\u307e\u3060^\u305f\u308d\u3046' == ds[0x00091005].value\n+        assert u'=\u5c71\u7530^\u592a\u90ce' == ds[0x00091006].value\n+        assert u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce' == ds[0x00091007].value\n+\n+    def test_empty_value(self):\n+        ds = Dataset()\n+        ds.add_new(0x00100010, 'PN', '')\n+        ds_json = json.loads(ds.to_json())\n+        assert '00100010' in ds_json\n+        assert 'Value' not in ds_json['00100010']\n+\n+    def test_multi_value_to_json(self):\n+        ds = Dataset()\n+        patient_names = [u'Buc^J\u00e9r\u00f4me', u'\u0394\u03b9\u03bf\u03bd\u03c5\u03c3\u03b9\u03bf\u03c2', u'\u041b\u044e\u043ace\u043c\u0431yp\u0433']\n+        ds.add_new(0x00091001, 'PN', patient_names)\n+        ds_json = json.loads(ds.to_json())\n+        assert [{'Alphabetic': u'Buc^J\u00e9r\u00f4me'},\n+                {'Alphabetic': u'\u0394\u03b9\u03bf\u03bd\u03c5\u03c3\u03b9\u03bf\u03c2'},\n+                {'Alphabetic': u'\u041b\u044e\u043ace\u043c\u0431yp\u0433'}] == ds_json['00091001']['Value']\n+\n+    def test_dataelem_from_json(self):\n+        tag = 0x0080090\n+        vr = \"PN\"\n+        value = [{\"Alphabetic\": \"\"}]\n+        dataelem = DataElement.from_json(Dataset, tag, vr, value, \"Value\")\n+        assert isinstance(dataelem.value, (PersonNameUnicode, PersonName3))\n+\n+\n+class TestAT(object):\n+    def test_to_json(self):\n+        ds = Dataset()\n+        ds.add_new(0x00091001, 'AT', [0x00100010, 0x00100020])\n+        ds.add_new(0x00091002, 'AT', Tag(0x28, 0x02))\n+        ds.add_new(0x00091003, 'AT', BaseTag(0x00280002))\n+        ds.add_new(0x00091004, 'AT', [0x00280002, Tag('PatientName')])\n+        ds_json = json.loads(ds.to_json())\n+\n+        assert ['00100010', '00100020'] == ds_json['00091001']['Value']\n+        assert ['00280002'] == ds_json['00091002']['Value']\n+        assert ['00280002'] == ds_json['00091003']['Value']\n+        assert ['00280002', '00100010'] == ds_json['00091004']['Value']\n+\n+    def test_from_json(self):\n+        ds_json = ('{\"00091001\": {\"vr\": \"AT\", \"Value\": [\"000910AF\"]}, '\n+                   '\"00091002\": {\"vr\": \"AT\", \"Value\": [\"00100010\", '\n+                   '\"00100020\", \"00100030\"]}}')\n+        ds = Dataset.from_json(ds_json)\n+        assert 0x000910AF == ds[0x00091001].value\n+        assert [0x00100010, 0x00100020, 0x00100030] == ds[0x00091002].value\n+\n+    def test_invalid_json(self):\n+        ds_json = ('{\"00091001\": {\"vr\": \"AT\", \"Value\": [\"000910AG\"]}, '\n+                   '\"00091002\": {\"vr\": \"AT\", \"Value\": [\"00100010\"]}}')\n+        with pytest.warns(UserWarning, match='Invalid value \"000910AG\" for '\n+                                             'AT element - ignoring it'):\n+            ds = Dataset.from_json(ds_json)\n+            assert ds[0x00091001].value is None\n+            assert 0x00100010 == ds[0x00091002].value\n+\n+\n+class TestDataSetToJson(object):\n+    def test_json_from_dicom_file(self):\n+        ds1 = Dataset(dcmread(get_testdata_files(\"CT_small.dcm\")[0]))\n+        ds_json = ds1.to_json(bulk_data_threshold=100000)\n+        ds2 = Dataset.from_json(ds_json)\n+        assert ds1 == ds2\n+\n+    def test_roundtrip(self):\n+        ds = Dataset()\n+        ds.add_new(0x00080005, 'CS', 'ISO_IR 100')\n+        ds.add_new(0x00090010, 'LO', 'Creator 1.0')\n+        ds.add_new(0x00091001, 'SH', 'Version1')\n+        ds.add_new(0x00091002, 'OB', b'BinaryContent')\n+        ds.add_new(0x00091003, 'OW', b'\\x0102\\x3040\\x5060')\n+        ds.add_new(0x00091004, 'OF', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07')\n+        ds.add_new(0x00091005, 'OD', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07'\n+                                     b'\\x01\\x01\\x02\\x03\\x04\\x05\\x06\\x07')\n+        ds.add_new(0x00091006, 'OL', b'\\x00\\x01\\x02\\x03\\x04\\x05\\x06\\x07'\n+                                     b'\\x01\\x01\\x02\\x03')\n+        ds.add_new(0x00091007, 'UI', '1.2.3.4.5.6')\n+        ds.add_new(0x00091008, 'DA', '20200101')\n+        ds.add_new(0x00091009, 'TM', '115500')\n+        ds.add_new(0x0009100a, 'DT', '20200101115500.000000')\n+        ds.add_new(0x0009100b, 'UL', 3000000000)\n+        ds.add_new(0x0009100c, 'SL', -2000000000)\n+        ds.add_new(0x0009100d, 'US', 40000)\n+        ds.add_new(0x0009100e, 'SS', -22222)\n+        ds.add_new(0x0009100f, 'FL', 3.14)\n+        ds.add_new(0x00091010, 'FD', 3.14159265)\n+        ds.add_new(0x00091011, 'CS', 'TEST MODE')\n+        ds.add_new(0x00091012, 'PN', 'CITIZEN^1')\n+        ds.add_new(0x00091013, 'PN', u'Yamada^Tarou=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046')\n+        ds.add_new(0x00091014, 'IS', '42')\n+        ds.add_new(0x00091015, 'DS', '3.14159265')\n+        ds.add_new(0x00091016, 'AE', b'CONQUESTSRV1')\n+        ds.add_new(0x00091017, 'AS', '055Y')\n+        ds.add_new(0x00091018, 'LT', 50 * u'\u041a\u0430\u043b\u0438\u043d\u043a\u0430,')\n+        ds.add_new(0x00091019, 'UC', 'LONG CODE VALUE')\n+        ds.add_new(0x0009101a, 'UN', b'\\x0102\\x3040\\x5060')\n+        ds.add_new(0x0009101b, 'UR', 'https://example.com')\n+        ds.add_new(0x0009101c, 'AT', [0x00100010, 0x00100020])\n+        ds.add_new(0x0009101d, 'ST', 100 * u'\u0639\u0644\u064a \u0628\u0627\u0628\u0627')\n+        ds.add_new(0x0009101e, 'SH', u'\u0394\u03b9\u03bf\u03bd\u03c5\u03c3\u03b9\u03bf\u03c2')\n+        ds.add_new(0x00090011, 'LO', 'Creator 2.0')\n+        ds.add_new(0x00091101, 'SH', 'Version2')\n+        ds.add_new(0x00091102, 'US', 2)\n+\n+        json_string = ds.to_json(bulk_data_threshold=100)\n+        json_model = json.loads(json_string)\n+        assert json_model['00080005']['Value'] == ['ISO_IR 100']\n+        assert json_model['00091007']['Value'] == ['1.2.3.4.5.6']\n+        assert json_model['0009100A']['Value'] == ['20200101115500.000000']\n+        assert json_model['0009100B']['Value'] == [3000000000]\n+        assert json_model['0009100C']['Value'] == [-2000000000]\n+        assert json_model['0009100D']['Value'] == [40000]\n+        assert json_model['0009100F']['Value'] == [3.14]\n+        assert json_model['00091010']['Value'] == [3.14159265]\n+        assert json_model['00091018']['Value'] == [50 * u'\u041a\u0430\u043b\u0438\u043d\u043a\u0430,']\n+\n+        ds2 = Dataset.from_json(json_string)\n+        assert ds == ds2\n+\n+    def test_json_private_DS_VM(self):\n+        test1_json = get_testdata_files(\"test1.json\")[0]\n+        jsonmodel = open(test1_json, 'r').read()\n+        ds = Dataset.from_json(jsonmodel)\n+        jsonmodel2 = ds.to_json(dump_handler=lambda d: json.dumps(d, indent=2))\n+        ds2 = Dataset.from_json(jsonmodel2)\n+\n+        assert ds.PatientIdentityRemoved == 'YES'\n+        assert ds2.PatientIdentityRemoved == 'YES'\n", "problem_statement": "To_Json 'str' object has no attribute 'components'\n<!-- Instructions For Filing a Bug: https://github.com/pydicom/pydicom/blob/master/CONTRIBUTING.md#filing-bugs -->\r\n\r\n#### Description\r\n<!-- Example: Attribute Error thrown when printing (0x0010, 0x0020) patient Id> 0-->\r\n\r\nWhen converting a dataset to json the following error occurs.\r\n```\r\nTraceback (most recent call last):\r\n  File \"/anaconda3/lib/python3.6/threading.py\", line 916, in _bootstrap_inner\r\n    self.run()\r\n  File \"/anaconda3/lib/python3.6/threading.py\", line 864, in run\r\n    self._target(*self._args, **self._kwargs)\r\n  File \"~/pacs-proxy/pacs/service.py\", line 172, in saveFunction\r\n    jsonObj = ds.to_json()\r\n  File \"~/lib/python3.6/site-packages/pydicom/dataset.py\", line 2046, in to_json\r\n    dump_handler=dump_handler\r\n  File \"~/lib/python3.6/site-packages/pydicom/dataelem.py\", line 447, in to_json\r\n    if len(elem_value.components) > 2:\r\nAttributeError: 'str' object has no attribute 'components'\r\n```\r\n#### Steps/Code to Reproduce\r\n\r\nds = pydicom.dcmread(\"testImg\")\r\njsonObj = ds.to_json()\r\n\r\nI'm working on getting an anonymous version of the image, will update. But any advice, suggestions would be appreciated.\r\n\r\n#### \n", "hints_text": "Thanks for the report! \r\nFor some reason, the value is not a `PersonName3` instance, as expected, but a string - this is obviously a bug. No need to provide a test dataset, as this is reproducible with existing test data. I will have a look tonight.\nAwesome. Thank you very much. This is the last hurdle for our project. :D", "created_at": "2019-08-06T19:27:04Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_json.py::TestPersonName::test_PN_components_to_json", "pydicom/tests/test_json.py::TestPersonName::test_PN_components_from_json", "pydicom/tests/test_json.py::TestPersonName::test_empty_value", "pydicom/tests/test_json.py::TestPersonName::test_multi_value_to_json", "pydicom/tests/test_json.py::TestAT::test_to_json", "pydicom/tests/test_json.py::TestAT::test_from_json", "pydicom/tests/test_json.py::TestAT::test_invalid_json", "pydicom/tests/test_json.py::TestDataSetToJson::test_json_from_dicom_file", "pydicom/tests/test_json.py::TestDataSetToJson::test_roundtrip"], "PASS_TO_PASS": ["pydicom/tests/test_json.py::TestPersonName::test_json_PN_from_file", "pydicom/tests/test_json.py::TestPersonName::test_dataelem_from_json", "pydicom/tests/test_json.py::TestDataSetToJson::test_json_private_DS_VM"], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-823", "base_commit": "a5c24f01e9b420968f5fda0d5c46ce2a4cf2c867", "patch": "diff --git a/pvlib/tracking.py b/pvlib/tracking.py\n--- a/pvlib/tracking.py\n+++ b/pvlib/tracking.py\n@@ -1,7 +1,7 @@\n import numpy as np\n import pandas as pd\n \n-from pvlib.tools import cosd, sind\n+from pvlib.tools import cosd, sind, tand\n from pvlib.pvsystem import _combine_localized_attributes\n from pvlib.pvsystem import PVSystem\n from pvlib.location import Location\n@@ -10,8 +10,9 @@\n \n class SingleAxisTracker(PVSystem):\n     \"\"\"\n-    Inherits the PV modeling methods from :py:class:`~pvlib.pvsystem.PVSystem`.\n-\n+    A class for single-axis trackers that inherits the PV modeling methods from\n+    :py:class:`~pvlib.pvsystem.PVSystem`. For details on calculating tracker\n+    rotation see :py:func:`pvlib.tracking.singleaxis`.\n \n     Parameters\n     ----------\n@@ -21,7 +22,7 @@ class SingleAxisTracker(PVSystem):\n \n     axis_azimuth : float, default 0\n         A value denoting the compass direction along which the axis of\n-        rotation lies. Measured in decimal degrees East of North.\n+        rotation lies. Measured in decimal degrees east of north.\n \n     max_angle : float, default 90\n         A value denoting the maximum rotation angle, in decimal degrees,\n@@ -43,16 +44,33 @@ class SingleAxisTracker(PVSystem):\n         between the tracking axes has a gcr of 2/6=0.333. If gcr is not\n         provided, a gcr of 2/7 is default. gcr must be <=1.\n \n+    cross_axis_tilt : float, default 0.0\n+        The angle, relative to horizontal, of the line formed by the\n+        intersection between the slope containing the tracker axes and a plane\n+        perpendicular to the tracker axes. Cross-axis tilt should be specified\n+        using a right-handed convention. For example, trackers with axis\n+        azimuth of 180 degrees (heading south) will have a negative cross-axis\n+        tilt if the tracker axes plane slopes down to the east and positive\n+        cross-axis tilt if the tracker axes plane slopes up to the east. Use\n+        :func:`~pvlib.tracking.calc_cross_axis_tilt` to calculate\n+        `cross_axis_tilt`. [degrees]\n+\n+    See also\n+    --------\n+    pvlib.tracking.singleaxis\n+    pvlib.tracking.calc_axis_tilt\n+    pvlib.tracking.calc_cross_axis_tilt\n     \"\"\"\n \n-    def __init__(self, axis_tilt=0, axis_azimuth=0,\n-                 max_angle=90, backtrack=True, gcr=2.0/7.0, **kwargs):\n+    def __init__(self, axis_tilt=0, axis_azimuth=0, max_angle=90,\n+                 backtrack=True, gcr=2.0/7.0, cross_axis_tilt=0.0, **kwargs):\n \n         self.axis_tilt = axis_tilt\n         self.axis_azimuth = axis_azimuth\n         self.max_angle = max_angle\n         self.backtrack = backtrack\n         self.gcr = gcr\n+        self.cross_axis_tilt = cross_axis_tilt\n \n         kwargs['surface_tilt'] = None\n         kwargs['surface_azimuth'] = None\n@@ -60,7 +78,8 @@ def __init__(self, axis_tilt=0, axis_azimuth=0,\n         super(SingleAxisTracker, self).__init__(**kwargs)\n \n     def __repr__(self):\n-        attrs = ['axis_tilt', 'axis_azimuth', 'max_angle', 'backtrack', 'gcr']\n+        attrs = ['axis_tilt', 'axis_azimuth', 'max_angle', 'backtrack', 'gcr',\n+                 'cross_axis_tilt']\n         sat_repr = ('SingleAxisTracker:\\n  ' + '\\n  '.join(\n             ('{}: {}'.format(attr, getattr(self, attr)) for attr in attrs)))\n         # get the parent PVSystem info\n@@ -88,8 +107,8 @@ def singleaxis(self, apparent_zenith, apparent_azimuth):\n         \"\"\"\n         tracking_data = singleaxis(apparent_zenith, apparent_azimuth,\n                                    self.axis_tilt, self.axis_azimuth,\n-                                   self.max_angle,\n-                                   self.backtrack, self.gcr)\n+                                   self.max_angle, self.backtrack,\n+                                   self.gcr, self.cross_axis_tilt)\n \n         return tracking_data\n \n@@ -215,13 +234,15 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n \n class LocalizedSingleAxisTracker(SingleAxisTracker, Location):\n     \"\"\"\n-    The LocalizedSingleAxisTracker class defines a standard set of\n-    installed PV system attributes and modeling functions. This class\n-    combines the attributes and methods of the SingleAxisTracker (a\n-    subclass of PVSystem) and Location classes.\n-\n-    The LocalizedSingleAxisTracker may have bugs due to the difficulty\n-    of robustly implementing multiple inheritance. See\n+    The :py:class:`~pvlib.tracking.LocalizedSingleAxisTracker` class defines a\n+    standard set of installed PV system attributes and modeling functions. This\n+    class combines the attributes and methods of the\n+    :py:class:`~pvlib.tracking.SingleAxisTracker` (a subclass of\n+    :py:class:`~pvlib.pvsystem.PVSystem`) and\n+    :py:class:`~pvlib.location.Location` classes.\n+\n+    The :py:class:`~pvlib.tracking.LocalizedSingleAxisTracker` may have bugs\n+    due to the difficulty of robustly implementing multiple inheritance. See\n     :py:class:`~pvlib.modelchain.ModelChain` for an alternative paradigm\n     for modeling PV systems at specific locations.\n     \"\"\"\n@@ -247,25 +268,25 @@ def __repr__(self):\n \n def singleaxis(apparent_zenith, apparent_azimuth,\n                axis_tilt=0, axis_azimuth=0, max_angle=90,\n-               backtrack=True, gcr=2.0/7.0):\n+               backtrack=True, gcr=2.0/7.0, cross_axis_tilt=0):\n     \"\"\"\n-    Determine the rotation angle of a single axis tracker when given a\n-    particular sun zenith and azimuth angle. See [1]_ for details about\n-    the equations.\n-    Backtracking may be specified, and if so, a ground coverage\n-    ratio is required.\n-\n-    Rotation angle is determined in a panel-oriented coordinate system.\n-    The tracker azimuth axis_azimuth defines the positive y-axis; the\n-    positive x-axis is 90 degress clockwise from the y-axis and parallel\n-    to the earth surface, and the positive z-axis is normal and oriented\n-    towards the sun. Rotation angle tracker_theta indicates tracker\n-    position relative to horizontal: tracker_theta = 0 is horizontal,\n-    and positive tracker_theta is a clockwise rotation around the y axis\n-    in the x, y, z coordinate system. For example, if tracker azimuth\n-    axis_azimuth is 180 (oriented south), tracker_theta = 30 is a\n-    rotation of 30 degrees towards the west, and tracker_theta = -90 is\n-    a rotation to the vertical plane facing east.\n+    Determine the rotation angle of a single-axis tracker when given particular\n+    solar zenith and azimuth angles.\n+\n+    See [1]_ for details about the equations. Backtracking may be specified,\n+    and if so, a ground coverage ratio is required.\n+\n+    Rotation angle is determined in a right-handed coordinate system. The\n+    tracker `axis_azimuth` defines the positive y-axis, the positive x-axis is\n+    90 degrees clockwise from the y-axis and parallel to the Earth's surface,\n+    and the positive z-axis is normal to both x & y-axes and oriented skyward.\n+    Rotation angle `tracker_theta` is a right-handed rotation around the y-axis\n+    in the x, y, z coordinate system and indicates tracker position relative to\n+    horizontal. For example, if tracker `axis_azimuth` is 180 (oriented south)\n+    and `axis_tilt` is zero, then a `tracker_theta` of zero is horizontal, a\n+    `tracker_theta` of 30 degrees is a rotation of 30 degrees towards the west,\n+    and a `tracker_theta` of -90 degrees is a rotation to the vertical plane\n+    facing east.\n \n     Parameters\n     ----------\n@@ -281,7 +302,7 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n \n     axis_azimuth : float, default 0\n         A value denoting the compass direction along which the axis of\n-        rotation lies. Measured in decimal degrees East of North.\n+        rotation lies. Measured in decimal degrees east of north.\n \n     max_angle : float, default 90\n         A value denoting the maximum rotation angle, in decimal degrees,\n@@ -303,24 +324,41 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n         between the tracking axes has a gcr of 2/6=0.333. If gcr is not\n         provided, a gcr of 2/7 is default. gcr must be <=1.\n \n+    cross_axis_tilt : float, default 0.0\n+        The angle, relative to horizontal, of the line formed by the\n+        intersection between the slope containing the tracker axes and a plane\n+        perpendicular to the tracker axes. Cross-axis tilt should be specified\n+        using a right-handed convention. For example, trackers with axis\n+        azimuth of 180 degrees (heading south) will have a negative cross-axis\n+        tilt if the tracker axes plane slopes down to the east and positive\n+        cross-axis tilt if the tracker axes plane slopes up to the east. Use\n+        :func:`~pvlib.tracking.calc_cross_axis_tilt` to calculate\n+        `cross_axis_tilt`. [degrees]\n+\n     Returns\n     -------\n     dict or DataFrame with the following columns:\n         * `tracker_theta`: The rotation angle of the tracker.\n           tracker_theta = 0 is horizontal, and positive rotation angles are\n-          clockwise.\n+          clockwise. [degrees]\n         * `aoi`: The angle-of-incidence of direct irradiance onto the\n-          rotated panel surface.\n+          rotated panel surface. [degrees]\n         * `surface_tilt`: The angle between the panel surface and the earth\n-          surface, accounting for panel rotation.\n+          surface, accounting for panel rotation. [degrees]\n         * `surface_azimuth`: The azimuth of the rotated panel, determined by\n           projecting the vector normal to the panel's surface to the earth's\n-          surface.\n+          surface. [degrees]\n+\n+    See also\n+    --------\n+    pvlib.tracking.calc_axis_tilt\n+    pvlib.tracking.calc_cross_axis_tilt\n \n     References\n     ----------\n-    .. [1] Lorenzo, E et al., 2011, \"Tracking and back-tracking\", Prog. in\n-       Photovoltaics: Research and Applications, v. 19, pp. 747-753.\n+    .. [1] Kevin Anderson and Mark Mikofski, \"Slope-Aware Backtracking for\n+       Single-Axis Trackers\", Technical Report NREL/TP-5K00-76626, July 2020.\n+       https://www.nrel.gov/docs/fy20osti/76626.pdf\n     \"\"\"\n \n     # MATLAB to Python conversion by\n@@ -338,118 +376,84 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n     if apparent_azimuth.ndim > 1 or apparent_zenith.ndim > 1:\n         raise ValueError('Input dimensions must not exceed 1')\n \n-    # Calculate sun position x, y, z using coordinate system as in [1], Eq 2.\n-\n-    # Positive y axis is oriented parallel to earth surface along tracking axis\n-    # (for the purpose of illustration, assume y is oriented to the south);\n-    # positive x axis is orthogonal, 90 deg clockwise from y-axis, and parallel\n-    # to the earth's surface (if y axis is south, x axis is west);\n-    # positive z axis is normal to x, y axes, pointed upward.\n-\n-    # Equations in [1] assume solar azimuth is relative to reference vector\n-    # pointed south, with clockwise positive.\n-    # Here, the input solar azimuth is degrees East of North,\n-    # i.e., relative to a reference vector pointed\n-    # north with clockwise positive.\n-    # Rotate sun azimuth to coordinate system as in [1]\n-    # to calculate sun position.\n-\n-    az = apparent_azimuth - 180\n-    apparent_elevation = 90 - apparent_zenith\n-    x = cosd(apparent_elevation) * sind(az)\n-    y = cosd(apparent_elevation) * cosd(az)\n-    z = sind(apparent_elevation)\n-\n-    # translate array azimuth from compass bearing to [1] coord system\n-    # wholmgren: strange to see axis_azimuth calculated differently from az,\n-    # (not that it matters, or at least it shouldn't...).\n-    axis_azimuth_south = axis_azimuth - 180\n-\n-    # translate input array tilt angle axis_tilt to [1] coordinate system.\n-\n-    # In [1] coordinates, axis_tilt is a rotation about the x-axis.\n-    # For a system with array azimuth (y-axis) oriented south,\n-    # the x-axis is oriented west, and a positive axis_tilt is a\n-    # counterclockwise rotation, i.e, lifting the north edge of the panel.\n-    # Thus, in [1] coordinate system, in the northern hemisphere a positive\n-    # axis_tilt indicates a rotation toward the equator,\n-    # whereas in the southern hemisphere rotation toward the equator is\n-    # indicated by axis_tilt<0.  Here, the input axis_tilt is\n-    # always positive and is a rotation toward the equator.\n-\n-    # Calculate sun position (xp, yp, zp) in panel-oriented coordinate system:\n-    # positive y-axis is oriented along tracking axis at panel tilt;\n-    # positive x-axis is orthogonal, clockwise, parallel to earth surface;\n-    # positive z-axis is normal to x-y axes, pointed upward.\n-    # Calculate sun position (xp,yp,zp) in panel coordinates using [1] Eq 11\n-    # note that equation for yp (y' in Eq. 11 of Lorenzo et al 2011) is\n-    # corrected, after conversation with paper's authors.\n-\n-    xp = x*cosd(axis_azimuth_south) - y*sind(axis_azimuth_south)\n-    yp = (x*cosd(axis_tilt)*sind(axis_azimuth_south) +\n-          y*cosd(axis_tilt)*cosd(axis_azimuth_south) -\n-          z*sind(axis_tilt))\n-    zp = (x*sind(axis_tilt)*sind(axis_azimuth_south) +\n-          y*sind(axis_tilt)*cosd(axis_azimuth_south) +\n-          z*cosd(axis_tilt))\n+    # Calculate sun position x, y, z using coordinate system as in [1], Eq 1.\n+\n+    # NOTE: solar elevation = 90 - solar zenith, then use trig identities:\n+    # sin(90-x) = cos(x) & cos(90-x) = sin(x)\n+    sin_zenith = sind(apparent_zenith)\n+    x = sin_zenith * sind(apparent_azimuth)\n+    y = sin_zenith * cosd(apparent_azimuth)\n+    z = cosd(apparent_zenith)\n+\n+    # Assume the tracker reference frame is right-handed. Positive y-axis is\n+    # oriented along tracking axis; from north, the y-axis is rotated clockwise\n+    # by the axis azimuth and tilted from horizontal by the axis tilt. The\n+    # positive x-axis is 90 deg clockwise from the y-axis and parallel to\n+    # horizontal (e.g., if the y-axis is south, the x-axis is west); the\n+    # positive z-axis is normal to the x and y axes, pointed upward.\n+\n+    # Calculate sun position (xp, yp, zp) in tracker coordinate system using\n+    # [1] Eq 4.\n+\n+    cos_axis_azimuth = cosd(axis_azimuth)\n+    sin_axis_azimuth = sind(axis_azimuth)\n+    cos_axis_tilt = cosd(axis_tilt)\n+    sin_axis_tilt = sind(axis_tilt)\n+    xp = x*cos_axis_azimuth - y*sin_axis_azimuth\n+    yp = (x*cos_axis_tilt*sin_axis_azimuth\n+          + y*cos_axis_tilt*cos_axis_azimuth\n+          - z*sin_axis_tilt)\n+    zp = (x*sin_axis_tilt*sin_axis_azimuth\n+          + y*sin_axis_tilt*cos_axis_azimuth\n+          + z*cos_axis_tilt)\n \n     # The ideal tracking angle wid is the rotation to place the sun position\n-    # vector (xp, yp, zp) in the (y, z) plane; i.e., normal to the panel and\n-    # containing the axis of rotation.  wid = 0 indicates that the panel is\n-    # horizontal.  Here, our convention is that a clockwise rotation is\n+    # vector (xp, yp, zp) in the (y, z) plane, which is normal to the panel and\n+    # contains the axis of rotation.  wid = 0 indicates that the panel is\n+    # horizontal. Here, our convention is that a clockwise rotation is\n     # positive, to view rotation angles in the same frame of reference as\n-    # azimuth.  For example, for a system with tracking axis oriented south,\n-    # a rotation toward the east is negative, and a rotation to the west is\n-    # positive.\n-\n-    # Use arctan2 and avoid the tmp corrections.\n-\n-    # angle from x-y plane to projection of sun vector onto x-z plane\n-#     tmp = np.degrees(np.arctan(zp/xp))\n-\n-    # Obtain wid by translating tmp to convention for rotation angles.\n-    # Have to account for which quadrant of the x-z plane in which the sun\n-    # vector lies.  Complete solution here but probably not necessary to\n-    # consider QIII and QIV.\n-#     wid = pd.Series(index=times)\n-#     wid[(xp>=0) & (zp>=0)] =  90 - tmp[(xp>=0) & (zp>=0)]  # QI\n-#     wid[(xp<0)  & (zp>=0)] = -90 - tmp[(xp<0)  & (zp>=0)]  # QII\n-#     wid[(xp<0)  & (zp<0)]  = -90 - tmp[(xp<0)  & (zp<0)]   # QIII\n-#     wid[(xp>=0) & (zp<0)]  =  90 - tmp[(xp>=0) & (zp<0)]   # QIV\n+    # azimuth. For example, for a system with tracking axis oriented south, a\n+    # rotation toward the east is negative, and a rotation to the west is\n+    # positive. This is a right-handed rotation around the tracker y-axis.\n \n     # Calculate angle from x-y plane to projection of sun vector onto x-z plane\n-    # and then obtain wid by translating tmp to convention for rotation angles.\n-    wid = 90 - np.degrees(np.arctan2(zp, xp))\n+    # using [1] Eq. 5.\n+\n+    wid = np.degrees(np.arctan2(xp, zp))\n \n     # filter for sun above panel horizon\n     zen_gt_90 = apparent_zenith > 90\n     wid[zen_gt_90] = np.nan\n \n-    # Account for backtracking; modified from [1] to account for rotation\n-    # angle convention being used here.\n+    # Account for backtracking\n     if backtrack:\n-        axes_distance = 1/gcr\n-        # clip needed for low angles. GH 656\n-        temp = np.clip(axes_distance*cosd(wid), -1, 1)\n+        # distance between rows in terms of rack lengths relative to cross-axis\n+        # tilt\n+        axes_distance = 1/(gcr * cosd(cross_axis_tilt))\n \n-        # backtrack angle\n-        # (always positive b/c acosd returns values between 0 and 180)\n-        wc = np.degrees(np.arccos(temp))\n+        # NOTE: account for rare angles below array, see GH 824\n+        temp = np.abs(axes_distance * cosd(wid - cross_axis_tilt))\n \n-        # Eq 4 applied when wid in QIV (wid < 0 evalulates True), QI\n+        # backtrack angle using [1], Eq. 14\n         with np.errstate(invalid='ignore'):\n-            # errstate for GH 622\n-            tracker_theta = np.where(wid < 0, wid + wc, wid - wc)\n+            wc = np.degrees(-np.sign(wid)*np.arccos(temp))\n+\n+        # NOTE: in the middle of the day, arccos(temp) is out of range because\n+        # there's no row-to-row shade to avoid, & backtracking is unnecessary\n+        # [1], Eqs. 15-16\n+        with np.errstate(invalid='ignore'):\n+            tracker_theta = wid + np.where(temp < 1, wc, 0)\n     else:\n         tracker_theta = wid\n \n-    tracker_theta = np.minimum(tracker_theta, max_angle)\n-    tracker_theta = np.maximum(tracker_theta, -max_angle)\n+    # NOTE: max_angle defined relative to zero-point rotation, not the\n+    # system-plane normal\n+    tracker_theta = np.clip(tracker_theta, -max_angle, max_angle)\n \n-    # calculate panel normal vector in panel-oriented x, y, z coordinates.\n-    # y-axis is axis of tracker rotation.  tracker_theta is a compass angle\n+    # Calculate panel normal vector in panel-oriented x, y, z coordinates.\n+    # y-axis is axis of tracker rotation. tracker_theta is a compass angle\n     # (clockwise is positive) rather than a trigonometric angle.\n-    # the *0 is a trick to preserve NaN values.\n+    # NOTE: the *0 is a trick to preserve NaN values.\n     panel_norm = np.array([sind(tracker_theta),\n                            tracker_theta*0,\n                            cosd(tracker_theta)])\n@@ -460,30 +464,28 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n     # calculate angle-of-incidence on panel\n     aoi = np.degrees(np.arccos(np.abs(np.sum(sun_vec*panel_norm, axis=0))))\n \n-    # calculate panel tilt and azimuth\n-    # in a coordinate system where the panel tilt is the\n-    # angle from horizontal, and the panel azimuth is\n-    # the compass angle (clockwise from north) to the projection\n-    # of the panel's normal to the earth's surface.\n-    # These outputs are provided for convenience and comparison\n-    # with other PV software which use these angle conventions.\n+    # Calculate panel tilt and azimuth in a coordinate system where the panel\n+    # tilt is the angle from horizontal, and the panel azimuth is the compass\n+    # angle (clockwise from north) to the projection of the panel's normal to\n+    # the earth's surface. These outputs are provided for convenience and\n+    # comparison with other PV software which use these angle conventions.\n \n-    # project normal vector to earth surface.\n-    # First rotate about x-axis by angle -axis_tilt so that y-axis is\n-    # also parallel to earth surface, then project.\n+    # Project normal vector to earth surface. First rotate about x-axis by\n+    # angle -axis_tilt so that y-axis is also parallel to earth surface, then\n+    # project.\n \n     # Calculate standard rotation matrix\n     rot_x = np.array([[1, 0, 0],\n                       [0, cosd(-axis_tilt), -sind(-axis_tilt)],\n                       [0, sind(-axis_tilt), cosd(-axis_tilt)]])\n \n-    # panel_norm_earth contains the normal vector\n-    # expressed in earth-surface coordinates\n-    # (z normal to surface, y aligned with tracker axis parallel to earth)\n+    # panel_norm_earth contains the normal vector expressed in earth-surface\n+    # coordinates (z normal to surface, y aligned with tracker axis parallel to\n+    # earth)\n     panel_norm_earth = np.dot(rot_x, panel_norm).T\n \n-    # projection to plane tangent to earth surface,\n-    # in earth surface coordinates\n+    # projection to plane tangent to earth surface, in earth surface\n+    # coordinates\n     projected_normal = np.array([panel_norm_earth[:, 0],\n                                  panel_norm_earth[:, 1],\n                                  panel_norm_earth[:, 2]*0]).T\n@@ -491,69 +493,22 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n     # calculate vector magnitudes\n     projected_normal_mag = np.sqrt(np.nansum(projected_normal**2, axis=1))\n \n-    # renormalize the projected vector\n-    # avoid creating nan values.\n+    # renormalize the projected vector, avoid creating nan values.\n     non_zeros = projected_normal_mag != 0\n     projected_normal[non_zeros] = (projected_normal[non_zeros].T /\n                                    projected_normal_mag[non_zeros]).T\n \n     # calculation of surface_azimuth\n-    # 1. Find the angle.\n-#     surface_azimuth = pd.Series(\n-#         np.degrees(np.arctan(projected_normal[:,1]/projected_normal[:,0])),\n-#                                 index=times)\n     surface_azimuth = \\\n         np.degrees(np.arctan2(projected_normal[:, 1], projected_normal[:, 0]))\n \n-    # 2. Clean up atan when x-coord or y-coord is zero\n-#     surface_azimuth[(projected_normal[:,0]==0) & (projected_normal[:,1]>0)] =  90\n-#     surface_azimuth[(projected_normal[:,0]==0) & (projected_normal[:,1]<0)] =  -90\n-#     surface_azimuth[(projected_normal[:,1]==0) & (projected_normal[:,0]>0)] =  0\n-#     surface_azimuth[(projected_normal[:,1]==0) & (projected_normal[:,0]<0)] = 180\n-\n-    # 3. Correct atan for QII and QIII\n-#     surface_azimuth[(projected_normal[:,0]<0) & (projected_normal[:,1]>0)] += 180 # QII\n-#     surface_azimuth[(projected_normal[:,0]<0) & (projected_normal[:,1]<0)] += 180 # QIII\n-\n-    # 4. Skip to below\n-\n-    # at this point surface_azimuth contains angles between -90 and +270,\n-    # where 0 is along the positive x-axis,\n-    # the y-axis is in the direction of the tracker azimuth,\n-    # and positive angles are rotations from the positive x axis towards\n-    # the positive y-axis.\n-    # Adjust to compass angles\n-    # (clockwise rotation from 0 along the positive y-axis)\n-#    surface_azimuth[surface_azimuth<=90] = 90 - surface_azimuth[surface_azimuth<=90]\n-#    surface_azimuth[surface_azimuth>90] = 450 - surface_azimuth[surface_azimuth>90]\n-\n-    # finally rotate to align y-axis with true north\n-    # PVLIB_MATLAB has this latitude correction,\n-    # but I don't think it's latitude dependent if you always\n-    # specify axis_azimuth with respect to North.\n-#     if latitude > 0 or True:\n-#         surface_azimuth = surface_azimuth - axis_azimuth\n-#     else:\n-#         surface_azimuth = surface_azimuth - axis_azimuth - 180\n-#     surface_azimuth[surface_azimuth<0] = 360 + surface_azimuth[surface_azimuth<0]\n-\n-    # the commented code above is mostly part of PVLIB_MATLAB.\n-    # My (wholmgren) take is that it can be done more simply.\n-    # Say that we're pointing along the postive x axis (likely west).\n-    # We just need to rotate 90 degrees to get from the x axis\n-    # to the y axis (likely south),\n-    # and then add the axis_azimuth to get back to North.\n-    # Anything left over is the azimuth that we want,\n-    # and we can map it into the [0,360) domain.\n-\n-    # 4. Rotate 0 reference from panel's x axis to it's y axis and\n-    #    then back to North.\n+    # Rotate 0 reference from panel's x-axis to its y-axis and then back to\n+    # north.\n     surface_azimuth = 90 - surface_azimuth + axis_azimuth\n \n-    # 5. Map azimuth into [0,360) domain.\n-    # surface_azimuth[surface_azimuth < 0] += 360\n-    # surface_azimuth[surface_azimuth >= 360] -= 360\n-    surface_azimuth = surface_azimuth % 360\n+    # Map azimuth into [0,360) domain.\n+    with np.errstate(invalid='ignore'):\n+        surface_azimuth = surface_azimuth % 360\n \n     # Calculate surface_tilt\n     dotproduct = (panel_norm_earth * projected_normal).sum(axis=1)\n@@ -561,7 +516,7 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n \n     # Bundle DataFrame for return values and filter for sun below horizon.\n     out = {'tracker_theta': tracker_theta, 'aoi': aoi,\n-           'surface_azimuth': surface_azimuth, 'surface_tilt': surface_tilt}\n+           'surface_tilt': surface_tilt, 'surface_azimuth': surface_azimuth}\n     if index is not None:\n         out = pd.DataFrame(out, index=index)\n         out = out[['tracker_theta', 'aoi', 'surface_azimuth', 'surface_tilt']]\n@@ -570,3 +525,154 @@ def singleaxis(apparent_zenith, apparent_azimuth,\n         out = {k: np.where(zen_gt_90, np.nan, v) for k, v in out.items()}\n \n     return out\n+\n+\n+def calc_axis_tilt(slope_azimuth, slope_tilt, axis_azimuth):\n+    \"\"\"\n+    Calculate tracker axis tilt in the global reference frame when on a sloped\n+    plane.\n+\n+    Parameters\n+    ----------\n+    slope_azimuth : float\n+        direction of normal to slope on horizontal [degrees]\n+    slope_tilt : float\n+        tilt of normal to slope relative to vertical [degrees]\n+    axis_azimuth : float\n+        direction of tracker axes on horizontal [degrees]\n+\n+    Returns\n+    -------\n+    axis_tilt : float\n+        tilt of tracker [degrees]\n+\n+    See also\n+    --------\n+    pvlib.tracking.singleaxis\n+    pvlib.tracking.calc_cross_axis_tilt\n+\n+    Notes\n+    -----\n+    See [1]_ for derivation of equations.\n+\n+    References\n+    ----------\n+    .. [1] Kevin Anderson and Mark Mikofski, \"Slope-Aware Backtracking for\n+       Single-Axis Trackers\", Technical Report NREL/TP-5K00-76626, July 2020.\n+       https://www.nrel.gov/docs/fy20osti/76626.pdf\n+    \"\"\"\n+    delta_gamma = axis_azimuth - slope_azimuth\n+    # equations 18-19\n+    tan_axis_tilt = cosd(delta_gamma) * tand(slope_tilt)\n+    return np.degrees(np.arctan(tan_axis_tilt))\n+\n+\n+def _calc_tracker_norm(ba, bg, dg):\n+    \"\"\"\n+    Calculate tracker normal, v, cross product of tracker axis and unit normal,\n+    N, to the system slope plane.\n+\n+    Parameters\n+    ----------\n+    ba : float\n+        axis tilt [degrees]\n+    bg : float\n+        ground tilt [degrees]\n+    dg : float\n+        delta gamma, difference between axis and ground azimuths [degrees]\n+\n+    Returns\n+    -------\n+    vector : tuple\n+        vx, vy, vz\n+    \"\"\"\n+    cos_ba = cosd(ba)\n+    cos_bg = cosd(bg)\n+    sin_bg = sind(bg)\n+    sin_dg = sind(dg)\n+    vx = sin_dg * cos_ba * cos_bg\n+    vy = sind(ba)*sin_bg + cosd(dg)*cos_ba*cos_bg\n+    vz = -sin_dg*sin_bg*cos_ba\n+    return vx, vy, vz\n+\n+\n+def _calc_beta_c(v, dg, ba):\n+    \"\"\"\n+    Calculate the cross-axis tilt angle.\n+\n+    Parameters\n+    ----------\n+    v : tuple\n+        tracker normal\n+    dg : float\n+        delta gamma, difference between axis and ground azimuths [degrees]\n+    ba : float\n+        axis tilt [degrees]\n+\n+    Returns\n+    -------\n+    beta_c : float\n+        cross-axis tilt angle [radians]\n+    \"\"\"\n+    vnorm = np.sqrt(np.dot(v, v))\n+    beta_c = np.arcsin(\n+        ((v[0]*cosd(dg) - v[1]*sind(dg)) * sind(ba) + v[2]*cosd(ba)) / vnorm)\n+    return beta_c\n+\n+\n+def calc_cross_axis_tilt(\n+        slope_azimuth, slope_tilt, axis_azimuth, axis_tilt):\n+    \"\"\"\n+    Calculate the angle, relative to horizontal, of the line formed by the\n+    intersection between the slope containing the tracker axes and a plane\n+    perpendicular to the tracker axes.\n+\n+    Use the cross-axis tilt to avoid row-to-row shade when backtracking on a\n+    slope not parallel with the axis azimuth. Cross-axis tilt should be\n+    specified using a right-handed convention. For example, trackers with axis\n+    azimuth of 180 degrees (heading south) will have a negative cross-axis tilt\n+    if the tracker axes plane slopes down to the east and positive cross-axis\n+    tilt if the tracker axes plane slopes up to the east.\n+\n+    Parameters\n+    ----------\n+    slope_azimuth : float\n+        direction of the normal to the slope containing the tracker axes, when\n+        projected on the horizontal [degrees]\n+    slope_tilt : float\n+        angle of the slope containing the tracker axes, relative to horizontal\n+        [degrees]\n+    axis_azimuth : float\n+        direction of tracker axes projected on the horizontal [degrees]\n+    axis_tilt : float\n+        tilt of trackers relative to horizontal [degrees]\n+\n+    Returns\n+    -------\n+    cross_axis_tilt : float\n+        angle, relative to horizontal, of the line formed by the intersection\n+        between the slope containing the tracker axes and a plane perpendicular\n+        to the tracker axes [degrees]\n+\n+    See also\n+    --------\n+    pvlib.tracking.singleaxis\n+    pvlib.tracking.calc_axis_tilt\n+\n+    Notes\n+    -----\n+    See [1]_ for derivation of equations.\n+\n+    References\n+    ----------\n+    .. [1] Kevin Anderson and Mark Mikofski, \"Slope-Aware Backtracking for\n+       Single-Axis Trackers\", Technical Report NREL/TP-5K00-76626, July 2020.\n+       https://www.nrel.gov/docs/fy20osti/76626.pdf\n+    \"\"\"\n+    # delta-gamma, difference between axis and slope azimuths\n+    delta_gamma = axis_azimuth - slope_azimuth\n+    # equation 22\n+    v = _calc_tracker_norm(axis_tilt, slope_tilt, delta_gamma)\n+    # equation 26\n+    beta_c = _calc_beta_c(v, delta_gamma, axis_tilt)\n+    return np.degrees(beta_c)\n", "test_patch": "diff --git a/pvlib/tests/test_tracking.py b/pvlib/tests/test_tracking.py\n--- a/pvlib/tests/test_tracking.py\n+++ b/pvlib/tests/test_tracking.py\n@@ -6,8 +6,10 @@\n from conftest import assert_frame_equal\n from numpy.testing import assert_allclose\n \n+import pvlib\n from pvlib.location import Location\n from pvlib import tracking\n+from conftest import DATA_DIR\n \n SINGLEAXIS_COL_ORDER = ['tracker_theta', 'aoi',\n                         'surface_azimuth', 'surface_tilt']\n@@ -41,7 +43,7 @@ def test_scalars():\n     expect = {'tracker_theta': 0, 'aoi': 10, 'surface_azimuth': 90,\n               'surface_tilt': 0}\n     for k, v in expect.items():\n-        assert_allclose(tracker_data[k], v)\n+        assert np.isclose(tracker_data[k], v)\n \n \n def test_arrays():\n@@ -55,7 +57,7 @@ def test_arrays():\n     expect = {'tracker_theta': 0, 'aoi': 10, 'surface_azimuth': 90,\n               'surface_tilt': 0}\n     for k, v in expect.items():\n-        assert_allclose(tracker_data[k], v)\n+        assert_allclose(tracker_data[k], v, atol=1e-7)\n \n \n def test_nans():\n@@ -71,7 +73,7 @@ def test_nans():\n               'surface_azimuth': np.array([90, nan, nan]),\n               'surface_tilt': np.array([0, nan, nan])}\n     for k, v in expect.items():\n-        assert_allclose(tracker_data[k], v)\n+        assert_allclose(tracker_data[k], v, atol=1e-7)\n \n     # repeat with Series because nans can differ\n     apparent_zenith = pd.Series(apparent_zenith)\n@@ -268,7 +270,7 @@ def test_horizon_tilted():\n     out = tracking.singleaxis(solar_zenith, solar_azimuth, axis_tilt=90,\n                               axis_azimuth=180, backtrack=False, max_angle=180)\n     expected = pd.DataFrame(np.array(\n-        [[ 180.,  45.,   0.,  90.],\n+        [[-180.,  45.,   0.,  90.],\n          [   0.,  45., 180.,  90.],\n          [ 179.,  45., 359.,  90.]]),\n         columns=['tracker_theta', 'aoi', 'surface_azimuth', 'surface_tilt'])\n@@ -276,15 +278,15 @@ def test_horizon_tilted():\n \n \n def test_low_sun_angles():\n-    # GH 656\n+    # GH 656, 824\n     result = tracking.singleaxis(\n         apparent_zenith=80, apparent_azimuth=338, axis_tilt=30,\n         axis_azimuth=180, max_angle=60, backtrack=True, gcr=0.35)\n     expected = {\n-        'tracker_theta': np.array([-50.31051385]),\n-        'aoi': np.array([61.35300178]),\n-        'surface_azimuth': np.array([112.53615425]),\n-        'surface_tilt': np.array([56.42233095])}\n+        'tracker_theta': np.array([60.0]),\n+        'aoi': np.array([80.420987]),\n+        'surface_azimuth': np.array([253.897886]),\n+        'surface_tilt': np.array([64.341094])}\n     for k, v in result.items():\n         assert_allclose(expected[k], v)\n \n@@ -447,6 +449,7 @@ def test_SingleAxisTracker___repr__():\n   max_angle: 45\n   backtrack: True\n   gcr: 0.25\n+  cross_axis_tilt: 0.0\n   name: None\n   surface_tilt: None\n   surface_azimuth: None\n@@ -471,6 +474,7 @@ def test_LocalizedSingleAxisTracker___repr__():\n   max_angle: 90\n   backtrack: True\n   gcr: 0.25\n+  cross_axis_tilt: 0.0\n   name: None\n   surface_tilt: None\n   surface_azimuth: None\n@@ -486,3 +490,87 @@ def test_LocalizedSingleAxisTracker___repr__():\n   tz: UTC\"\"\"\n \n     assert localized_system.__repr__() == expected\n+\n+\n+def test_calc_axis_tilt():\n+    # expected values\n+    expected_axis_tilt = 2.239  # [degrees]\n+    expected_side_slope = 9.86649274360294  # [degrees]\n+    expected = DATA_DIR / 'singleaxis_tracker_wslope.csv'\n+    expected = pd.read_csv(expected, index_col='timestamp', parse_dates=True)\n+    # solar positions\n+    starttime = '2017-01-01T00:30:00-0300'\n+    stoptime = '2017-12-31T23:59:59-0300'\n+    lat, lon = -27.597300, -48.549610\n+    times = pd.DatetimeIndex(pd.date_range(starttime, stoptime, freq='H'))\n+    solpos = pvlib.solarposition.get_solarposition(times, lat, lon)\n+    # singleaxis tracker w/slope data\n+    slope_azimuth, slope_tilt = 77.34, 10.1149\n+    axis_azimuth = 0.0\n+    max_angle = 75.0\n+    # Note: GCR is relative to horizontal distance between rows\n+    gcr = 0.33292759  # GCR = length / horizontal_pitch = 1.64 / 5 / cos(9.86)\n+    # calculate tracker axis zenith\n+    axis_tilt = tracking.calc_axis_tilt(\n+        slope_azimuth, slope_tilt, axis_azimuth=axis_azimuth)\n+    assert np.isclose(axis_tilt, expected_axis_tilt)\n+    # calculate cross-axis tilt and relative rotation\n+    cross_axis_tilt = tracking.calc_cross_axis_tilt(\n+        slope_azimuth, slope_tilt, axis_azimuth, axis_tilt)\n+    assert np.isclose(cross_axis_tilt, expected_side_slope)\n+    sat = tracking.singleaxis(\n+        solpos.apparent_zenith, solpos.azimuth, axis_tilt, axis_azimuth,\n+        max_angle, backtrack=True, gcr=gcr, cross_axis_tilt=cross_axis_tilt)\n+    np.testing.assert_allclose(\n+        sat['tracker_theta'], expected['tracker_theta'], atol=1e-7)\n+    np.testing.assert_allclose(sat['aoi'], expected['aoi'], atol=1e-7)\n+    np.testing.assert_allclose(\n+        sat['surface_azimuth'], expected['surface_azimuth'], atol=1e-7)\n+    np.testing.assert_allclose(\n+        sat['surface_tilt'], expected['surface_tilt'], atol=1e-7)\n+\n+\n+def test_slope_aware_backtracking():\n+    \"\"\"\n+    Test validation data set from https://www.nrel.gov/docs/fy20osti/76626.pdf\n+    \"\"\"\n+    expected_data = np.array(\n+        [('2019-01-01T08:00-0500',  2.404287, 122.79177, -84.440, -10.899),\n+         ('2019-01-01T09:00-0500', 11.263058, 133.288729, -72.604, -25.747),\n+         ('2019-01-01T10:00-0500', 18.733558, 145.285552, -59.861, -59.861),\n+         ('2019-01-01T11:00-0500', 24.109076, 158.939435, -45.578, -45.578),\n+         ('2019-01-01T12:00-0500', 26.810735, 173.931802, -28.764, -28.764),\n+         ('2019-01-01T13:00-0500', 26.482495, 189.371536, -8.475, -8.475),\n+         ('2019-01-01T14:00-0500', 23.170447, 204.13681, 15.120, 15.120),\n+         ('2019-01-01T15:00-0500', 17.296785, 217.446538, 39.562, 39.562),\n+         ('2019-01-01T16:00-0500',  9.461862, 229.102218, 61.587, 32.339),\n+         ('2019-01-01T17:00-0500',  0.524817, 239.330401, 79.530, 5.490)],\n+        dtype=[\n+            ('Time', '<M8[h]'), ('ApparentElevation', '<f8'),\n+            ('SolarAzimuth', '<f8'), ('TrueTracking', '<f8'),\n+            ('Backtracking', '<f8')])\n+    expected_axis_tilt = 9.666\n+    expected_slope_angle = -2.576\n+    slope_azimuth, slope_tilt = 180.0, 10.0\n+    axis_azimuth = 195.0\n+    axis_tilt = tracking.calc_axis_tilt(\n+        slope_azimuth, slope_tilt, axis_azimuth)\n+    assert np.isclose(axis_tilt, expected_axis_tilt, rtol=1e-3, atol=1e-3)\n+    cross_axis_tilt = tracking.calc_cross_axis_tilt(\n+        slope_azimuth, slope_tilt, axis_azimuth, axis_tilt)\n+    assert np.isclose(\n+        cross_axis_tilt, expected_slope_angle, rtol=1e-3, atol=1e-3)\n+    sat = tracking.singleaxis(\n+        90.0-expected_data['ApparentElevation'], expected_data['SolarAzimuth'],\n+        axis_tilt, axis_azimuth, max_angle=90.0, backtrack=True, gcr=0.5,\n+        cross_axis_tilt=cross_axis_tilt)\n+    np.testing.assert_allclose(\n+        sat['tracker_theta'], expected_data['Backtracking'],\n+        rtol=1e-3, atol=1e-3)\n+    truetracking = tracking.singleaxis(\n+        90.0-expected_data['ApparentElevation'], expected_data['SolarAzimuth'],\n+        axis_tilt, axis_azimuth, max_angle=90.0, backtrack=False, gcr=0.5,\n+        cross_axis_tilt=cross_axis_tilt)\n+    np.testing.assert_allclose(\n+        truetracking['tracker_theta'], expected_data['TrueTracking'],\n+        rtol=1e-3, atol=1e-3)\n", "problem_statement": "backtracking for rare case when sun below tracker improvement\n**Describe the bug**\r\n- related to #656\r\n- in the rare case when the sun rays are below the tracker, then the top of the next row is shaded\r\n- currently tracker backtracks away from sun, back is facing sun instead of front\r\n- this only happens for tilted trackers and very low sun angles, either early morning or late evening when the sun rays are furthest north or south\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n1. create a tilted tracker\r\n```python\r\n# in Brazil so facing north\r\naxis_azimuth = 0.0 \r\naxis_tilt = 20\r\nmax_angle = 75.0\r\ngcr = 0.35\r\n```\r\n2. pick the earliest morning (or latest evening) timestamp\r\n```python\r\nimport pvlib\r\nimport pandas as pd\r\n\r\n# Brazil, timezone is UTC-3[hrs]\r\nstarttime = '2017-01-01T00:30:00-0300'\r\nstoptime = '2017-12-31T23:59:59-0300'\r\nlat, lon = -27.597300, -48.549610\r\ntimes = pd.DatetimeIndex(pd.date_range(\r\n    starttime, stoptime, freq='H'))\r\nsolpos = pvlib.solarposition.get_solarposition(\r\n    times, lat, lon)\r\n# get the early times\r\nts0 = '2017-01-01 05:30:00-03:00'\r\nts1 = '2017-01-01 12:30:00-03:00'\r\napparent_zenith = solpos['apparent_zenith'][ts0:ts1]\r\nazimuth = solpos['azimuth'][ts0:ts1]\r\nsat = pvlib.tracking.singleaxis(\r\n    apparent_zenith, azimuth, axis_tilt, axis_azimuth, max_angle, True, gcr)\r\n```\r\n3. notice that the tracker suddenly jumps from one side facing east to west\r\n```\r\n                           tracker_theta        aoi  surface_azimuth  surface_tilt\r\n2017-01-01 05:30:00-03:00     -21.964540  62.721237       310.299287     29.368272\r\n2017-01-01 06:30:00-03:00      16.231156  69.264752        40.403367     25.546154\r\n2017-01-01 07:30:00-03:00      69.073645  20.433849        82.548858     70.389280\r\n2017-01-01 08:30:00-03:00      54.554616  18.683626        76.316479     56.978562\r\n2017-01-01 09:30:00-03:00      40.131687  17.224233        67.917292     44.072837\r\n2017-01-01 10:30:00-03:00      25.769332  16.144347        54.683567     32.194782\r\n2017-01-01 11:30:00-03:00      11.439675  15.509532        30.610665     22.923644\r\n2017-01-01 12:30:00-03:00      -2.877428  15.358209       351.639727     20.197537\r\n```\r\n\r\n4. AOI is also wrong\r\n\r\n**Expected behavior**\r\nThe tracker should avoid shade. It should not jump from one direction to the other. If the sun ray is below the tracker then it will need to track to it's max rotation or backtrack. If there is shading at it's max rotation then it should track backtrack to zero, or perhaps parallel to the sun rays. Perhaps if bifacial, then it could go backwards, 180 from the correct backtrack position to show it's backside to the sun.\r\n\r\nproposed algorithm (_updated after [this comment](#issuecomment-559154895)_):\r\n```python\r\nif backtracking:\r\n    # cos(R) = L / Lx, R is rotation, L is surface length,\r\n    # Lx is shadow on ground, tracker shades when Lx > x\r\n    # x is row spacing related to GCR, x = L/GCR\r\n    lrot = np.cos(tr_rot_no_lim)  # tracker rotation not limited by max angle\r\n\r\n    # Note: if tr_rot > 90[deg] then lrot < 0 \r\n    # which *can* happen at low angles if axis tilt > 0\r\n    # tracker should never backtrack more than 90[deg], when lrot = 0\r\n    cos_rot = np.minimum(np.abs(lrot) / self.gcr, 1)\r\n\r\n    # so if lrot<0 tracker should backtrack forward\r\n    # backtrack_rot = np.sign(lrot) * np.arccos(cos_rot)\r\n\r\n    # NOTE: updated after comment from @kevinsa5 at Nov 27, 2019, 8:16 AM PST\r\n    # to remove sign()\r\n    backtrack_rot = np.arccos(cos_rot)\r\n```\r\n\r\nalso remove abs from aoi calculation\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/c699575cb6857674f0a96348b77e10c805e741c7/pvlib/tracking.py#L461\r\n\r\n**Screenshots**\r\nIf applicable, add screenshots to help explain your problem.\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``:  0.6.3\r\n - ``pandas.__version__``:  0.24\r\n - python: 3.7\r\n\r\n**Additional context**\r\nAdd any other context about the problem here.\r\n\n[STY] remove old comments from single axis tracking\n**Describe the bug**\r\nAfter #823 is merged there may be stale comments in `pvlib.tracking.singleaxis` and commented code that can be removed. This might make the code more readable. It would also resolve some stickler complaints about long lines.\r\n\r\n**To Reproduce**\r\nComments to remove:\r\n1. [L375-L379](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L375-L379) - the tracking algorithm now follows [1] that uses clockwise rotation around z-axis from north\r\n2. [L393-L395](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L393-L395) - ditto\r\n3. [L400-L410](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L400-L410) - ditto\r\n4. [L441-L452](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L441-L452) - pvlib has been using arctan2(x,z) in `pvlib.tracking.singleaxis` for 6 years since 1fb82cc262e43e1d2b55e4b5510a1a5e7e340667, so I believe these comments are unnecessary now\r\n5. [L471-L472](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L471-L472) - this commented code was updated in #823, should we leave it or delete it?\r\n3. [L553-L555](../blob/e210b8253458a60c95fc21939e9817271cf51934/pvlib/tracking.py#L553-L555)\r\n\r\netc.\r\n\r\n[1] https://www.nrel.gov/docs/fy20osti/76626.pdf\r\n\r\n**Expected behavior**\r\nA clear and concise description of what you expected to happen.\r\n\r\n**Screenshots**\r\nIf applicable, add screenshots to help explain your problem.\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: \r\n - ``pandas.__version__``: \r\n - python:\r\n\r\n**Additional context**\r\nAdd any other context about the problem here.\r\n\n", "hints_text": "\n", "created_at": "2019-11-22T09:31:54Z", "version": "0.6", "FAIL_TO_PASS": ["pvlib/tests/test_tracking.py::test_horizon_tilted", "pvlib/tests/test_tracking.py::test_low_sun_angles", "pvlib/tests/test_tracking.py::test_SingleAxisTracker___repr__", "pvlib/tests/test_tracking.py::test_LocalizedSingleAxisTracker___repr__", "pvlib/tests/test_tracking.py::test_slope_aware_backtracking"], "PASS_TO_PASS": ["pvlib/tests/test_tracking.py::test_solar_noon", "pvlib/tests/test_tracking.py::test_scalars", "pvlib/tests/test_tracking.py::test_arrays", "pvlib/tests/test_tracking.py::test_nans", "pvlib/tests/test_tracking.py::test_arrays_multi", "pvlib/tests/test_tracking.py::test_azimuth_north_south", "pvlib/tests/test_tracking.py::test_max_angle", "pvlib/tests/test_tracking.py::test_backtrack", "pvlib/tests/test_tracking.py::test_axis_tilt", "pvlib/tests/test_tracking.py::test_axis_azimuth", "pvlib/tests/test_tracking.py::test_horizon_flat", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_creation", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_tracking", "pvlib/tests/test_tracking.py::test_LocalizedSingleAxisTracker_creation", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_localize", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_localize_location", "pvlib/tests/test_tracking.py::test_get_aoi", "pvlib/tests/test_tracking.py::test_get_irradiance"], "environment_setup_commit": "b91d178868d193afd56f8e3b013661a473d699c3"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-863", "base_commit": "b4b44acbf1ddcaf03df16210aac46cb3a8acd6b9", "patch": "diff --git a/pydicom/charset.py b/pydicom/charset.py\n--- a/pydicom/charset.py\n+++ b/pydicom/charset.py\n@@ -86,6 +86,7 @@\n }\n \n ENCODINGS_TO_CODES = {v: k for k, v in CODES_TO_ENCODINGS.items()}\n+ENCODINGS_TO_CODES['shift_jis'] = ESC + b')I'\n \n # Multi-byte character sets except Korean are handled by Python.\n # To decode them, the escape sequence shall be preserved in the input byte\n@@ -95,6 +96,182 @@\n                      'iso_ir_58')\n \n \n+def _encode_to_jis_x_0201(value, errors='strict'):\n+    \"\"\"Convert a unicode string into JIS X 0201 byte string using shift_jis\n+    encodings.\n+    shift_jis is a superset of jis_x_0201. So we can regard the encoded value\n+    as jis_x_0201 if it is single byte character.\n+\n+    Parameters\n+    ----------\n+    value : text type\n+        The unicode string as presented to the user.\n+    errors : str\n+        The behavior of a character which could not be encoded. If 'strict' is\n+        passed, raise an UnicodeEncodeError. If any other value is passed,\n+        non ISO IR 14 characters are replaced by the ASCII '?'.\n+\n+    Returns\n+    -------\n+    byte string\n+        The encoded string. If some characters in value could not be encoded to\n+        JIS X 0201, and `errors` is not set to 'strict', they are replaced to\n+        '?'.\n+\n+    Raises\n+    ------\n+    UnicodeEncodeError\n+        If errors is set to 'strict' and `value` could not be encoded with\n+        JIS X 0201.\n+    \"\"\"\n+\n+    Encoder = codecs.getincrementalencoder('shift_jis')\n+    encoder = Encoder()\n+\n+    # If errors is not strict, this function is used as fallback.\n+    # In this case, we use only ISO IR 14 to encode given value\n+    # without escape sequence.\n+    if errors != 'strict' or value == '':\n+        encoded = b''\n+        for c in value:\n+            try:\n+                b = encoder.encode(c)\n+            except UnicodeEncodeError as e:\n+                b = b'?'\n+\n+            if len(b) != 1 or 0x80 <= ord(b):\n+                b = b'?'\n+            encoded += b\n+        return encoded\n+\n+    encoded = encoder.encode(value[0])\n+    if len(encoded) != 1:\n+        raise UnicodeEncodeError(\n+            'shift_jis', value, 0, len(value), 'illegal multibyte sequence')\n+\n+    msb = ord(encoded) & 0x80  # msb is 1 for ISO IR 13, 0 for ISO IR 14\n+    for i, c in enumerate(value[1:], 1):\n+        try:\n+            b = encoder.encode(c)\n+        except UnicodeEncodeError as e:\n+            e.start = i\n+            e.end = len(value)\n+            raise e\n+        if len(b) != 1 or ((ord(b) & 0x80) ^ msb) != 0:\n+            character_set = 'ISO IR 14' if msb == 0 else 'ISO IR 13'\n+            msg = 'Given character is out of {}'.format(character_set)\n+            raise UnicodeEncodeError('shift_jis', value, i, len(value), msg)\n+        encoded += b\n+\n+    return encoded\n+\n+\n+def _encode_to_jis_x_0208(value, errors='strict'):\n+    \"\"\"Convert a unicode string into JIS X 0208 byte string using iso2022_jp\n+    encodings.\n+    The escape sequence which is located at the end of the encoded value has\n+    to vary depending on the value 1 of SpecificCharacterSet. So we have to\n+    trim it and append the correct escape sequence manually.\n+\n+    Parameters\n+    ----------\n+    value : text type\n+        The unicode string as presented to the user.\n+    errors : str\n+        The behavior of a character which could not be encoded. This value\n+        is passed to errors argument of str.encode().\n+\n+    Returns\n+    -------\n+    byte string\n+        The encoded string. If some characters in value could not be encoded to\n+        JIS X 0208, it depends on the behavior of iso2022_jp encoder.\n+\n+    Raises\n+    ------\n+    UnicodeEncodeError\n+        If errors is set to 'strict' and `value` could not be encoded with\n+        JIS X 0208.\n+    \"\"\"\n+\n+    # If errors is not strict, this function is used as fallback.\n+    # So keep the tail escape sequence of encoded for backward compatibility.\n+    if errors != 'strict':\n+        return value.encode('iso2022_jp', errors=errors)\n+\n+    Encoder = codecs.getincrementalencoder('iso2022-jp')\n+    encoder = Encoder()\n+\n+    encoded = encoder.encode(value[0])\n+    if encoded[:3] != ENCODINGS_TO_CODES['iso2022_jp']:\n+        raise UnicodeEncodeError(\n+            'iso2022_jp', value, 0, len(value),\n+            'Given character is out of ISO IR 87')\n+\n+    for i, c in enumerate(value[1:], 1):\n+        try:\n+            b = encoder.encode(c)\n+        except UnicodeEncodeError as e:\n+            e.start = i\n+            e.end = len(value)\n+            raise e\n+        if b[:3] == ENCODINGS_TO_CODES['iso8859']:\n+            raise UnicodeEncodeError(\n+                'iso2022_jp', value, i, len(value),\n+                'Given character is out of ISO IR 87')\n+        encoded += b\n+    return encoded\n+\n+\n+def _get_escape_sequence_for_encoding(encoding, encoded=None):\n+    \"\"\" Return an escape sequence corresponding to the given encoding. If\n+    encoding is 'shift_jis', return 'ESC)I' or 'ESC(J' depending on the first\n+    byte of encoded.\n+\n+    Parameters\n+    ----------\n+    encoding : str\n+        An encoding is used to specify  an escape sequence.\n+\n+    encoded : bytes or str\n+        The encoded value is used to chose an escape sequence if encoding is\n+        'shift_jis'\n+\n+    Returns\n+    -------\n+    string\n+        Escape sequence for encoded value.\n+    \"\"\"\n+\n+    ESC_ISO_IR_14 = ESC + b'(J'\n+    ESC_ISO_IR_13 = ESC + b')I'\n+\n+    if encoding == 'shift_jis':\n+        if encoded is None:\n+            return ESC_ISO_IR_14\n+\n+        if not in_py2:\n+            first_byte = encoded[0]\n+        else:\n+            first_byte = ord(encoded[0])\n+        if 0x80 <= first_byte:\n+            return ESC_ISO_IR_13\n+\n+        return ESC_ISO_IR_14\n+    return ENCODINGS_TO_CODES.get(encoding, b'')\n+\n+\n+# These encodings need escape sequence to handle alphanumeric characters.\n+need_tail_escape_sequence_encodings = ('iso2022_jp', 'iso-2022-jp')\n+\n+\n+custom_encoders = {\n+    'shift_jis': _encode_to_jis_x_0201,\n+    'iso2022_jp': _encode_to_jis_x_0208,\n+    'iso-2022-jp': _encode_to_jis_x_0208\n+}\n+\n+\n def decode_string(value, encodings, delimiters):\n     \"\"\"Convert a raw byte string into a unicode string using the given\n     list of encodings.\n@@ -281,9 +458,14 @@ def encode_string(value, encodings):\n     \"\"\"\n     for i, encoding in enumerate(encodings):\n         try:\n-            encoded = value.encode(encoding)\n+            encoded = _encode_string_impl(value, encoding)\n+\n             if i > 0 and encoding not in handled_encodings:\n-                return ENCODINGS_TO_CODES.get(encoding, b'') + encoded\n+                escape_sequence = _get_escape_sequence_for_encoding(\n+                        encoding, encoded=encoded)\n+                encoded = escape_sequence + encoded\n+            if encoding in need_tail_escape_sequence_encodings:\n+                encoded += _get_escape_sequence_for_encoding(encodings[0])\n             return encoded\n         except UnicodeError:\n             continue\n@@ -304,7 +486,7 @@ def encode_string(value, encodings):\n         warnings.warn(\"Failed to encode value with encodings: {} - using \"\n                       \"replacement characters in encoded string\"\n                       .format(', '.join(encodings)))\n-        return value.encode(encodings[0], errors='replace')\n+        return _encode_string_impl(value, encodings[0], errors='replace')\n \n \n def _encode_string_parts(value, encodings):\n@@ -337,14 +519,14 @@ def _encode_string_parts(value, encodings):\n     \"\"\"\n     encoded = bytearray()\n     unencoded_part = value\n+    best_encoding = None\n     while unencoded_part:\n         # find the encoding that can encode the longest part of the rest\n         # of the string still to be encoded\n         max_index = 0\n-        best_encoding = None\n         for encoding in encodings:\n             try:\n-                unencoded_part.encode(encoding)\n+                _encode_string_impl(unencoded_part, encoding)\n                 # if we get here, the whole rest of the value can be encoded\n                 best_encoding = encoding\n                 max_index = len(unencoded_part)\n@@ -355,20 +537,36 @@ def _encode_string_parts(value, encodings):\n                     max_index = e.start\n                     best_encoding = encoding\n         # none of the given encodings can encode the first character - give up\n-        if best_encoding is None:\n-            raise ValueError()\n+        if max_index == 0:\n+            raise ValueError(\"None of the given encodings can encode the \"\n+                             \"first character\")\n \n         # encode the part that can be encoded with the found encoding\n-        encoded_part = unencoded_part[:max_index].encode(best_encoding)\n+        encoded_part = _encode_string_impl(unencoded_part[:max_index],\n+                                           best_encoding)\n         if best_encoding not in handled_encodings:\n-            encoded += ENCODINGS_TO_CODES.get(best_encoding, b'')\n+            encoded += _get_escape_sequence_for_encoding(\n+                    best_encoding, encoded=encoded_part)\n         encoded += encoded_part\n         # set remaining unencoded part of the string and handle that\n         unencoded_part = unencoded_part[max_index:]\n     # unencoded_part is empty - we are done, return the encoded string\n+    if best_encoding in need_tail_escape_sequence_encodings:\n+        encoded += _get_escape_sequence_for_encoding(encodings[0])\n     return encoded\n \n \n+def _encode_string_impl(value, encoding, errors='strict'):\n+    \"\"\"Convert a unicode string into a byte string. If given encoding is in\n+    custom_encoders, use a corresponding custom_encoder. If given encoding\n+    is not in custom_encoders, use a corresponding python handled encoder.\n+    \"\"\"\n+    if encoding in custom_encoders:\n+        return custom_encoders[encoding](value, errors=errors)\n+    else:\n+        return value.encode(encoding, errors=errors)\n+\n+\n # DICOM PS3.5-2008 6.1.1 (p 18) says:\n #   default is ISO-IR 6 G0, equiv to common chr set of ISO 8859 (PS3.5 6.1.2.1)\n #    (0008,0005)  value 1 can *replace* the default encoding...\n", "test_patch": "diff --git a/pydicom/tests/test_charset.py b/pydicom/tests/test_charset.py\n--- a/pydicom/tests/test_charset.py\n+++ b/pydicom/tests/test_charset.py\n@@ -404,3 +404,36 @@ def test_invalid_second_encoding(self):\n                                 u\"- using default encoding instead\"):\n             pydicom.charset.decode(\n                 elem, ['ISO 2022 IR 100', 'ISO 2022 IR 146'])\n+\n+    def test_japanese_multi_byte_personname(self):\n+        \"\"\"Test japanese person name which has multi byte strings are\n+        correctly encoded.\"\"\"\n+        file_path = get_charset_files('chrH32.dcm')[0]\n+        ds = dcmread(file_path)\n+        ds.decode()\n+\n+        if hasattr(ds.PatientName, 'original_string'):\n+            original_string = ds.PatientName.original_string\n+            ds.PatientName.original_string = None\n+            fp = DicomBytesIO()\n+            fp.is_implicit_VR = False\n+            fp.is_little_endian = True\n+            ds.save_as(fp, write_like_original=False)\n+            fp.seek(0)\n+            ds_out = dcmread(fp)\n+            assert original_string == ds_out.PatientName.original_string\n+\n+    def test_japanese_multi_byte_encoding(self):\n+        \"\"\"Test japanese multi byte strings are correctly encoded.\"\"\"\n+        encoded = pydicom.charset.encode_string(u'\u3042a\uff71\u30a2',\n+                                                ['shift_jis', 'iso2022_jp'])\n+        assert b'\\x1b$B$\"\\x1b(Ja\\x1b)I\\xb1\\x1b$B%\"\\x1b(J' == encoded\n+\n+    def test_bad_japanese_encoding(self):\n+        \"\"\"Test japanese multi byte strings are not correctly encoded.\"\"\"\n+        with pytest.warns(UserWarning,\n+                          match=u\"Failed to encode value with encodings\"\n+                                u\": shift_jis - using replacement character\"\n+                                u\"s in encoded string\"):\n+            encoded = pydicom.charset.encode_string(u'\u3042a\uff71\u30a2', ['shift_jis'])\n+            assert b'?a??' == encoded\n", "problem_statement": "Wrong encoding occurs if the value 1 of SpecificCharacterSets is ISO 2022 IR 13.\n<!-- Instructions For Filing a Bug: https://github.com/pydicom/pydicom/blob/master/CONTRIBUTING.md#filing-bugs -->\r\n\r\n#### Description\r\nAll Japanese characters are encoded into shift_jis if the value 1 of SpecificCharacterSets (0x0008, 0x0005) is  ISO 2022 IR 13.\r\n\r\n#### Steps/Code to Reproduce\r\nThe japanese_pn and expect_encoded in the following code came from \r\n[H.3.2 Value 1 of Attribute Specific Character Set (0008,0005) is ISO 2022 IR 13.](http://dicom.nema.org/medical/dicom/2015b/output/chtml/part05/sect_H.3.2.html)\r\n\r\n```py\r\nimport pydicom\r\n\r\njapanese_pn = u\"\uff94\uff8f\uff80\uff9e^\uff80\uff9b\uff73=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046\"\r\nspecific_character_sets = [\"ISO 2022 IR 13\", \"ISO 2022 IR 87\"]\r\nexpect_encoded = (\r\n    b\"\\xd4\\xcf\\xc0\\xde\\x5e\\xc0\\xdb\\xb3\\x3d\\x1b\\x24\\x42\\x3b\\x33\"\r\n    b\"\\x45\\x44\\x1b\\x28\\x4a\\x5e\\x1b\\x24\\x42\\x42\\x40\\x4f\\x3a\\x1b\"\r\n    b\"\\x28\\x4a\\x3d\\x1b\\x24\\x42\\x24\\x64\\x24\\x5e\\x24\\x40\\x1b\\x28\"\r\n    b\"\\x4a\\x5e\\x1b\\x24\\x42\\x24\\x3f\\x24\\x6d\\x24\\x26\\x1b\\x28\\x4a\"\r\n)\r\n\r\npython_encodings = pydicom.charset.convert_encodings(specific_character_sets)\r\nactual_encoded = pydicom.charset.encode_string(japanese_pn, python_encodings)\r\n\r\nprint(\"actual:{}\".format(actual_encoded))\r\nprint(\"expect:{}\".format(expect_encoded))\r\n```\r\n<!--\r\nExample:\r\n```py\r\nfrom io import BytesIO\r\nfrom pydicom import dcmread\r\n\r\nbytestream = b'\\x02\\x00\\x02\\x00\\x55\\x49\\x16\\x00\\x31\\x2e\\x32\\x2e\\x38\\x34\\x30\\x2e\\x31' \\\r\n             b'\\x30\\x30\\x30\\x38\\x2e\\x35\\x2e\\x31\\x2e\\x31\\x2e\\x39\\x00\\x02\\x00\\x10\\x00' \\\r\n             b'\\x55\\x49\\x12\\x00\\x31\\x2e\\x32\\x2e\\x38\\x34\\x30\\x2e\\x31\\x30\\x30\\x30\\x38' \\\r\n             b'\\x2e\\x31\\x2e\\x32\\x00\\x20\\x20\\x10\\x00\\x02\\x00\\x00\\x00\\x01\\x00\\x20\\x20' \\\r\n             b'\\x20\\x00\\x06\\x00\\x00\\x00\\x4e\\x4f\\x52\\x4d\\x41\\x4c'\r\n\r\nfp = BytesIO(bytestream)\r\nds = dcmread(fp, force=True)\r\n\r\nprint(ds.PatientID)\r\n```\r\nIf the code is too long, feel free to put it in a public gist and link\r\nit in the issue: https://gist.github.com\r\n\r\nWhen possible use pydicom testing examples to reproduce the errors. Otherwise, provide\r\nan anonymous version of the data in order to replicate the errors.\r\n-->\r\n\r\n#### Expected Results\r\n<!-- Please paste or describe the expected results.\r\nExample: No error is thrown and the name of the patient is printed.-->\r\n```\r\nb'\\xd4\\xcf\\xc0\\xde^\\xc0\\xdb\\xb3=\\x1b$B;3ED\\x1b(J^\\x1b$BB@O:\\x1b(J=\\x1b$B$d$^$@\\x1b(J^\\x1b$B$?$m$&\\x1b(J'\r\n```\r\n\r\n#### Actual Results\r\n<!-- Please paste or specifically describe the actual output or traceback.\r\n(Use %xmode to deactivate ipython's trace beautifier)\r\nExample: ```AttributeError: 'FileDataset' object has no attribute 'PatientID'```\r\n-->\r\n```\r\nb'\\xd4\\xcf\\xc0\\xde^\\xc0\\xdb\\xb3=\\x8eR\\x93c^\\x91\\xbe\\x98Y=\\x82\\xe2\\x82\\xdc\\x82\\xbe^\\x82\\xbd\\x82\\xeb\\x82\\xa4'\r\n```\r\n\r\n#### Versions\r\n<!--\r\nPlease run the following snippet and paste the output below.\r\nimport platform; print(platform.platform())\r\nimport sys; print(\"Python\", sys.version)\r\nimport pydicom; print(\"pydicom\", pydicom.__version__)\r\n-->\r\n```\r\nLinux-4.15.0-50-generic-x86_64-with-debian-buster-sid\r\nPython 3.6.8 |Anaconda, Inc.| (default, Dec 30 2018, 01:22:34)\r\n[GCC 7.3.0]\r\npydicom 1.2.2\r\n```\r\n\r\n<!-- Thanks for contributing! -->\r\n\n", "hints_text": "In my opinion, this issue is caused by encoding to shift_jis doesn't raise UnicodeError when characters which are out of JIS X 0201 are given.  So I guess that this is fixed by encoding to jis correctly.\r\n\r\nIf you don't mind, please assign this issue to me. I will make a PR  for this issue.\nThanks for the report - of course you can make a PR for this, please go ahead! \nDear all.\r\nI'm trying to solve this issue. And I want some advice about the scope of this issue and the way of implementation.\r\nMay I discuss them in this issue thread? Or should I create a PR and add W.I.P to its title?\nWhatever suits you better - if you want to discuss concrete code, it may be easier to add a PR to be able to comment on specific lines, but that's completely up to you!\n@mrbean-bremen \r\n\r\nThank for your quick reply. I got it. First, I will write some concrete codes. And then I'll make a PR and want to discuss there. ", "created_at": "2019-06-15T15:09:57Z", "version": "1.2", "FAIL_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_personname", "pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_japanese_encoding"], "PASS_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_encodings", "pydicom/tests/test_charset.py::TestCharset::test_nested_character_sets", "pydicom/tests/test_charset.py::TestCharset::test_inherited_character_set_in_sequence", "pydicom/tests/test_charset.py::TestCharset::test_standard_file", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set_enforce_valid", "pydicom/tests/test_charset.py::TestCharset::test_decoding_with_specific_tags", "pydicom/tests/test_charset.py::TestCharset::test_bad_charset", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_code_extensions_not_allowed", "pydicom/tests/test_charset.py::TestCharset::test_convert_encodings_warnings", "pydicom/tests/test_charset.py::TestCharset::test_convert_python_encodings", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_patched_charset", "pydicom/tests/test_charset.py::TestCharset::test_patched_code_extension_charset", "pydicom/tests/test_charset.py::TestCharset::test_multi_charset_default_value", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_personname", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_text", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_code_extensions[ISO", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrArab-\\u0642\\u0628\\u0627\\u0646\\u064a^\\u0644\\u0646\\u0632\\u0627\\u0631]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFren-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFrenMulti-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGerm-\\xc4neas^R\\xfcdiger]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGreek-\\u0394\\u03b9\\u03bf\\u03bd\\u03c5\\u03c3\\u03b9\\u03bf\\u03c2]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH31-Yamada^Tarou=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH32-\\uff94\\uff8f\\uff80\\uff9e^\\uff80\\uff9b\\uff73=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrHbrw-\\u05e9\\u05e8\\u05d5\\u05df^\\u05d3\\u05d1\\u05d5\\u05e8\\u05d4]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrI2-Hong^Gildong=\\u6d2a^\\u5409\\u6d1e=\\ud64d^\\uae38\\ub3d9]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMulti-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMultiExplicitIR6-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrKoreanMulti-\\uae40\\ud76c\\uc911]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrRuss-\\u041b\\u044e\\u043ace\\u043c\\u0431yp\\u0433]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX1-Wang^XiaoDong=\\u738b^\\u5c0f\\u6771]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX2-Wang^XiaoDong=\\u738b^\\u5c0f\\u4e1c]", "pydicom/tests/test_charset.py::TestCharset::test_changed_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_second_encoding"], "environment_setup_commit": "b4b44acbf1ddcaf03df16210aac46cb3a8acd6b9"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1181", "base_commit": "8b98768818ee5ad85d9479877533651a2e9dc2cd", "patch": "diff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -63,11 +63,10 @@\n \n \n def basic_chain(times, latitude, longitude,\n+                surface_tilt, surface_azimuth,\n                 module_parameters, temperature_model_parameters,\n                 inverter_parameters,\n                 irradiance=None, weather=None,\n-                surface_tilt=None, surface_azimuth=None,\n-                orientation_strategy=None,\n                 transposition_model='haydavies',\n                 solar_position_method='nrel_numpy',\n                 airmass_model='kastenyoung1989',\n@@ -91,6 +90,17 @@ def basic_chain(times, latitude, longitude,\n         Positive is east of the prime meridian.\n         Use decimal degrees notation.\n \n+    surface_tilt : numeric\n+        Surface tilt angles in decimal degrees.\n+        The tilt angle is defined as degrees from horizontal\n+        (e.g. surface facing up = 0, surface facing horizon = 90)\n+\n+    surface_azimuth : numeric\n+        Surface azimuth angles in decimal degrees.\n+        The azimuth convention is defined\n+        as degrees east of north\n+        (North=0, South=180, East=90, West=270).\n+\n     module_parameters : None, dict or Series\n         Module parameters as defined by the SAPM. See pvsystem.sapm for\n         details.\n@@ -112,23 +122,6 @@ def basic_chain(times, latitude, longitude,\n         wind speed is 0 m/s.\n         Columns must be 'wind_speed', 'temp_air'.\n \n-    surface_tilt : None, float or Series, default None\n-        Surface tilt angles in decimal degrees.\n-        The tilt angle is defined as degrees from horizontal\n-        (e.g. surface facing up = 0, surface facing horizon = 90)\n-\n-    surface_azimuth : None, float or Series, default None\n-        Surface azimuth angles in decimal degrees.\n-        The azimuth convention is defined\n-        as degrees east of north\n-        (North=0, South=180, East=90, West=270).\n-\n-    orientation_strategy : None or str, default None\n-        The strategy for aligning the modules.\n-        If not None, sets the ``surface_azimuth`` and ``surface_tilt``\n-        properties of the ``system``. Allowed strategies include 'flat',\n-        'south_at_latitude_tilt'. Ignored for SingleAxisTracker systems.\n-\n     transposition_model : str, default 'haydavies'\n         Passed to system.get_irradiance.\n \n@@ -157,17 +150,6 @@ def basic_chain(times, latitude, longitude,\n         power (Series).\n     \"\"\"\n \n-    # use surface_tilt and surface_azimuth if provided,\n-    # otherwise set them using the orientation_strategy\n-    if surface_tilt is not None and surface_azimuth is not None:\n-        pass\n-    elif orientation_strategy is not None:\n-        surface_tilt, surface_azimuth = \\\n-            get_orientation(orientation_strategy, latitude=latitude)\n-    else:\n-        raise ValueError('orientation_strategy or surface_tilt and '\n-                         'surface_azimuth must be provided')\n-\n     if altitude is None and pressure is None:\n         altitude = 0.\n         pressure = 101325.\n@@ -332,12 +314,6 @@ class ModelChain:\n         A :py:class:`~pvlib.location.Location` object that represents\n         the physical location at which to evaluate the model.\n \n-    orientation_strategy : None or str, default None\n-        The strategy for aligning the modules. If not None, sets the\n-        ``surface_azimuth`` and ``surface_tilt`` properties of the\n-        ``system``. Allowed strategies include 'flat',\n-        'south_at_latitude_tilt'. Ignored for SingleAxisTracker systems.\n-\n     clearsky_model : str, default 'ineichen'\n         Passed to location.get_clearsky.\n \n@@ -395,7 +371,6 @@ class ModelChain:\n                          'dc', 'ac', 'diode_params', 'tracking']\n \n     def __init__(self, system, location,\n-                 orientation_strategy=None,\n                  clearsky_model='ineichen',\n                  transposition_model='haydavies',\n                  solar_position_method='nrel_numpy',\n@@ -421,7 +396,6 @@ def __init__(self, system, location,\n         self.temperature_model = temperature_model\n \n         self.losses_model = losses_model\n-        self.orientation_strategy = orientation_strategy\n \n         self.weather = None\n         self.times = None\n@@ -451,7 +425,6 @@ def __setattr__(self, key, value):\n \n     @classmethod\n     def with_pvwatts(cls, system, location,\n-                     orientation_strategy=None,\n                      clearsky_model='ineichen',\n                      airmass_model='kastenyoung1989',\n                      name=None,\n@@ -469,12 +442,6 @@ def with_pvwatts(cls, system, location,\n             A :py:class:`~pvlib.location.Location` object that represents\n             the physical location at which to evaluate the model.\n \n-        orientation_strategy : None or str, default None\n-            The strategy for aligning the modules. If not None, sets the\n-            ``surface_azimuth`` and ``surface_tilt`` properties of the\n-            ``system``. Allowed strategies include 'flat',\n-            'south_at_latitude_tilt'. Ignored for SingleAxisTracker systems.\n-\n         clearsky_model : str, default 'ineichen'\n             Passed to location.get_clearsky.\n \n@@ -502,7 +469,6 @@ def with_pvwatts(cls, system, location,\n         >>> ModelChain.with_pvwatts(system, location)\n         ModelChain:\n           name: None\n-          orientation_strategy: None\n           clearsky_model: ineichen\n           transposition_model: perez\n           solar_position_method: nrel_numpy\n@@ -518,7 +484,6 @@ def with_pvwatts(cls, system, location,\n         config.update(kwargs)\n         return ModelChain(\n             system, location,\n-            orientation_strategy=orientation_strategy,\n             clearsky_model=clearsky_model,\n             airmass_model=airmass_model,\n             name=name,\n@@ -527,7 +492,6 @@ def with_pvwatts(cls, system, location,\n \n     @classmethod\n     def with_sapm(cls, system, location,\n-                  orientation_strategy=None,\n                   clearsky_model='ineichen',\n                   transposition_model='haydavies',\n                   solar_position_method='nrel_numpy',\n@@ -548,12 +512,6 @@ def with_sapm(cls, system, location,\n             A :py:class:`~pvlib.location.Location` object that represents\n             the physical location at which to evaluate the model.\n \n-        orientation_strategy : None or str, default None\n-            The strategy for aligning the modules. If not None, sets the\n-            ``surface_azimuth`` and ``surface_tilt`` properties of the\n-            ``system``. Allowed strategies include 'flat',\n-            'south_at_latitude_tilt'. Ignored for SingleAxisTracker systems.\n-\n         clearsky_model : str, default 'ineichen'\n             Passed to location.get_clearsky.\n \n@@ -589,7 +547,6 @@ def with_sapm(cls, system, location,\n         >>> ModelChain.with_sapm(system, location)\n         ModelChain:\n           name: None\n-          orientation_strategy: None\n           clearsky_model: ineichen\n           transposition_model: haydavies\n           solar_position_method: nrel_numpy\n@@ -605,7 +562,6 @@ def with_sapm(cls, system, location,\n         config.update(kwargs)\n         return ModelChain(\n             system, location,\n-            orientation_strategy=orientation_strategy,\n             clearsky_model=clearsky_model,\n             transposition_model=transposition_model,\n             solar_position_method=solar_position_method,\n@@ -616,7 +572,7 @@ def with_sapm(cls, system, location,\n \n     def __repr__(self):\n         attrs = [\n-            'name', 'orientation_strategy', 'clearsky_model',\n+            'name', 'clearsky_model',\n             'transposition_model', 'solar_position_method',\n             'airmass_model', 'dc_model', 'ac_model', 'aoi_model',\n             'spectral_model', 'temperature_model', 'losses_model'\n@@ -634,21 +590,6 @@ def getmcattr(self, attr):\n         return ('ModelChain: \\n  ' + '\\n  '.join(\n             f'{attr}: {getmcattr(self, attr)}' for attr in attrs))\n \n-    @property\n-    def orientation_strategy(self):\n-        return self._orientation_strategy\n-\n-    @orientation_strategy.setter\n-    def orientation_strategy(self, strategy):\n-        if strategy == 'None':\n-            strategy = None\n-\n-        if strategy is not None:\n-            self.system.surface_tilt, self.system.surface_azimuth = \\\n-                get_orientation(strategy, latitude=self.location.latitude)\n-\n-        self._orientation_strategy = strategy\n-\n     @property\n     def dc_model(self):\n         return self._dc_model\n", "test_patch": "diff --git a/pvlib/tests/test_modelchain.py b/pvlib/tests/test_modelchain.py\n--- a/pvlib/tests/test_modelchain.py\n+++ b/pvlib/tests/test_modelchain.py\n@@ -333,22 +333,6 @@ def test_with_pvwatts(pvwatts_dc_pvwatts_ac_system, location, weather):\n     mc.run_model(weather)\n \n \n-@pytest.mark.parametrize('strategy, expected', [\n-    (None, (32.2, 180)), ('None', (32.2, 180)), ('flat', (0, 180)),\n-    ('south_at_latitude_tilt', (32.2, 180))\n-])\n-def test_orientation_strategy(strategy, expected, sapm_dc_snl_ac_system,\n-                              location):\n-    mc = ModelChain(sapm_dc_snl_ac_system, location,\n-                    orientation_strategy=strategy)\n-\n-    # the || accounts for the coercion of 'None' to None\n-    assert (mc.orientation_strategy == strategy or\n-            mc.orientation_strategy is None)\n-    assert sapm_dc_snl_ac_system.surface_tilt == expected[0]\n-    assert sapm_dc_snl_ac_system.surface_azimuth == expected[1]\n-\n-\n def test_run_model_with_irradiance(sapm_dc_snl_ac_system, location):\n     mc = ModelChain(sapm_dc_snl_ac_system, location)\n     times = pd.date_range('20160101 1200-0700', periods=2, freq='6H')\n@@ -1235,8 +1219,7 @@ def test_infer_spectral_model(location, sapm_dc_snl_ac_system,\n                   'cec': cec_dc_snl_ac_system,\n                   'cec_native': cec_dc_native_snl_ac_system}\n     system = dc_systems[dc_model]\n-    mc = ModelChain(system, location,\n-                    orientation_strategy='None', aoi_model='physical')\n+    mc = ModelChain(system, location, aoi_model='physical')\n     assert isinstance(mc, ModelChain)\n \n \n@@ -1252,8 +1235,7 @@ def test_infer_temp_model(location, sapm_dc_snl_ac_system,\n                   'faiman_temp': pvwatts_dc_pvwatts_ac_faiman_temp_system,\n                   'fuentes_temp': pvwatts_dc_pvwatts_ac_fuentes_temp_system}\n     system = dc_systems[temp_model]\n-    mc = ModelChain(system, location,\n-                    orientation_strategy='None', aoi_model='physical',\n+    mc = ModelChain(system, location, aoi_model='physical',\n                     spectral_model='no_loss')\n     assert temp_model == mc.temperature_model.__name__\n     assert isinstance(mc, ModelChain)\n@@ -1263,14 +1245,12 @@ def test_infer_temp_model_invalid(location, sapm_dc_snl_ac_system):\n     sapm_dc_snl_ac_system.temperature_model_parameters.pop('a')\n     with pytest.raises(ValueError):\n         ModelChain(sapm_dc_snl_ac_system, location,\n-                   orientation_strategy='None', aoi_model='physical',\n-                   spectral_model='no_loss')\n+                   aoi_model='physical', spectral_model='no_loss')\n \n \n def test_temperature_model_inconsistent(location, sapm_dc_snl_ac_system):\n     with pytest.raises(ValueError):\n-        ModelChain(sapm_dc_snl_ac_system, location,\n-                   orientation_strategy='None', aoi_model='physical',\n+        ModelChain(sapm_dc_snl_ac_system, location, aoi_model='physical',\n                    spectral_model='no_loss', temperature_model='pvsyst')\n \n \n@@ -1441,17 +1421,14 @@ def test_aoi_model_user_func(sapm_dc_snl_ac_system, location, weather, mocker):\n def test_infer_aoi_model(location, system_no_aoi, aoi_model):\n     for k in iam._IAM_MODEL_PARAMS[aoi_model]:\n         system_no_aoi.module_parameters.update({k: 1.0})\n-    mc = ModelChain(system_no_aoi, location,\n-                    orientation_strategy='None',\n-                    spectral_model='no_loss')\n+    mc = ModelChain(system_no_aoi, location, spectral_model='no_loss')\n     assert isinstance(mc, ModelChain)\n \n \n def test_infer_aoi_model_invalid(location, system_no_aoi):\n     exc_text = 'could not infer AOI model'\n     with pytest.raises(ValueError, match=exc_text):\n-        ModelChain(system_no_aoi, location, orientation_strategy='None',\n-                   spectral_model='no_loss')\n+        ModelChain(system_no_aoi, location, spectral_model='no_loss')\n \n \n def constant_spectral_loss(mc):\n@@ -1623,23 +1600,6 @@ def test_ModelChain_attributes_deprecated_10(sapm_dc_snl_ac_system, location):\n         mc.aoi = 5\n \n \n-def test_basic_chain_required(sam_data, cec_inverter_parameters,\n-                              sapm_temperature_cs5p_220m):\n-    times = pd.date_range(start='20160101 1200-0700',\n-                          end='20160101 1800-0700', freq='6H')\n-    latitude = 32\n-    longitude = -111\n-    altitude = 700\n-    modules = sam_data['sandiamod']\n-    module_parameters = modules['Canadian_Solar_CS5P_220M___2009_']\n-    temp_model_params = sapm_temperature_cs5p_220m.copy()\n-    with pytest.raises(ValueError):\n-        dc, ac = modelchain.basic_chain(\n-            times, latitude, longitude, module_parameters, temp_model_params,\n-            cec_inverter_parameters, altitude=altitude\n-        )\n-\n-\n @requires_tables\n def test_basic_chain_alt_az(sam_data, cec_inverter_parameters,\n                             sapm_temperature_cs5p_220m):\n@@ -1653,37 +1613,15 @@ def test_basic_chain_alt_az(sam_data, cec_inverter_parameters,\n     module_parameters = modules['Canadian_Solar_CS5P_220M___2009_']\n     temp_model_params = sapm_temperature_cs5p_220m.copy()\n     dc, ac = modelchain.basic_chain(times, latitude, longitude,\n-                                    module_parameters,  temp_model_params,\n-                                    cec_inverter_parameters,\n-                                    surface_tilt=surface_tilt,\n-                                    surface_azimuth=surface_azimuth)\n+                                    surface_tilt, surface_azimuth,\n+                                    module_parameters, temp_model_params,\n+                                    cec_inverter_parameters)\n \n     expected = pd.Series(np.array([111.621405, -2.00000000e-02]),\n                          index=times)\n     assert_series_equal(ac, expected)\n \n \n-@requires_tables\n-def test_basic_chain_strategy(sam_data, cec_inverter_parameters,\n-                              sapm_temperature_cs5p_220m):\n-    times = pd.date_range(start='20160101 1200-0700',\n-                          end='20160101 1800-0700', freq='6H')\n-    latitude = 32.2\n-    longitude = -111\n-    altitude = 700\n-    modules = sam_data['sandiamod']\n-    module_parameters = modules['Canadian_Solar_CS5P_220M___2009_']\n-    temp_model_params = sapm_temperature_cs5p_220m.copy()\n-    dc, ac = modelchain.basic_chain(\n-        times, latitude, longitude, module_parameters, temp_model_params,\n-        cec_inverter_parameters, orientation_strategy='south_at_latitude_tilt',\n-        altitude=altitude)\n-\n-    expected = pd.Series(np.array([178.382754, -2.00000000e-02]),\n-                         index=times)\n-    assert_series_equal(ac, expected)\n-\n-\n @requires_tables\n def test_basic_chain_altitude_pressure(sam_data, cec_inverter_parameters,\n                                        sapm_temperature_cs5p_220m):\n@@ -1698,10 +1636,9 @@ def test_basic_chain_altitude_pressure(sam_data, cec_inverter_parameters,\n     module_parameters = modules['Canadian_Solar_CS5P_220M___2009_']\n     temp_model_params = sapm_temperature_cs5p_220m.copy()\n     dc, ac = modelchain.basic_chain(times, latitude, longitude,\n+                                    surface_tilt, surface_azimuth,\n                                     module_parameters, temp_model_params,\n                                     cec_inverter_parameters,\n-                                    surface_tilt=surface_tilt,\n-                                    surface_azimuth=surface_azimuth,\n                                     pressure=93194)\n \n     expected = pd.Series(np.array([113.190045, -2.00000000e-02]),\n@@ -1709,10 +1646,9 @@ def test_basic_chain_altitude_pressure(sam_data, cec_inverter_parameters,\n     assert_series_equal(ac, expected)\n \n     dc, ac = modelchain.basic_chain(times, latitude, longitude,\n+                                    surface_tilt, surface_azimuth,\n                                     module_parameters, temp_model_params,\n                                     cec_inverter_parameters,\n-                                    surface_tilt=surface_tilt,\n-                                    surface_azimuth=surface_azimuth,\n                                     altitude=altitude)\n \n     expected = pd.Series(np.array([113.189814, -2.00000000e-02]),\n@@ -1720,34 +1656,6 @@ def test_basic_chain_altitude_pressure(sam_data, cec_inverter_parameters,\n     assert_series_equal(ac, expected)\n \n \n-@pytest.mark.parametrize('strategy, strategy_str', [\n-    ('south_at_latitude_tilt', 'south_at_latitude_tilt'),\n-    (None, 'None')])  # GitHub issue 352\n-def test_ModelChain___repr__(sapm_dc_snl_ac_system, location, strategy,\n-                             strategy_str):\n-\n-    mc = ModelChain(sapm_dc_snl_ac_system, location,\n-                    orientation_strategy=strategy, name='my mc')\n-\n-    expected = '\\n'.join([\n-        'ModelChain: ',\n-        '  name: my mc',\n-        '  orientation_strategy: ' + strategy_str,\n-        '  clearsky_model: ineichen',\n-        '  transposition_model: haydavies',\n-        '  solar_position_method: nrel_numpy',\n-        '  airmass_model: kastenyoung1989',\n-        '  dc_model: sapm',\n-        '  ac_model: sandia_inverter',\n-        '  aoi_model: sapm_aoi_loss',\n-        '  spectral_model: sapm_spectral_loss',\n-        '  temperature_model: sapm_temp',\n-        '  losses_model: no_extra_losses'\n-    ])\n-\n-    assert mc.__repr__() == expected\n-\n-\n def test_complete_irradiance_clean_run(sapm_dc_snl_ac_system, location):\n     \"\"\"The DataFrame should not change if all columns are passed\"\"\"\n     mc = ModelChain(sapm_dc_snl_ac_system, location)\n", "problem_statement": "remove ModelChain.orientation_strategy\nI don't like that `ModelChain(system, location, orientation_strategy='flat`|`south_at_latitude_tilt`) modifies the `system` object. It's not something we do anywhere else in pvlib. `orientation_strategy` only supports flat and south_at_latitude_tilt, neither of which are commonly used in the real world in 2020. \r\n\r\nI think we should remove it, maybe even without deprecation, in 0.8.\r\n\r\nI'm ok with keeping the `modelchain.get_orientation` function for now.\n", "hints_text": "", "created_at": "2021-02-28T23:03:01Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_modelchain.py::test_basic_chain_alt_az", "pvlib/tests/test_modelchain.py::test_basic_chain_altitude_pressure"], "PASS_TO_PASS": ["pvlib/tests/test_modelchain.py::test_ModelChain_creation", "pvlib/tests/test_modelchain.py::test_with_sapm", "pvlib/tests/test_modelchain.py::test_with_pvwatts", "pvlib/tests/test_modelchain.py::test_run_model_with_irradiance", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_invalid_inverter_params_arrays[adr]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_no_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_arrays_one_missing_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_times_error_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_times_arrays", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dhi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[ghi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dni]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_perez", "pvlib/tests/test_modelchain.py::test_run_model_gueymard_perez", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_sapm_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_pvsyst_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_faiman_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_fuentes_temp", "pvlib/tests/test_modelchain.py::test_run_model_tracker", "pvlib/tests/test_modelchain.py::test_run_model_tracker_list", "pvlib/tests/test_modelchain.py::test__assign_total_irrad", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_different_indices", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_missing_column", "pvlib/tests/test_modelchain.py::test__prepare_temperature", "pvlib/tests/test_modelchain.py::test__prepare_temperature_arrays_weather", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params0-sapm_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params1-pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params2-faiman_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params3-fuentes_temp]", "pvlib/tests/test_modelchain.py::test_run_model_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_tracking", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_poa_global_differs", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_minimal_input", "pvlib/tests/test_modelchain.py::test_run_model_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_weather_single_array", "pvlib/tests/test_modelchain.py::test_infer_dc_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[desoto]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[singlediode]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvwatts_dc]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[cec]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[desoto]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec_native]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[sapm_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[faiman_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[fuentes_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model_invalid", "pvlib/tests/test_modelchain.py::test_temperature_model_inconsistent", "pvlib/tests/test_modelchain.py::test_dc_model_user_func", "pvlib/tests/test_modelchain.py::test_pvwatts_dc_multiple_strings", "pvlib/tests/test_modelchain.py::test_ac_models[sandia]", "pvlib/tests/test_modelchain.py::test_ac_models[adr]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts]", "pvlib/tests/test_modelchain.py::test_ac_models[sandia_multi]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts_multi]", "pvlib/tests/test_modelchain.py::test_ac_model_user_func", "pvlib/tests/test_modelchain.py::test_ac_model_not_a_model", "pvlib/tests/test_modelchain.py::test_infer_ac_model_invalid_params", "pvlib/tests/test_modelchain.py::test_aoi_models[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_model_no_loss", "pvlib/tests/test_modelchain.py::test_aoi_model_user_func", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[ashrae]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[physical]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model_invalid", "pvlib/tests/test_modelchain.py::test_spectral_models[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts_arrays", "pvlib/tests/test_modelchain.py::test_losses_models_ext_def", "pvlib/tests/test_modelchain.py::test_losses_models_no_loss", "pvlib/tests/test_modelchain.py::test_invalid_dc_model_params", "pvlib/tests/test_modelchain.py::test_invalid_models[dc_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[ac_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[aoi_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[spectral_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[temperature_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[losses_model]", "pvlib/tests/test_modelchain.py::test_bad_get_orientation", "pvlib/tests/test_modelchain.py::test_with_sapm_pvsystem_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_no_extra_kwargs", "pvlib/tests/test_modelchain.py::test_ModelChain_attributes_deprecated_10", "pvlib/tests/test_modelchain.py::test_complete_irradiance_clean_run", "pvlib/tests/test_modelchain.py::test_complete_irradiance", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_unknown_attribute", "pvlib/tests/test_modelchain.py::test_inconsistent_array_params", "pvlib/tests/test_modelchain.py::test_modelchain__common_keys", "pvlib/tests/test_modelchain.py::test__irrad_for_celltemp"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-927", "base_commit": "cd8b6a43192724128af68605ef22aabf3254f495", "patch": "diff --git a/astroid/context.py b/astroid/context.py\n--- a/astroid/context.py\n+++ b/astroid/context.py\n@@ -102,7 +102,7 @@ def clone(self):\n         starts with the same context but diverge as each side is inferred\n         so the InferenceContext will need be cloned\"\"\"\n         # XXX copy lookupname/callcontext ?\n-        clone = InferenceContext(self.path, inferred=self.inferred)\n+        clone = InferenceContext(self.path.copy(), inferred=self.inferred.copy())\n         clone.callcontext = self.callcontext\n         clone.boundnode = self.boundnode\n         clone.extra_context = self.extra_context\n", "test_patch": "diff --git a/tests/unittest_brain_numpy_core_umath.py b/tests/unittest_brain_numpy_core_umath.py\n--- a/tests/unittest_brain_numpy_core_umath.py\n+++ b/tests/unittest_brain_numpy_core_umath.py\n@@ -14,7 +14,7 @@\n except ImportError:\n     HAS_NUMPY = False\n \n-from astroid import bases, builder, nodes, util\n+from astroid import bases, builder, nodes\n \n \n @unittest.skipUnless(HAS_NUMPY, \"This test requires the numpy library.\")\n@@ -220,9 +220,7 @@ def test_numpy_core_umath_functions_return_type(self):\n             with self.subTest(typ=func_):\n                 inferred_values = list(self._inferred_numpy_func_call(func_))\n                 self.assertTrue(\n-                    len(inferred_values) == 1\n-                    or len(inferred_values) == 2\n-                    and inferred_values[-1].pytype() is util.Uninferable,\n+                    len(inferred_values) == 1,\n                     msg=\"Too much inferred values ({}) for {:s}\".format(\n                         inferred_values[-1].pytype(), func_\n                     ),\ndiff --git a/tests/unittest_inference.py b/tests/unittest_inference.py\n--- a/tests/unittest_inference.py\n+++ b/tests/unittest_inference.py\n@@ -1704,7 +1704,8 @@ def __init__(self):\n         \"\"\"\n         ast = extract_node(code, __name__)\n         expr = ast.func.expr\n-        self.assertIs(next(expr.infer()), util.Uninferable)\n+        with pytest.raises(exceptions.InferenceError):\n+            next(expr.infer())\n \n     def test_tuple_builtin_inference(self):\n         code = \"\"\"\n@@ -6032,5 +6033,37 @@ def test_infer_list_of_uninferables_does_not_crash():\n     assert not inferred.elts\n \n \n+# https://github.com/PyCQA/astroid/issues/926\n+def test_issue926_infer_stmts_referencing_same_name_is_not_uninferable():\n+    code = \"\"\"\n+    pair = [1, 2]\n+    ex = pair[0]\n+    if 1 + 1 == 2:\n+        ex = pair[1]\n+    ex\n+    \"\"\"\n+    node = extract_node(code)\n+    inferred = list(node.infer())\n+    assert len(inferred) == 2\n+    assert isinstance(inferred[0], nodes.Const)\n+    assert inferred[0].value == 1\n+    assert isinstance(inferred[1], nodes.Const)\n+    assert inferred[1].value == 2\n+\n+\n+# https://github.com/PyCQA/astroid/issues/926\n+def test_issue926_binop_referencing_same_name_is_not_uninferable():\n+    code = \"\"\"\n+    pair = [1, 2]\n+    ex = pair[0] + pair[1]\n+    ex\n+    \"\"\"\n+    node = extract_node(code)\n+    inferred = list(node.infer())\n+    assert len(inferred) == 1\n+    assert isinstance(inferred[0], nodes.Const)\n+    assert inferred[0].value == 3\n+\n+\n if __name__ == \"__main__\":\n     unittest.main()\ndiff --git a/tests/unittest_regrtest.py b/tests/unittest_regrtest.py\n--- a/tests/unittest_regrtest.py\n+++ b/tests/unittest_regrtest.py\n@@ -99,7 +99,7 @@ def test_numpy_crash(self):\n         astroid = builder.string_build(data, __name__, __file__)\n         callfunc = astroid.body[1].value.func\n         inferred = callfunc.inferred()\n-        self.assertEqual(len(inferred), 2)\n+        self.assertEqual(len(inferred), 1)\n \n     def test_nameconstant(self):\n         # used to fail for Python 3.4\n", "problem_statement": "infer_stmts cannot infer multiple uses of the same AssignName\nGiven multiple assignments to the same target which both reference the same AssignName, infer_stmts fails for subsequent attempts after the first.\r\n\r\n### Steps to reproduce\r\n\r\nThis appears to be a minimum working example, removing any part removes the effect:\r\n\r\n```python\r\nfails = astroid.extract_node(\"\"\"\r\n    pair = [1, 2]\r\n    ex = pair[0]\r\n    if 1 + 1 == 2:\r\n        ex = pair[1]\r\n    ex\r\n\"\"\")\r\nprint(list(fails.infer()))\r\n# [<Const.int l.2 at 0x...>, Uninferable]\r\n```\r\n\r\nFor some context, I originally saw this with attributes on an imported module, i.e.\r\n\r\n```python\r\nimport mod\r\nex = mod.One()\r\n# later ... or in some branch\r\nex = mod.Two()\r\n```\r\n\r\n### Current behavior\r\n\r\nSee above.\r\n\r\n### Expected behavior\r\n\r\nInlining the variable or switching to a different name works fine:\r\n\r\n```python\r\nworks = astroid.extract_node(\"\"\"\r\n    # pair = [1, 2]\r\n    ex = [1, 2][0]\r\n    if 1 + 1 == 2:\r\n        ex = [1, 2][1]\r\n    ex\r\n\"\"\")\r\nprint(list(works.infer()))\r\n# [<Const.int l.3 at 0x...>, <Const.int l.5 at 0x...>]\r\n\r\nworks = astroid.extract_node(\"\"\"\r\n    first = [1, 2]\r\n    second = [1, 2]\r\n    ex = first[0]\r\n    if 1 + 1 == 2:\r\n        ex = second[1]\r\n    ex\r\n\"\"\")\r\nprint(list(works.infer()))\r\n# [<Const.int l.2 at 0x...>, <Const.int l.3 at 0x...>]\r\n```\r\n\r\nI would expect that the first failing example would work similarly. This (only) worked\r\nin astroid 2.5 and appears to have been \"broken\" by the revert of cc3bfc5 in 03d15b0 (astroid 2.5.1 and above).\r\n\r\n### ``python -c \"from astroid import __pkginfo__; print(__pkginfo__.version)\"`` output\r\n\r\n```\r\n$ python -c \"from astroid import __pkginfo__; print(__pkginfo__.version)\"\r\n2.5-dev\r\n$ git rev-parse HEAD\r\n03d15b0f32f7d7c9b2cb062b9321e531bd954344\r\n```\r\n\n", "hints_text": "It appears that this is caused by `InferenceContext` maintaining a strong reference to the mutable set that is shared between clones, see this simplified example:\r\n\r\n```python\r\nclass Context:\r\n    def __init__(self, path=None):\r\n        self.path = path or set()\r\n    def clone(self):\r\n        return Context(path=self.path)\r\n\r\na = Context()\r\na.path.add('hello')\r\nb = a.clone()\r\nb.path.add('world')\r\nprint(a.path, b.path)\r\n# (set(['world', 'hello']), set(['world', 'hello']))\r\nprint(a.path is b.path)\r\n# True\r\n```", "created_at": "2021-03-29T12:55:48Z", "version": "2.5", "FAIL_TO_PASS": ["tests/unittest_inference.py::InferenceTest::test_stop_iteration_leak", "tests/unittest_inference.py::test_issue926_infer_stmts_referencing_same_name_is_not_uninferable", "tests/unittest_inference.py::test_issue926_binop_referencing_same_name_is_not_uninferable"], "PASS_TO_PASS": ["tests/unittest_inference.py::InferenceUtilsTest::test_path_wrapper", "tests/unittest_inference.py::InferenceTest::test__new__", "tests/unittest_inference.py::InferenceTest::test__new__bound_methods", "tests/unittest_inference.py::InferenceTest::test_advanced_tupleassign_name_inference1", "tests/unittest_inference.py::InferenceTest::test_advanced_tupleassign_name_inference2", "tests/unittest_inference.py::InferenceTest::test_ancestors_inference", "tests/unittest_inference.py::InferenceTest::test_ancestors_inference2", "tests/unittest_inference.py::InferenceTest::test_args_default_inference1", "tests/unittest_inference.py::InferenceTest::test_args_default_inference2", "tests/unittest_inference.py::InferenceTest::test_aug_different_types_aug_not_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_different_types_aug_not_implemented_rop_fallback", "tests/unittest_inference.py::InferenceTest::test_aug_different_types_augop_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_different_types_no_method_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_same_type_aug_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_same_type_aug_not_implemented_normal_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_same_type_not_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_subtype_aug_op_is_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_subtype_both_not_implemented", "tests/unittest_inference.py::InferenceTest::test_aug_op_subtype_normal_op_is_implemented", "tests/unittest_inference.py::InferenceTest::test_augassign", "tests/unittest_inference.py::InferenceTest::test_augop_supertypes_augop_implemented", "tests/unittest_inference.py::InferenceTest::test_augop_supertypes_none_implemented", "tests/unittest_inference.py::InferenceTest::test_augop_supertypes_normal_binop_implemented", "tests/unittest_inference.py::InferenceTest::test_augop_supertypes_not_implemented_returned_for_all", "tests/unittest_inference.py::InferenceTest::test_augop_supertypes_reflected_binop_implemented", "tests/unittest_inference.py::InferenceTest::test_bin_op_classes", "tests/unittest_inference.py::InferenceTest::test_bin_op_supertype_more_complicated_example", "tests/unittest_inference.py::InferenceTest::test_binary_op_custom_class", "tests/unittest_inference.py::InferenceTest::test_binary_op_float_div", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_add", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_bitand", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_bitor", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_bitxor", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_shiftleft", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_shiftright", "tests/unittest_inference.py::InferenceTest::test_binary_op_int_sub", "tests/unittest_inference.py::InferenceTest::test_binary_op_list_mul", "tests/unittest_inference.py::InferenceTest::test_binary_op_list_mul_int", "tests/unittest_inference.py::InferenceTest::test_binary_op_list_mul_none", "tests/unittest_inference.py::InferenceTest::test_binary_op_on_self", "tests/unittest_inference.py::InferenceTest::test_binary_op_other_type", "tests/unittest_inference.py::InferenceTest::test_binary_op_other_type_using_reflected_operands", "tests/unittest_inference.py::InferenceTest::test_binary_op_reflected_and_not_implemented_is_type_error", "tests/unittest_inference.py::InferenceTest::test_binary_op_str_mul", "tests/unittest_inference.py::InferenceTest::test_binary_op_tuple_add", "tests/unittest_inference.py::InferenceTest::test_binary_op_type_errors", "tests/unittest_inference.py::InferenceTest::test_binop_ambiguity", "tests/unittest_inference.py::InferenceTest::test_binop_different_types_no_method_implemented", "tests/unittest_inference.py::InferenceTest::test_binop_different_types_normal_not_implemented_and_reflected", "tests/unittest_inference.py::InferenceTest::test_binop_different_types_reflected_and_normal_not_implemented", "tests/unittest_inference.py::InferenceTest::test_binop_different_types_reflected_only", "tests/unittest_inference.py::InferenceTest::test_binop_different_types_unknown_bases", "tests/unittest_inference.py::InferenceTest::test_binop_inference_errors", "tests/unittest_inference.py::InferenceTest::test_binop_list_with_elts", "tests/unittest_inference.py::InferenceTest::test_binop_same_types", "tests/unittest_inference.py::InferenceTest::test_binop_subtype", "tests/unittest_inference.py::InferenceTest::test_binop_subtype_implemented_in_parent", "tests/unittest_inference.py::InferenceTest::test_binop_subtype_not_implemented", "tests/unittest_inference.py::InferenceTest::test_binop_supertype", "tests/unittest_inference.py::InferenceTest::test_binop_supertype_both_not_implemented", "tests/unittest_inference.py::InferenceTest::test_binop_supertype_rop_not_implemented", "tests/unittest_inference.py::InferenceTest::test_bool_value", "tests/unittest_inference.py::InferenceTest::test_bool_value_instances", "tests/unittest_inference.py::InferenceTest::test_bool_value_recursive", "tests/unittest_inference.py::InferenceTest::test_bool_value_variable", "tests/unittest_inference.py::InferenceTest::test_bound_method_inference", "tests/unittest_inference.py::InferenceTest::test_bt_ancestor_crash", "tests/unittest_inference.py::InferenceTest::test_builtin_help", "tests/unittest_inference.py::InferenceTest::test_builtin_inference_py3k", "tests/unittest_inference.py::InferenceTest::test_builtin_name_inference", "tests/unittest_inference.py::InferenceTest::test_builtin_open", "tests/unittest_inference.py::InferenceTest::test_builtin_types", "tests/unittest_inference.py::InferenceTest::test_bytes_subscript", "tests/unittest_inference.py::InferenceTest::test_callfunc_context_func", "tests/unittest_inference.py::InferenceTest::test_callfunc_context_lambda", "tests/unittest_inference.py::InferenceTest::test_callfunc_inference", "tests/unittest_inference.py::InferenceTest::test_class_inference", "tests/unittest_inference.py::InferenceTest::test_classmethod_inferred_by_context", "tests/unittest_inference.py::InferenceTest::test_context_call_for_context_managers", "tests/unittest_inference.py::InferenceTest::test_conversion_of_dict_methods", "tests/unittest_inference.py::InferenceTest::test_del1", "tests/unittest_inference.py::InferenceTest::test_del2", "tests/unittest_inference.py::InferenceTest::test_delayed_attributes_without_slots", "tests/unittest_inference.py::InferenceTest::test_dict_inference", "tests/unittest_inference.py::InferenceTest::test_dict_inference_for_multiple_starred", "tests/unittest_inference.py::InferenceTest::test_dict_inference_kwargs", "tests/unittest_inference.py::InferenceTest::test_dict_inference_unpack_repeated_key", "tests/unittest_inference.py::InferenceTest::test_dict_invalid_args", "tests/unittest_inference.py::InferenceTest::test_exc_ancestors", "tests/unittest_inference.py::InferenceTest::test_except_inference", "tests/unittest_inference.py::InferenceTest::test_f_arg_f", "tests/unittest_inference.py::InferenceTest::test_factory_method", "tests/unittest_inference.py::InferenceTest::test_factory_methods_cls_call", "tests/unittest_inference.py::InferenceTest::test_factory_methods_object_new_call", "tests/unittest_inference.py::InferenceTest::test_float_complex_ambiguity", "tests/unittest_inference.py::InferenceTest::test_frozenset_builtin_inference", "tests/unittest_inference.py::InferenceTest::test_function_inference", "tests/unittest_inference.py::InferenceTest::test_genexpr_bool_value", "tests/unittest_inference.py::InferenceTest::test_getattr_inference1", "tests/unittest_inference.py::InferenceTest::test_getattr_inference2", "tests/unittest_inference.py::InferenceTest::test_getattr_inference3", "tests/unittest_inference.py::InferenceTest::test_getattr_inference4", "tests/unittest_inference.py::InferenceTest::test_getitem_of_class_raised_type_error", "tests/unittest_inference.py::InferenceTest::test_im_func_unwrap", "tests/unittest_inference.py::InferenceTest::test_import_as", "tests/unittest_inference.py::InferenceTest::test_infer_abstract_property_return_values", "tests/unittest_inference.py::InferenceTest::test_infer_arg_called_object_when_used_as_index_is_uninferable", "tests/unittest_inference.py::InferenceTest::test_infer_arg_called_type_defined_in_outer_scope_is_uninferable", "tests/unittest_inference.py::InferenceTest::test_infer_arg_called_type_is_uninferable", "tests/unittest_inference.py::InferenceTest::test_infer_arg_called_type_when_used_as_index_is_uninferable", "tests/unittest_inference.py::InferenceTest::test_infer_arg_called_type_when_used_as_subscript_is_uninferable", "tests/unittest_inference.py::InferenceTest::test_infer_arguments", "tests/unittest_inference.py::InferenceTest::test_infer_call_result_crash", "tests/unittest_inference.py::InferenceTest::test_infer_call_result_invalid_dunder_call_on_instance", "tests/unittest_inference.py::InferenceTest::test_infer_cls_in_class_methods", "tests/unittest_inference.py::InferenceTest::test_infer_coercion_rules_for_floats_complex", "tests/unittest_inference.py::InferenceTest::test_infer_empty_nodes", "tests/unittest_inference.py::InferenceTest::test_infer_nested", "tests/unittest_inference.py::InferenceTest::test_infer_variable_arguments", "tests/unittest_inference.py::InferenceTest::test_inference_restrictions", "tests/unittest_inference.py::InferenceTest::test_inferring_context_manager_skip_index_error", "tests/unittest_inference.py::InferenceTest::test_inferring_context_manager_unpacking_inference_error", "tests/unittest_inference.py::InferenceTest::test_inferring_with_contextlib_contextmanager", "tests/unittest_inference.py::InferenceTest::test_inferring_with_contextlib_contextmanager_failures", "tests/unittest_inference.py::InferenceTest::test_inferring_with_statement", "tests/unittest_inference.py::InferenceTest::test_inferring_with_statement_failures", "tests/unittest_inference.py::InferenceTest::test_infinite_loop_for_decorators", "tests/unittest_inference.py::InferenceTest::test_inner_value_redefined_by_subclass", "tests/unittest_inference.py::InferenceTest::test_instance_binary_operations", "tests/unittest_inference.py::InferenceTest::test_instance_binary_operations_multiple_methods", "tests/unittest_inference.py::InferenceTest::test_instance_binary_operations_parent", "tests/unittest_inference.py::InferenceTest::test_instance_slicing", "tests/unittest_inference.py::InferenceTest::test_instance_slicing_fails", "tests/unittest_inference.py::InferenceTest::test_instance_slicing_slices", "tests/unittest_inference.py::InferenceTest::test_invalid_slicing_primaries", "tests/unittest_inference.py::InferenceTest::test_invalid_subscripts", "tests/unittest_inference.py::InferenceTest::test_lambda_as_methods", "tests/unittest_inference.py::InferenceTest::test_list_builtin_inference", "tests/unittest_inference.py::InferenceTest::test_list_inference", "tests/unittest_inference.py::InferenceTest::test_listassign_name_inference", "tests/unittest_inference.py::InferenceTest::test_lookup_cond_branches", "tests/unittest_inference.py::InferenceTest::test_matmul", "tests/unittest_inference.py::InferenceTest::test_metaclass__getitem__", "tests/unittest_inference.py::InferenceTest::test_metaclass_custom_dunder_call", "tests/unittest_inference.py::InferenceTest::test_metaclass_custom_dunder_call_boundnode", "tests/unittest_inference.py::InferenceTest::test_metaclass_subclasses_arguments_are_classes_not_instances", "tests/unittest_inference.py::InferenceTest::test_metaclass_with_keyword_args", "tests/unittest_inference.py::InferenceTest::test_method_argument", "tests/unittest_inference.py::InferenceTest::test_module_inference", "tests/unittest_inference.py::InferenceTest::test_mul_list_supports__index__", "tests/unittest_inference.py::InferenceTest::test_mulassign_inference", "tests/unittest_inference.py::InferenceTest::test_name_bool_value", "tests/unittest_inference.py::InferenceTest::test_no_infinite_ancestor_loop", "tests/unittest_inference.py::InferenceTest::test_no_runtime_error_in_repeat_inference", "tests/unittest_inference.py::InferenceTest::test_nonregr_absolute_import", "tests/unittest_inference.py::InferenceTest::test_nonregr_func_arg", "tests/unittest_inference.py::InferenceTest::test_nonregr_func_global", "tests/unittest_inference.py::InferenceTest::test_nonregr_getitem_empty_tuple", "tests/unittest_inference.py::InferenceTest::test_nonregr_inference_modifying_col_offset", "tests/unittest_inference.py::InferenceTest::test_nonregr_instance_attrs", "tests/unittest_inference.py::InferenceTest::test_nonregr_lambda_arg", "tests/unittest_inference.py::InferenceTest::test_pluggable_inference", "tests/unittest_inference.py::InferenceTest::test_property", "tests/unittest_inference.py::InferenceTest::test_python25_no_relative_import", "tests/unittest_inference.py::InferenceTest::test_scope_lookup_same_attributes", "tests/unittest_inference.py::InferenceTest::test_set_builtin_inference", "tests/unittest_inference.py::InferenceTest::test_simple_for", "tests/unittest_inference.py::InferenceTest::test_simple_for_genexpr", "tests/unittest_inference.py::InferenceTest::test_simple_subscript", "tests/unittest_inference.py::InferenceTest::test_simple_tuple", "tests/unittest_inference.py::InferenceTest::test_slicing_list", "tests/unittest_inference.py::InferenceTest::test_slicing_str", "tests/unittest_inference.py::InferenceTest::test_slicing_tuple", "tests/unittest_inference.py::InferenceTest::test_special_method_masquerading_as_another", "tests/unittest_inference.py::InferenceTest::test_starred_in_list_literal", "tests/unittest_inference.py::InferenceTest::test_starred_in_literals_inference_issues", "tests/unittest_inference.py::InferenceTest::test_starred_in_mapping_inference_issues", "tests/unittest_inference.py::InferenceTest::test_starred_in_mapping_literal", "tests/unittest_inference.py::InferenceTest::test_starred_in_mapping_literal_no_inference_possible", "tests/unittest_inference.py::InferenceTest::test_starred_in_mapping_literal_non_const_keys_values", "tests/unittest_inference.py::InferenceTest::test_starred_in_set_literal", "tests/unittest_inference.py::InferenceTest::test_starred_in_tuple_literal", "tests/unittest_inference.py::InferenceTest::test_str_methods", "tests/unittest_inference.py::InferenceTest::test_subscript_inference_error", "tests/unittest_inference.py::InferenceTest::test_subscript_multi_slice", "tests/unittest_inference.py::InferenceTest::test_subscript_multi_value", "tests/unittest_inference.py::InferenceTest::test_subscript_supports__index__", "tests/unittest_inference.py::InferenceTest::test_swap_assign_inference", "tests/unittest_inference.py::InferenceTest::test_tuple_builtin_inference", "tests/unittest_inference.py::InferenceTest::test_tuple_then_list", "tests/unittest_inference.py::InferenceTest::test_tupleassign_name_inference", "tests/unittest_inference.py::InferenceTest::test_two_parents_from_same_module", "tests/unittest_inference.py::InferenceTest::test_type__new__invalid_attrs", "tests/unittest_inference.py::InferenceTest::test_type__new__invalid_bases", "tests/unittest_inference.py::InferenceTest::test_type__new__invalid_mcs_argument", "tests/unittest_inference.py::InferenceTest::test_type__new__invalid_name", "tests/unittest_inference.py::InferenceTest::test_type__new__metaclass_and_ancestors_lookup", "tests/unittest_inference.py::InferenceTest::test_type__new__metaclass_lookup", "tests/unittest_inference.py::InferenceTest::test_type__new__not_enough_arguments", "tests/unittest_inference.py::InferenceTest::test_type__new__with_metaclass", "tests/unittest_inference.py::InferenceTest::test_unary_empty_type_errors", "tests/unittest_inference.py::InferenceTest::test_unary_not", "tests/unittest_inference.py::InferenceTest::test_unary_op_assignment", "tests/unittest_inference.py::InferenceTest::test_unary_op_classes", "tests/unittest_inference.py::InferenceTest::test_unary_op_instance_method_not_callable", "tests/unittest_inference.py::InferenceTest::test_unary_op_leaks_stop_iteration", "tests/unittest_inference.py::InferenceTest::test_unary_op_numbers", "tests/unittest_inference.py::InferenceTest::test_unary_operands", "tests/unittest_inference.py::InferenceTest::test_unary_type_errors", "tests/unittest_inference.py::InferenceTest::test_unary_type_errors_for_non_instance_objects", "tests/unittest_inference.py::InferenceTest::test_unbound_method_inference", "tests/unittest_inference.py::InferenceTest::test_unicode_methods", "tests/unittest_inference.py::GetattrTest::test_attribute_missing", "tests/unittest_inference.py::GetattrTest::test_attrname_not_string", "tests/unittest_inference.py::GetattrTest::test_default", "tests/unittest_inference.py::GetattrTest::test_lambda", "tests/unittest_inference.py::GetattrTest::test_lookup", "tests/unittest_inference.py::GetattrTest::test_yes_when_unknown", "tests/unittest_inference.py::HasattrTest::test_attribute_is_missing", "tests/unittest_inference.py::HasattrTest::test_attribute_is_not_missing", "tests/unittest_inference.py::HasattrTest::test_inference_errors", "tests/unittest_inference.py::HasattrTest::test_lambda", "tests/unittest_inference.py::BoolOpTest::test_bool_ops", "tests/unittest_inference.py::BoolOpTest::test_other_nodes", "tests/unittest_inference.py::BoolOpTest::test_yes_when_unknown", "tests/unittest_inference.py::TestCallable::test_callable", "tests/unittest_inference.py::TestCallable::test_callable_methods", "tests/unittest_inference.py::TestCallable::test_inference_errors", "tests/unittest_inference.py::TestCallable::test_not_callable", "tests/unittest_inference.py::TestBool::test_bool", "tests/unittest_inference.py::TestBool::test_bool_bool_special_method", "tests/unittest_inference.py::TestBool::test_bool_instance_not_callable", "tests/unittest_inference.py::TestType::test_type", "tests/unittest_inference.py::ArgumentsTest::test_args", "tests/unittest_inference.py::ArgumentsTest::test_defaults", "tests/unittest_inference.py::ArgumentsTest::test_fail_to_infer_args", "tests/unittest_inference.py::ArgumentsTest::test_kwargs", "tests/unittest_inference.py::ArgumentsTest::test_kwargs_access_by_name", "tests/unittest_inference.py::ArgumentsTest::test_kwargs_and_other_named_parameters", "tests/unittest_inference.py::ArgumentsTest::test_kwargs_are_overridden", "tests/unittest_inference.py::ArgumentsTest::test_kwonly_args", "tests/unittest_inference.py::ArgumentsTest::test_multiple_kwargs", "tests/unittest_inference.py::ArgumentsTest::test_multiple_starred_args", "tests/unittest_inference.py::SliceTest::test_slice", "tests/unittest_inference.py::SliceTest::test_slice_attributes", "tests/unittest_inference.py::SliceTest::test_slice_inference_error", "tests/unittest_inference.py::SliceTest::test_slice_type", "tests/unittest_inference.py::CallSiteTest::test_call_site", "tests/unittest_inference.py::CallSiteTest::test_call_site_starred_args", "tests/unittest_inference.py::CallSiteTest::test_call_site_valid_arguments", "tests/unittest_inference.py::CallSiteTest::test_duplicated_keyword_arguments", "tests/unittest_inference.py::ObjectDunderNewTest::test_object_dunder_new_is_inferred_if_decorator", "tests/unittest_inference.py::test_infer_custom_inherit_from_property", "tests/unittest_inference.py::test_cannot_infer_call_result_for_builtin_methods", "tests/unittest_inference.py::test_unpack_dicts_in_assignment", "tests/unittest_inference.py::test_slice_inference_in_for_loops", "tests/unittest_inference.py::test_slice_inference_in_for_loops_not_working", "tests/unittest_inference.py::test_unpacking_starred_and_dicts_in_assignment", "tests/unittest_inference.py::test_unpacking_starred_empty_list_in_assignment", "tests/unittest_inference.py::test_regression_infinite_loop_decorator", "tests/unittest_inference.py::test_stop_iteration_in_int", "tests/unittest_inference.py::test_call_on_instance_with_inherited_dunder_call_method", "tests/unittest_inference.py::TestInferencePropagation::test_call_starargs_propagation", "tests/unittest_inference.py::TestInferencePropagation::test_call_kwargs_propagation", "tests/unittest_inference.py::test_limit_inference_result_amount", "tests/unittest_inference.py::test_attribute_inference_should_not_access_base_classes", "tests/unittest_inference.py::test_attribute_mro_object_inference", "tests/unittest_inference.py::test_inferred_sequence_unpacking_works", "tests/unittest_inference.py::test_recursion_error_inferring_slice", "tests/unittest_inference.py::test_exception_lookup_last_except_handler_wins", "tests/unittest_inference.py::test_exception_lookup_name_bound_in_except_handler", "tests/unittest_inference.py::test_builtin_inference_list_of_exceptions", "tests/unittest_inference.py::test_cannot_getattr_ann_assigns", "tests/unittest_inference.py::test_prevent_recursion_error_in_igetattr_and_context_manager_inference", "tests/unittest_inference.py::test_infer_context_manager_with_unknown_args", "tests/unittest_inference.py::test_subclass_of_exception[\\n", "tests/unittest_inference.py::test_ifexp_inference", "tests/unittest_inference.py::test_assert_last_function_returns_none_on_inference", "tests/unittest_inference.py::test_posonlyargs_inference", "tests/unittest_inference.py::test_infer_args_unpacking_of_self", "tests/unittest_inference.py::test_infer_exception_instance_attributes", "tests/unittest_inference.py::test_inference_is_limited_to_the_boundnode[\\n", "tests/unittest_inference.py::test_property_inference", "tests/unittest_inference.py::test_property_as_string", "tests/unittest_inference.py::test_property_callable_inference", "tests/unittest_inference.py::test_recursion_error_inferring_builtin_containers", "tests/unittest_inference.py::test_inferaugassign_picking_parent_instead_of_stmt", "tests/unittest_inference.py::test_classmethod_from_builtins_inferred_as_bound", "tests/unittest_inference.py::test_infer_dict_passes_context", "tests/unittest_inference.py::test_custom_decorators_for_classmethod_and_staticmethods[\\n", "tests/unittest_inference.py::test_dataclasses_subscript_inference_recursion_error_39", "tests/unittest_inference.py::test_self_reference_infer_does_not_trigger_recursion_error", "tests/unittest_inference.py::test_inferring_properties_multiple_time_does_not_mutate_locals_multiple_times", "tests/unittest_inference.py::test_getattr_fails_on_empty_values", "tests/unittest_inference.py::test_infer_first_argument_of_static_method_in_metaclass", "tests/unittest_inference.py::test_recursion_error_metaclass_monkeypatching", "tests/unittest_inference.py::test_allow_retrieving_instance_attrs_and_special_attrs_for_functions", "tests/unittest_inference.py::test_implicit_parameters_bound_method", "tests/unittest_inference.py::test_super_inference_of_abstract_property", "tests/unittest_inference.py::test_infer_generated_setter", "tests/unittest_inference.py::test_infer_list_of_uninferables_does_not_crash", "tests/unittest_regrtest.py::NonRegressionTests::test_ancestors_missing_from_function", "tests/unittest_regrtest.py::NonRegressionTests::test_ancestors_patching_class_recursion", "tests/unittest_regrtest.py::NonRegressionTests::test_ancestors_yes_in_bases", "tests/unittest_regrtest.py::NonRegressionTests::test_binop_generates_nodes_with_parents", "tests/unittest_regrtest.py::NonRegressionTests::test_decorator_callchain_issue42", "tests/unittest_regrtest.py::NonRegressionTests::test_decorator_names_inference_error_leaking", "tests/unittest_regrtest.py::NonRegressionTests::test_filter_stmts_scoping", "tests/unittest_regrtest.py::NonRegressionTests::test_living_property", "tests/unittest_regrtest.py::NonRegressionTests::test_module_path", "tests/unittest_regrtest.py::NonRegressionTests::test_nameconstant", "tests/unittest_regrtest.py::NonRegressionTests::test_package_sidepackage", "tests/unittest_regrtest.py::NonRegressionTests::test_recursion_regression_issue25", "tests/unittest_regrtest.py::NonRegressionTests::test_recursive_property_method", "tests/unittest_regrtest.py::NonRegressionTests::test_regression_inference_of_self_in_lambda", "tests/unittest_regrtest.py::NonRegressionTests::test_ssl_protocol", "tests/unittest_regrtest.py::NonRegressionTests::test_unicode_in_docstring", "tests/unittest_regrtest.py::NonRegressionTests::test_uninferable_string_argument_of_namedtuple", "tests/unittest_regrtest.py::test_ancestor_looking_up_redefined_function", "tests/unittest_regrtest.py::test_crash_in_dunder_inference_prevented"], "environment_setup_commit": "d2ae3940a1a374253cc95d8fe83cb96f7a57e843"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-941", "base_commit": "962becc0ae86c16f7b33140f43cd6ed8f1e8a045", "patch": "diff --git a/astroid/brain/brain_namedtuple_enum.py b/astroid/brain/brain_namedtuple_enum.py\n--- a/astroid/brain/brain_namedtuple_enum.py\n+++ b/astroid/brain/brain_namedtuple_enum.py\n@@ -315,6 +315,7 @@ def infer_enum_class(node):\n         if node.root().name == \"enum\":\n             # Skip if the class is directly from enum module.\n             break\n+        dunder_members = {}\n         for local, values in node.locals.items():\n             if any(not isinstance(value, nodes.AssignName) for value in values):\n                 continue\n@@ -372,7 +373,16 @@ def name(self):\n                 for method in node.mymethods():\n                     fake.locals[method.name] = [method]\n                 new_targets.append(fake.instantiate_class())\n+                dunder_members[local] = fake\n             node.locals[local] = new_targets\n+        members = nodes.Dict(parent=node)\n+        members.postinit(\n+            [\n+                (nodes.Const(k, parent=members), nodes.Name(v.name, parent=members))\n+                for k, v in dunder_members.items()\n+            ]\n+        )\n+        node.locals[\"__members__\"] = [members]\n         break\n     return node\n \ndiff --git a/astroid/scoped_nodes.py b/astroid/scoped_nodes.py\n--- a/astroid/scoped_nodes.py\n+++ b/astroid/scoped_nodes.py\n@@ -2554,7 +2554,7 @@ def igetattr(self, name, context=None, class_context=True):\n         context = contextmod.copy_context(context)\n         context.lookupname = name\n \n-        metaclass = self.declared_metaclass(context=context)\n+        metaclass = self.metaclass(context=context)\n         try:\n             attributes = self.getattr(name, context, class_context=class_context)\n             # If we have more than one attribute, make sure that those starting from\n@@ -2587,9 +2587,12 @@ def igetattr(self, name, context=None, class_context=True):\n                         yield from function.infer_call_result(\n                             caller=self, context=context\n                         )\n-                    # If we have a metaclass, we're accessing this attribute through\n-                    # the class itself, which means we can solve the property\n-                    elif metaclass:\n+                    # If we're in a class context, we need to determine if the property\n+                    # was defined in the metaclass (a derived class must be a subclass of\n+                    # the metaclass of all its bases), in which case we can resolve the\n+                    # property. If not, i.e. the property is defined in some base class\n+                    # instead, then we return the property object\n+                    elif metaclass and function.parent.scope() is metaclass:\n                         # Resolve a property as long as it is not accessed through\n                         # the class itself.\n                         yield from function.infer_call_result(\n", "test_patch": "diff --git a/tests/unittest_scoped_nodes.py b/tests/unittest_scoped_nodes.py\n--- a/tests/unittest_scoped_nodes.py\n+++ b/tests/unittest_scoped_nodes.py\n@@ -1923,6 +1923,153 @@ def update(self):\n         builder.parse(data)\n \n \n+def test_issue940_metaclass_subclass_property():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        @property\n+        def __members__(cls):\n+            return ['a', 'property']\n+    class Parent(metaclass=BaseMeta):\n+        pass\n+    class Derived(Parent):\n+        pass\n+    Derived.__members__\n+    \"\"\"\n+    )\n+    inferred = next(node.infer())\n+    assert isinstance(inferred, nodes.List)\n+    assert [c.value for c in inferred.elts] == [\"a\", \"property\"]\n+\n+\n+def test_issue940_property_grandchild():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class Grandparent:\n+        @property\n+        def __members__(self):\n+            return ['a', 'property']\n+    class Parent(Grandparent):\n+        pass\n+    class Child(Parent):\n+        pass\n+    Child().__members__\n+    \"\"\"\n+    )\n+    inferred = next(node.infer())\n+    assert isinstance(inferred, nodes.List)\n+    assert [c.value for c in inferred.elts] == [\"a\", \"property\"]\n+\n+\n+def test_issue940_metaclass_property():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        @property\n+        def __members__(cls):\n+            return ['a', 'property']\n+    class Parent(metaclass=BaseMeta):\n+        pass\n+    Parent.__members__\n+    \"\"\"\n+    )\n+    inferred = next(node.infer())\n+    assert isinstance(inferred, nodes.List)\n+    assert [c.value for c in inferred.elts] == [\"a\", \"property\"]\n+\n+\n+def test_issue940_with_metaclass_class_context_property():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        pass\n+    class Parent(metaclass=BaseMeta):\n+        @property\n+        def __members__(self):\n+            return ['a', 'property']\n+    class Derived(Parent):\n+        pass\n+    Derived.__members__\n+    \"\"\"\n+    )\n+    inferred = next(node.infer())\n+    assert not isinstance(inferred, nodes.List)\n+    assert isinstance(inferred, objects.Property)\n+\n+\n+def test_issue940_metaclass_values_funcdef():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        def __members__(cls):\n+            return ['a', 'func']\n+    class Parent(metaclass=BaseMeta):\n+        pass\n+    Parent.__members__()\n+    \"\"\"\n+    )\n+    inferred = next(node.infer())\n+    assert isinstance(inferred, nodes.List)\n+    assert [c.value for c in inferred.elts] == [\"a\", \"func\"]\n+\n+\n+def test_issue940_metaclass_derived_funcdef():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        def __members__(cls):\n+            return ['a', 'func']\n+    class Parent(metaclass=BaseMeta):\n+        pass\n+    class Derived(Parent):\n+        pass\n+    Derived.__members__()\n+    \"\"\"\n+    )\n+    inferred_result = next(node.infer())\n+    assert isinstance(inferred_result, nodes.List)\n+    assert [c.value for c in inferred_result.elts] == [\"a\", \"func\"]\n+\n+\n+def test_issue940_metaclass_funcdef_is_not_datadescriptor():\n+    node = builder.extract_node(\n+        \"\"\"\n+    class BaseMeta(type):\n+        def __members__(cls):\n+            return ['a', 'property']\n+    class Parent(metaclass=BaseMeta):\n+        @property\n+        def __members__(cls):\n+            return BaseMeta.__members__()\n+    class Derived(Parent):\n+        pass\n+    Derived.__members__\n+    \"\"\"\n+    )\n+    # Here the function is defined on the metaclass, but the property\n+    # is defined on the base class. When loading the attribute in a\n+    # class context, this should return the property object instead of\n+    # resolving the data descriptor\n+    inferred = next(node.infer())\n+    assert isinstance(inferred, objects.Property)\n+\n+\n+def test_issue940_enums_as_a_real_world_usecase():\n+    node = builder.extract_node(\n+        \"\"\"\n+    from enum import Enum\n+    class Sounds(Enum):\n+        bee = \"buzz\"\n+        cat = \"meow\"\n+    Sounds.__members__\n+    \"\"\"\n+    )\n+    inferred_result = next(node.infer())\n+    assert isinstance(inferred_result, nodes.Dict)\n+    actual = [k.value for k, _ in inferred_result.items]\n+    assert sorted(actual) == [\"bee\", \"cat\"]\n+\n+\n def test_metaclass_cannot_infer_call_yields_an_instance():\n     node = builder.extract_node(\n         \"\"\"\n", "problem_statement": "@property members defined in metaclasses of a base class are not correctly inferred\nRef https://github.com/PyCQA/astroid/issues/927#issuecomment-817244963\r\n\r\nInference works on the parent class but not the child in the following example:\r\n\r\n```python\r\nclass BaseMeta(type):\r\n    @property\r\n    def __members__(cls):\r\n        return ['a', 'property']\r\nclass Parent(metaclass=BaseMeta):\r\n    pass\r\nclass Derived(Parent):\r\n    pass\r\nParent.__members__  # [<Set.set l.10 at 0x...>]\r\nDerived.__members__  # [<Property.__members__ l.8 at 0x...>]\r\n```\n", "hints_text": "Looks like this is caused by https://github.com/PyCQA/astroid/blob/f2b197a4f8af0ceeddf435747a5c937c8632872a/astroid/scoped_nodes.py#L2590-L2603. When we are inferring an attribute on a derived class then `class_context` is `True` but `metaclass` is `False` so the property itself is returned.", "created_at": "2021-04-11T11:57:22Z", "version": "2.5", "FAIL_TO_PASS": ["tests/unittest_scoped_nodes.py::test_issue940_metaclass_subclass_property", "tests/unittest_scoped_nodes.py::test_issue940_enums_as_a_real_world_usecase"], "PASS_TO_PASS": ["tests/unittest_scoped_nodes.py::ModuleNodeTest::test_dict_interface", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_file_stream_api", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_file_stream_in_memory", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_file_stream_physical", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_getattr", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_import_1", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_import_2", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_module_getattr", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_public_names", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_relative_to_absolute_name", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_relative_to_absolute_name_beyond_top_level", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_special_attributes", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_stream_api", "tests/unittest_scoped_nodes.py::ModuleNodeTest::test_wildcard_import_names", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_argnames", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_decorator_builtin_descriptors", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_default_value", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_dict_interface", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_dunder_class_local_to_classmethod", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_dunder_class_local_to_function", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_dunder_class_local_to_method", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_format_args", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_format_args_keyword_only_args", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_four_args", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_func_instance_attr", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_igetattr", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_is_abstract", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_is_abstract_decorated", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_is_generator", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_is_method", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_lambda_pytype", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_lambda_qname", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_method_init_subclass", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_navigation", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_return_annotation_is_not_the_last", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_return_nothing", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_special_attributes", "tests/unittest_scoped_nodes.py::FunctionNodeTest::test_type_builtin_descriptor_subclasses", "tests/unittest_scoped_nodes.py::ClassNodeTest::test__bases__attribute", "tests/unittest_scoped_nodes.py::ClassNodeTest::test__mro__attribute", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_add_metaclass", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_all_ancestors_need_slots", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_ancestors", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_extra_decorators", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_extra_decorators_frame_is_not_class", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_extra_decorators_only_assignment_names_are_considered", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_extra_decorators_only_callfunc_are_considered", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_extra_decorators_only_same_name_considered", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_getattr", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_class_keywords", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_classmethod_attributes", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_cls_special_attributes_1", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_cls_special_attributes_2", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_dict_interface", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_duplicate_bases_namedtuple", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_extra_decorators_only_class_level_assignments", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_function_with_decorator_lineno", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_generator_from_infer_call_result_parent", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_getattr_from_grandpa", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_getattr_method_transform", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_has_dynamic_getattr", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_implicit_metaclass", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_implicit_metaclass_lookup", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_inner_classes", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_attr_ancestors", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_bound_method_lambdas", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_bound_method_lambdas_2", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_getattr", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_getattr_with_class_attr", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_instance_special_attributes", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_kite_graph", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_local_attr_ancestors", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_local_attr_invalid_mro", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_local_attr_mro", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_error", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_lookup", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_lookup_inference_errors", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_lookup_using_same_class", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_type", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_metaclass_yes_leak", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_methods", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_1", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_2", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_3", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_4", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_5", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_6", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_7", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_error_1", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_generic_error_2", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_with_attribute_classes", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_mro_with_factories", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_navigation", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_no_infinite_metaclass_loop", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_no_infinite_metaclass_loop_with_redefine", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_nonregr_infer_callresult", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_slots", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_slots_added_dynamically_still_inferred", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_slots_empty_list_of_slots", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_slots_for_dict_keys", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_slots_taken_from_parents", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_type", "tests/unittest_scoped_nodes.py::ClassNodeTest::test_type_three_arguments", "tests/unittest_scoped_nodes.py::test_issue940_property_grandchild", "tests/unittest_scoped_nodes.py::test_issue940_metaclass_property", "tests/unittest_scoped_nodes.py::test_issue940_with_metaclass_class_context_property", "tests/unittest_scoped_nodes.py::test_issue940_metaclass_values_funcdef", "tests/unittest_scoped_nodes.py::test_issue940_metaclass_derived_funcdef", "tests/unittest_scoped_nodes.py::test_issue940_metaclass_funcdef_is_not_datadescriptor", "tests/unittest_scoped_nodes.py::test_metaclass_cannot_infer_call_yields_an_instance", "tests/unittest_scoped_nodes.py::test_posonlyargs_python_38[\\ndef", "tests/unittest_scoped_nodes.py::test_posonlyargs_default_value"], "environment_setup_commit": "d2ae3940a1a374253cc95d8fe83cb96f7a57e843"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-2641", "base_commit": "b0ad239095b3cefb294c1b12c73c41e9f229aa81", "patch": "diff --git a/src/sqlfluff/core/rules/analysis/select_crawler.py b/src/sqlfluff/core/rules/analysis/select_crawler.py\n--- a/src/sqlfluff/core/rules/analysis/select_crawler.py\n+++ b/src/sqlfluff/core/rules/analysis/select_crawler.py\n@@ -129,6 +129,7 @@ class Query:\n     parent: Optional[\"Query\"] = field(default=None)\n     # Children (could be CTE, subselect, or other).\n     children: List[\"Query\"] = field(default_factory=list)\n+    cte_name_segment: Optional[BaseSegment] = field(default=None)\n \n     def lookup_cte(self, name: str, pop: bool = True) -> Optional[\"Query\"]:\n         \"\"\"Look up a CTE by name, in the current or any parent scope.\"\"\"\n@@ -233,7 +234,7 @@ def finish_segment():\n \n         # Stores the last CTE name we saw, so we can associate it with the\n         # corresponding Query.\n-        cte_name = None\n+        cte_name_segment: Optional[BaseSegment] = None\n \n         # Visit segment and all its children\n         for event, path in SelectCrawler.visit_segments(segment):\n@@ -273,10 +274,14 @@ def finish_segment():\n                                 append_query(query)\n                     else:\n                         # We're processing a \"with\" statement.\n-                        if cte_name:\n+                        if cte_name_segment:\n                             # If we have a CTE name, this is the Query for that\n                             # name.\n-                            query = self.query_class(QueryType.Simple, dialect)\n+                            query = self.query_class(\n+                                QueryType.Simple,\n+                                dialect,\n+                                cte_name_segment=cte_name_segment,\n+                            )\n                             if path[-1].is_type(\"select_statement\", \"values_clause\"):\n                                 # Add to the Query object we just created.\n                                 query.selectables.append(Selectable(path[-1], dialect))\n@@ -286,8 +291,8 @@ def finish_segment():\n                                 # to the Query later when we encounter those\n                                 # child segments.\n                                 pass\n-                            query_stack[-1].ctes[cte_name] = query\n-                            cte_name = None\n+                            query_stack[-1].ctes[cte_name_segment.raw] = query\n+                            cte_name_segment = None\n                             append_query(query)\n                         else:\n                             # There's no CTE name, so we're probably processing\n@@ -313,13 +318,13 @@ def finish_segment():\n                 elif path[-1].is_type(\"with_compound_statement\"):\n                     # Beginning a \"with\" statement, i.e. a block of CTEs.\n                     query = self.query_class(QueryType.WithCompound, dialect)\n-                    if cte_name:\n-                        query_stack[-1].ctes[cte_name] = query\n-                        cte_name = None\n+                    if cte_name_segment:\n+                        query_stack[-1].ctes[cte_name_segment.raw] = query\n+                        cte_name_segment = None\n                     append_query(query)\n                 elif path[-1].is_type(\"common_table_expression\"):\n                     # This is a \"<<cte name>> AS\". Grab the name for later.\n-                    cte_name = path[-1].segments[0].raw\n+                    cte_name_segment = path[-1].segments[0]\n             elif event == \"end\":\n                 finish_segment()\n \ndiff --git a/src/sqlfluff/rules/L045.py b/src/sqlfluff/rules/L045.py\n--- a/src/sqlfluff/rules/L045.py\n+++ b/src/sqlfluff/rules/L045.py\n@@ -1,7 +1,5 @@\n \"\"\"Implementation of Rule L045.\"\"\"\n-from typing import Optional\n-\n-from sqlfluff.core.rules.base import BaseRule, LintResult, RuleContext\n+from sqlfluff.core.rules.base import BaseRule, EvalResultType, LintResult, RuleContext\n from sqlfluff.core.rules.analysis.select_crawler import Query, SelectCrawler\n \n \n@@ -49,12 +47,21 @@ def _visit_sources(cls, query: Query):\n                 if isinstance(source, Query):\n                     cls._visit_sources(source)\n \n-    def _eval(self, context: RuleContext) -> Optional[LintResult]:\n+    def _eval(self, context: RuleContext) -> EvalResultType:\n         if context.segment.is_type(\"statement\"):\n             crawler = SelectCrawler(context.segment, context.dialect)\n             if crawler.query_tree:\n                 # Begin analysis at the final, outer query (key=None).\n                 self._visit_sources(crawler.query_tree)\n                 if crawler.query_tree.ctes:\n-                    return LintResult(anchor=context.segment)\n+                    return [\n+                        LintResult(\n+                            anchor=query.cte_name_segment,\n+                            description=f\"Query defines CTE \"\n+                            f'\"{query.cte_name_segment.raw}\" '\n+                            f\"but does not use it.\",\n+                        )\n+                        for query in crawler.query_tree.ctes.values()\n+                        if query.cte_name_segment\n+                    ]\n         return None\n", "test_patch": "diff --git a/test/fixtures/rules/std_rule_cases/L045.yml b/test/fixtures/rules/std_rule_cases/L045.yml\n--- a/test/fixtures/rules/std_rule_cases/L045.yml\n+++ b/test/fixtures/rules/std_rule_cases/L045.yml\n@@ -224,3 +224,13 @@ test_pass_spark3_values_clause_cte_2:\n   configs:\n     core:\n       dialect: spark3\n+\n+test_fail_query_uses_templating:\n+  fail_str: |\n+    WITH\n+    random_gibberish AS (\n+        SELECT 1\n+    )\n+\n+    SELECT var_bar\n+    FROM {{ ref('issue_2235') }}\ndiff --git a/test/rules/std_L045_test.py b/test/rules/std_L045_test.py\nnew file mode 100644\n--- /dev/null\n+++ b/test/rules/std_L045_test.py\n@@ -0,0 +1,46 @@\n+\"\"\"Tests the python routines within L045.\"\"\"\n+\n+import sqlfluff\n+\n+\n+def test__rules__std_L045_multiple_unused_ctes():\n+    \"\"\"Verify that L045 returns multiple lint issues, one per unused CTE.\"\"\"\n+    sql = \"\"\"\n+    WITH\n+    cte_1 AS (\n+        SELECT 1\n+    ),\n+    cte_2 AS (\n+        SELECT 2\n+    ),\n+    cte_3 AS (\n+        SELECT 3\n+    ),\n+    cte_4 AS (\n+        SELECT 4\n+    )\n+\n+    SELECT var_bar\n+    FROM cte_3\n+    \"\"\"\n+    result = sqlfluff.lint(sql, rules=[\"L045\"])\n+    assert result == [\n+        {\n+            \"code\": \"L045\",\n+            \"description\": 'Query defines CTE \"cte_1\" but does not use it.',\n+            \"line_no\": 3,\n+            \"line_pos\": 5,\n+        },\n+        {\n+            \"code\": \"L045\",\n+            \"description\": 'Query defines CTE \"cte_2\" but does not use it.',\n+            \"line_no\": 6,\n+            \"line_pos\": 5,\n+        },\n+        {\n+            \"code\": \"L045\",\n+            \"description\": 'Query defines CTE \"cte_4\" but does not use it.',\n+            \"line_no\": 12,\n+            \"line_pos\": 5,\n+        },\n+    ]\n", "problem_statement": "L045:  Unused CTEs are not automatically detected when using jinja/dbt as a templater\n## Expected Behaviour\r\nWhen unused CTEs are used with jinja or dbt as a templater, these are detected by L045. \r\n\r\n## Observed Behaviour\r\nWhen ref() statements are included in a SQL file and dbt is used as a templater, these seem to interfere with the ability for rule L045 to detect the unused CTEs.  The same behavior is observed when Jinja is included under the \"FROM\" statement of the relevant queries.\r\n\r\n## Steps to Reproduce\r\n(1). Generate a valid dbt project with at least two models with one variable each.  For the purposes of this reproduction example, I am going to assume that one model is 'foo' with variable 'var_foo' and one model is 'bar' with variable 'var_bar'.\r\n \r\n(2) Using DBT as a templater and BigQuery as a dialect, run dbt lint on the following SQL file:\r\n\r\n```sql\r\nWITH\r\nrandom_gibberish AS (\r\n    SELECT var_foo\r\n    FROM\r\n        {{ ref('foo') }}\r\n)\r\n\r\nSELECT var_bar\r\nFROM\r\n    {{ ref('bar') }}\r\n```\r\n\r\nIf the templater is switched to Jinja, L045 again doesn't produce any errors.\r\n\r\n## Dialect\r\nBigquery\r\n\r\n## Version\r\nSQLFluff version is 0.10.0.  Python version is 3.8.10.\r\nI'm using dbt 1.0.1 but the same issue occurs when Jinja is used as a templater.\r\n\r\n## Configuration\r\n```\r\n[sqlfluff]\r\ndialect = bigquery\r\nexclude_rules = L003,L008,L011,L014,L016,L029,L031,L034\r\n\r\n[sqlfluff:rules]\r\nmax_line_length = 120\r\ncomma_style = leading\r\n\r\n[sqlfluff:rules:L010]\r\ncapitalisation_policy = upper\r\n\r\n[sqlfluff:rules:L030]\r\ncapitalisation_policy = upper\r\n```\n", "hints_text": "", "created_at": "2022-02-14T20:13:17Z", "version": "0.9", "FAIL_TO_PASS": ["test/rules/std_L045_test.py::test__rules__std_L045_multiple_unused_ctes"], "PASS_TO_PASS": [], "environment_setup_commit": "a4dcf3f08d95cbde4efb39969b0ab8e33a791f21"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1413", "base_commit": "f909c76e31f759246cec3708dadd173c5d6e84b1", "patch": "diff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -433,13 +433,24 @@ def value(self) -> Any:\n     @value.setter\n     def value(self, val: Any) -> None:\n         \"\"\"Convert (if necessary) and set the value of the element.\"\"\"\n+        # Ignore backslash characters in these VRs, based on:\n+        # * Which str VRs can have backslashes in Part 5, Section 6.2\n+        # * All byte VRs\n+        exclusions = [\n+            'LT', 'OB', 'OD', 'OF', 'OL', 'OV', 'OW', 'ST', 'UN', 'UT',\n+            'OB/OW', 'OW/OB', 'OB or OW', 'OW or OB',\n+            # Probably not needed\n+            'AT', 'FD', 'FL', 'SQ', 'SS', 'SL', 'UL',\n+        ]\n+\n         # Check if is a string with multiple values separated by '\\'\n         # If so, turn them into a list of separate strings\n         #  Last condition covers 'US or SS' etc\n-        if isinstance(val, (str, bytes)) and self.VR not in \\\n-                ['UT', 'ST', 'LT', 'FL', 'FD', 'AT', 'OB', 'OW', 'OF', 'SL',\n-                 'SQ', 'SS', 'UL', 'OB/OW', 'OW/OB', 'OB or OW',\n-                 'OW or OB', 'UN'] and 'US' not in self.VR:\n+        if (\n+            isinstance(val, (str, bytes))\n+            and self.VR not in exclusions\n+            and 'US' not in self.VR\n+        ):\n             try:\n                 if _backslash_str in val:\n                     val = cast(str, val).split(_backslash_str)\n", "test_patch": "diff --git a/pydicom/tests/test_valuerep.py b/pydicom/tests/test_valuerep.py\n--- a/pydicom/tests/test_valuerep.py\n+++ b/pydicom/tests/test_valuerep.py\n@@ -1546,3 +1546,16 @@ def test_set_value(vr, pytype, vm0, vmN, keyword):\n     elem = ds[keyword]\n     assert elem.value == list(vmN)\n     assert list(vmN) == elem.value\n+\n+\n+@pytest.mark.parametrize(\"vr, pytype, vm0, vmN, keyword\", VALUE_REFERENCE)\n+def test_assigning_bytes(vr, pytype, vm0, vmN, keyword):\n+    \"\"\"Test that byte VRs are excluded from the backslash check.\"\"\"\n+    if pytype == bytes:\n+        ds = Dataset()\n+        value = b\"\\x00\\x01\" + b\"\\\\\" + b\"\\x02\\x03\"\n+        setattr(ds, keyword, value)\n+        elem = ds[keyword]\n+        assert elem.VR == vr\n+        assert elem.value == value\n+        assert elem.VM == 1\n", "problem_statement": "Error : a bytes-like object is required, not 'MultiValue'\nHello,\r\n\r\nI am getting following error while updating the tag LongTrianglePointIndexList (0066,0040),\r\n**TypeError: a bytes-like object is required, not 'MultiValue'**\r\n\r\nI noticed that the error  gets produced only when the VR is given as \"OL\" , works fine with \"OB\", \"OF\" etc.\r\n\r\nsample code (assume 'lineSeq' is the dicom dataset sequence):\r\n```python\r\nimport pydicom\r\nimport array\r\ndata=list(range(1,10))\r\ndata=array.array('H', indexData).tostring()  # to convert to unsigned short\r\nlineSeq.add_new(0x00660040, 'OL', data)   \r\nds.save_as(\"mydicom\")\r\n```\r\noutcome: **TypeError: a bytes-like object is required, not 'MultiValue'**\r\n\r\nusing version - 2.0.0.0\r\n\r\nAny help is appreciated.\r\n\r\nThank you\n", "hints_text": "Also tried following code to get the byte string, but same error.\r\n1. data=array.array('L', indexData).tostring()  # to convert to long -> same error\r\n2. data=array.array('Q', indexData).tostring()  # to convert to long long -> same error\r\n\r\n\nO* VRs should be `bytes`. Use `array.tobytes()` instead of `tostring()`?\r\n\r\nAlso, in the future if have an issue it's much more helpful if you post the full traceback rather than the error since we can look at it to figure out where in the code the exception is occurring.\r\n\r\nIt would also help if you posted the version of Python you're using. \r\n\r\nThis works fine for me with Python 3.9 and pydicom 2.1.2:\r\n```python\r\nfrom pydicom import Dataset\r\nimport array\r\n\r\narr = array.array('H', range(10))\r\nds = Dataset()\r\nds.is_little_endian = True\r\nds.is_implicit_VR = False\r\nds.LongTrianglePointIndexList = arr.tobytes()\r\nprint(ds[\"LongTrianglePointIndexList\"].VR)  # 'OL'\r\nds.save_as('temp.dcm')\r\n```\r\nThis also works fine:\r\n```python\r\nds = Dataset()\r\nds.add_new(0x00660040, 'OL', arr.tobytes())\r\n```\nThank you for the answer.\r\nUnfortunately the error still persists with above code.\r\nPlease find the attached detailed error.\r\n[error.txt](https://github.com/pydicom/pydicom/files/6661451/error.txt)\r\n\r\nOne more information is that the 'ds' is actually read from a file in the disk (ds=pydicom.read_file(filename)). \r\nand this byte array is stored under the following sequence\r\nds[0x0066,0x0002][0][0x0066,0x0013][0][0x0066,0x0028][0][0x0066,0x0040] = arr.tobytes()\r\n\r\npydicom - 2.0.0.0\r\npython - 3.6.4\r\n\r\nThank you.\nCould you post a minimal code sample that reproduces the issue please?\r\n\r\nIf you're using something like this:\r\n`ds[0x0066,0x0002][0][0x0066,0x0013][0][0x0066,0x0028][0][0x0066,0x0040] = arr.tobytes()`\r\n\r\nThen you're missing the `.value` assignment:\r\n`ds[0x0066,0x0002][0][0x0066,0x0013][0][0x0066,0x0028][0][0x0066,0x0040].value = arr.tobytes()`\nHello,\r\nabove code line I just mentioned to give an idea where the actual data is stored (tree level).\r\n\r\nPlease find the actual code used below,\r\n```python\r\nimport pydicom\r\nfrom pydicom.sequence import Sequence\r\nfrom pydicom.dataelem import DataElement\r\nfrom pydicom.dataset import Dataset\r\n\r\nds = pydicom.read_file(filename)\r\nsurfaceSeq= ds[0x0066,0x0002]\r\n\r\n#// read existing sequence items in the dataset\r\nseqlist=[]\r\nfor n in surfaceSeq:\r\n    seqlist.append(n)\r\n\r\nnewDs = Dataset()\r\n \r\nsurfaceMeshPrimitiveSq = Dataset()\r\nlineSeq = Dataset()\r\nindexData = list(range(1,100))\r\nindexData = array.array('H', indexData)\r\nindexData = indexData.tobytes()\r\nlineSeq.add_new(0x00660040, 'OL', indexData) \r\nsurfaceMeshPrimitiveSq.add_new(0x00660028, 'SQ', [lineSeq])\r\nnewDs.add_new(0x00660013, 'SQ', [surfaceMeshPrimitiveSq])\r\n\r\n#add the new sequnce item to the list\r\nseqlist.append(newDs)\r\nds[0x0066,0x0002] = DataElement(0x00660002,\"SQ\",seqlist)\r\nds.save_as(filename)\r\n```\nOK, I can reproduce with:\r\n```python\r\n\r\nimport array\r\n\r\nfrom pydicom import Dataset\r\nfrom pydicom.uid import ExplicitVRLittleEndian\r\n\r\nds = Dataset()\r\nds.file_meta = Dataset()\r\nds.file_meta.TransferSyntaxUID = ExplicitVRLittleEndian\r\n\r\nb = array.array('H', range(100)).tobytes()\r\n\r\nds.LongPrimitivePointIndexList = b\r\nds.save_as('1421.dcm')\r\n```\r\nAnd `print(ds)` gives:\r\n```\r\n(0066, 0040) Long Primitive Point Index List     OL: [b'\\x00\\x00\\x01\\x00\\x02\\x00\\x03\\x00\\x04\\x00\\x05\\x00\\x06\\x00\\x07\\x00\\x08\\x00\\t\\x00\\n\\x00\\x0b\\x00\\x0c\\x00\\r\\x00\\x0e\\x00\\x0f\\x00\\x10\\x00\\x11\\x00\\x12\\x00\\x13\\x00\\x14\\x00\\x15\\x00\\x16\\x00\\x17\\x00\\x18\\x00\\x19\\x00\\x1a\\x00\\x1b\\x00\\x1c\\x00\\x1d\\x00\\x1e\\x00\\x1f\\x00 \\x00!\\x00\"\\x00#\\x00$\\x00%\\x00&\\x00\\'\\x00(\\x00)\\x00*\\x00+\\x00,\\x00-\\x00.\\x00/\\x000\\x001\\x002\\x003\\x004\\x005\\x006\\x007\\x008\\x009\\x00:\\x00;\\x00<\\x00=\\x00>\\x00?\\x00@\\x00A\\x00B\\x00C\\x00D\\x00E\\x00F\\x00G\\x00H\\x00I\\x00J\\x00K\\x00L\\x00M\\x00N\\x00O\\x00P\\x00Q\\x00R\\x00S\\x00T\\x00U\\x00V\\x00W\\x00X\\x00Y\\x00Z\\x00[\\x00', b'\\x00]\\x00^\\x00_\\x00`\\x00a\\x00b\\x00c\\x00']\r\n```\r\nI think this is because the byte value is hitting the hex for the backslash character during assignment. Ouch, that's kinda nasty.", "created_at": "2021-06-16T09:47:08Z", "version": "2.1", "FAIL_TO_PASS": ["pydicom/tests/test_valuerep.py::test_assigning_bytes[OD-bytes-vm017-vmN17-DoubleFloatPixelData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OL-bytes-vm019-vmN19-TrackPointIndexList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OV-bytes-vm020-vmN20-SelectorOVValue]"], "PASS_TO_PASS": ["pydicom/tests/test_valuerep.py::TestTM::test_pickling", "pydicom/tests/test_valuerep.py::TestTM::test_pickling_tm_from_time", "pydicom/tests/test_valuerep.py::TestTM::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestTM::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestTM::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestTM::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestTM::test_comparison", "pydicom/tests/test_valuerep.py::TestTM::test_time_behavior", "pydicom/tests/test_valuerep.py::TestDT::test_pickling", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_with_timezone", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_dt_from_datetime", "pydicom/tests/test_valuerep.py::TestDT::test_pickling_dt_from_datetime_with_timezone", "pydicom/tests/test_valuerep.py::TestDT::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestDT::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestDT::test_comparison", "pydicom/tests/test_valuerep.py::TestDT::test_datetime_behavior", "pydicom/tests/test_valuerep.py::TestDA::test_pickling", "pydicom/tests/test_valuerep.py::TestDA::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDA::test_str_and_repr", "pydicom/tests/test_valuerep.py::TestDA::test_comparison", "pydicom/tests/test_valuerep.py::TestDA::test_date_behavior", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[3.14159265358979]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[-1234.456e78]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E-5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E+5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[+1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[42", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[nan]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[3.141592653589793]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1,000]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[127.0.0.1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1.e]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.0-1.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.0-0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.0--0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.123-0.123]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.321--0.321]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1e-05-1e-05]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[3.141592653589793-3.14159265358979]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-3.141592653589793--3.1415926535898]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[5.385940192876374e-07-5.3859401929e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-5.385940192876374e-07--5.385940193e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[12342534378.125532-12342534378.1255]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[64070869985876.78-64070869985876.8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.7976931348623157e+308-1.797693135e+308]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_wrong_type", "pydicom/tests/test_valuerep.py::TestDS::test_empty_value", "pydicom/tests/test_valuerep.py::TestDS::test_float_values", "pydicom/tests/test_valuerep.py::TestDSfloat::test_pickling", "pydicom/tests/test_valuerep.py::TestDSfloat::test_new_empty", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str_value", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str", "pydicom/tests/test_valuerep.py::TestDSfloat::test_repr", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSdecimal", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_from_invalid_DS", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_length", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat_auto_format", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-nan]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan2]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestDSfloat::test_hash", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_pickling", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_float_value", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_new_empty", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_str_value", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSfloat", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSdecimal", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_repr", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_from_invalid_DS", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val4]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val5]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val6]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val7]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSdecimal_auto_format", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_hash", "pydicom/tests/test_valuerep.py::TestIS::test_empty_value", "pydicom/tests/test_valuerep.py::TestIS::test_str_value", "pydicom/tests/test_valuerep.py::TestIS::test_valid_value", "pydicom/tests/test_valuerep.py::TestIS::test_invalid_value", "pydicom/tests/test_valuerep.py::TestIS::test_pickling", "pydicom/tests/test_valuerep.py::TestIS::test_longint", "pydicom/tests/test_valuerep.py::TestIS::test_overflow", "pydicom/tests/test_valuerep.py::TestIS::test_str", "pydicom/tests/test_valuerep.py::TestIS::test_repr", "pydicom/tests/test_valuerep.py::TestIS::test_comparison_operators", "pydicom/tests/test_valuerep.py::TestIS::test_hash", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_default", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_enforce_valid_value", "pydicom/tests/test_valuerep.py::TestDecimalString::test_DS_decimal_set", "pydicom/tests/test_valuerep.py::TestDecimalString::test_valid_decimal_strings", "pydicom/tests/test_valuerep.py::TestDecimalString::test_invalid_decimal_strings", "pydicom/tests/test_valuerep.py::TestPersonName::test_last_first", "pydicom/tests/test_valuerep.py::TestPersonName::test_copy", "pydicom/tests/test_valuerep.py::TestPersonName::test_three_component", "pydicom/tests/test_valuerep.py::TestPersonName::test_formatting", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_kr", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_comp_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_caret_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_not_equal", "pydicom/tests/test_valuerep.py::TestPersonName::test_encoding_carried", "pydicom/tests/test_valuerep.py::TestPersonName::test_hash", "pydicom/tests/test_valuerep.py::TestPersonName::test_next", "pydicom/tests/test_valuerep.py::TestPersonName::test_iterator", "pydicom/tests/test_valuerep.py::TestPersonName::test_contains", "pydicom/tests/test_valuerep.py::TestPersonName::test_length", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_veterinary", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator_from_bytes", "pydicom/tests/test_valuerep.py::TestDateTime::test_date", "pydicom/tests/test_valuerep.py::TestDateTime::test_date_time", "pydicom/tests/test_valuerep.py::TestDateTime::test_time", "pydicom/tests/test_valuerep.py::test_person_name_unicode_warns", "pydicom/tests/test_valuerep.py::test_set_value[AE-str-vm00-vmN0-Receiver]", "pydicom/tests/test_valuerep.py::test_set_value[AS-str-vm01-vmN1-PatientAge]", "pydicom/tests/test_valuerep.py::test_set_value[AT-int-vm02-vmN2-OffendingElement]", "pydicom/tests/test_valuerep.py::test_set_value[CS-str-vm03-vmN3-QualityControlSubject]", "pydicom/tests/test_valuerep.py::test_set_value[DA-str-vm04-vmN4-PatientBirthDate]", "pydicom/tests/test_valuerep.py::test_set_value[DS-str-vm05-vmN5-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DS-int-vm06-vmN6-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DS-float-vm07-vmN7-PatientWeight]", "pydicom/tests/test_valuerep.py::test_set_value[DT-str-vm08-vmN8-AcquisitionDateTime]", "pydicom/tests/test_valuerep.py::test_set_value[FD-float-vm09-vmN9-RealWorldValueLUTData]", "pydicom/tests/test_valuerep.py::test_set_value[FL-float-vm010-vmN10-VectorAccuracy]", "pydicom/tests/test_valuerep.py::test_set_value[IS-str-vm011-vmN11-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[IS-int-vm012-vmN12-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[IS-float-vm013-vmN13-BeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[LO-str-vm014-vmN14-DataSetSubtype]", "pydicom/tests/test_valuerep.py::test_set_value[LT-str-vm015-vmN15-ExtendedCodeMeaning]", "pydicom/tests/test_valuerep.py::test_set_value[OB-bytes-vm016-vmN16-FillPattern]", "pydicom/tests/test_valuerep.py::test_set_value[OD-bytes-vm017-vmN17-DoubleFloatPixelData]", "pydicom/tests/test_valuerep.py::test_set_value[OF-bytes-vm018-vmN18-UValueData]", "pydicom/tests/test_valuerep.py::test_set_value[OL-bytes-vm019-vmN19-TrackPointIndexList]", "pydicom/tests/test_valuerep.py::test_set_value[OV-bytes-vm020-vmN20-SelectorOVValue]", "pydicom/tests/test_valuerep.py::test_set_value[OW-bytes-vm021-vmN21-TrianglePointIndexList]", "pydicom/tests/test_valuerep.py::test_set_value[PN-str-vm022-vmN22-PatientName]", "pydicom/tests/test_valuerep.py::test_set_value[SH-str-vm023-vmN23-CodeValue]", "pydicom/tests/test_valuerep.py::test_set_value[SL-int-vm024-vmN24-RationalNumeratorValue]", "pydicom/tests/test_valuerep.py::test_set_value[SQ-list-vm025-vmN25-BeamSequence]", "pydicom/tests/test_valuerep.py::test_set_value[SS-int-vm026-vmN26-SelectorSSValue]", "pydicom/tests/test_valuerep.py::test_set_value[ST-str-vm027-vmN27-InstitutionAddress]", "pydicom/tests/test_valuerep.py::test_set_value[SV-int-vm028-vmN28-SelectorSVValue]", "pydicom/tests/test_valuerep.py::test_set_value[TM-str-vm029-vmN29-StudyTime]", "pydicom/tests/test_valuerep.py::test_set_value[UC-str-vm030-vmN30-LongCodeValue]", "pydicom/tests/test_valuerep.py::test_set_value[UI-str-vm031-vmN31-SOPClassUID]", "pydicom/tests/test_valuerep.py::test_set_value[UL-int-vm032-vmN32-SimpleFrameList]", "pydicom/tests/test_valuerep.py::test_set_value[UN-bytes-vm033-vmN33-SelectorUNValue]", "pydicom/tests/test_valuerep.py::test_set_value[UR-str-vm034-vmN34-CodingSchemeURL]", "pydicom/tests/test_valuerep.py::test_set_value[US-int-vm035-vmN35-SourceAcquisitionBeamNumber]", "pydicom/tests/test_valuerep.py::test_set_value[UT-str-vm036-vmN36-StrainAdditionalInformation]", "pydicom/tests/test_valuerep.py::test_set_value[UV-int-vm037-vmN37-SelectorUVValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AE-str-vm00-vmN0-Receiver]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AS-str-vm01-vmN1-PatientAge]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[AT-int-vm02-vmN2-OffendingElement]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[CS-str-vm03-vmN3-QualityControlSubject]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DA-str-vm04-vmN4-PatientBirthDate]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-str-vm05-vmN5-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-int-vm06-vmN6-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DS-float-vm07-vmN7-PatientWeight]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[DT-str-vm08-vmN8-AcquisitionDateTime]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[FD-float-vm09-vmN9-RealWorldValueLUTData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[FL-float-vm010-vmN10-VectorAccuracy]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-str-vm011-vmN11-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-int-vm012-vmN12-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[IS-float-vm013-vmN13-BeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[LO-str-vm014-vmN14-DataSetSubtype]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[LT-str-vm015-vmN15-ExtendedCodeMeaning]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OB-bytes-vm016-vmN16-FillPattern]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OF-bytes-vm018-vmN18-UValueData]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[OW-bytes-vm021-vmN21-TrianglePointIndexList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[PN-str-vm022-vmN22-PatientName]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SH-str-vm023-vmN23-CodeValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SL-int-vm024-vmN24-RationalNumeratorValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SQ-list-vm025-vmN25-BeamSequence]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SS-int-vm026-vmN26-SelectorSSValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[ST-str-vm027-vmN27-InstitutionAddress]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[SV-int-vm028-vmN28-SelectorSVValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[TM-str-vm029-vmN29-StudyTime]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UC-str-vm030-vmN30-LongCodeValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UI-str-vm031-vmN31-SOPClassUID]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UL-int-vm032-vmN32-SimpleFrameList]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UN-bytes-vm033-vmN33-SelectorUNValue]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UR-str-vm034-vmN34-CodingSchemeURL]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[US-int-vm035-vmN35-SourceAcquisitionBeamNumber]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UT-str-vm036-vmN36-StrainAdditionalInformation]", "pydicom/tests/test_valuerep.py::test_assigning_bytes[UV-int-vm037-vmN37-SelectorUVValue]"], "environment_setup_commit": "506ecea8f378dc687d5c504788fc78810a190b7a"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-901", "base_commit": "3746878d8edf1cbda6fbcf35eec69f9ba79301ca", "patch": "diff --git a/pydicom/config.py b/pydicom/config.py\n--- a/pydicom/config.py\n+++ b/pydicom/config.py\n@@ -62,10 +62,7 @@ def DS_decimal(use_Decimal_boolean=True):\n \n # Logging system and debug function to change logging level\n logger = logging.getLogger('pydicom')\n-handler = logging.StreamHandler()\n-formatter = logging.Formatter(\"%(message)s\")\n-handler.setFormatter(formatter)\n-logger.addHandler(handler)\n+logger.addHandler(logging.NullHandler())\n \n \n import pydicom.pixel_data_handlers.numpy_handler as np_handler  # noqa\n@@ -110,16 +107,29 @@ def get_pixeldata(ds):\n \"\"\"\n \n \n-def debug(debug_on=True):\n-    \"\"\"Turn debugging of DICOM file reading and writing on or off.\n+def debug(debug_on=True, default_handler=True):\n+    \"\"\"Turn on/off debugging of DICOM file reading and writing.\n+\n     When debugging is on, file location and details about the\n     elements read at that location are logged to the 'pydicom'\n     logger using python's logging module.\n \n-    :param debug_on: True (default) to turn on debugging,\n-    False to turn off.\n+    Parameters\n+    ----------\n+    debug_on : bool, optional\n+        If True (default) then turn on debugging, False to turn off.\n+    default_handler : bool, optional\n+        If True (default) then use ``logging.StreamHandler()`` as the handler\n+        for log messages.\n     \"\"\"\n     global logger, debugging\n+\n+    if default_handler:\n+        handler = logging.StreamHandler()\n+        formatter = logging.Formatter(\"%(message)s\")\n+        handler.setFormatter(formatter)\n+        logger.addHandler(handler)\n+\n     if debug_on:\n         logger.setLevel(logging.DEBUG)\n         debugging = True\n@@ -129,4 +139,4 @@ def debug(debug_on=True):\n \n \n # force level=WARNING, in case logging default is set differently (issue 103)\n-debug(False)\n+debug(False, False)\n", "test_patch": "diff --git a/pydicom/tests/test_config.py b/pydicom/tests/test_config.py\nnew file mode 100644\n--- /dev/null\n+++ b/pydicom/tests/test_config.py\n@@ -0,0 +1,107 @@\n+# Copyright 2008-2019 pydicom authors. See LICENSE file for details.\n+\"\"\"Unit tests for the pydicom.config module.\"\"\"\n+\n+import logging\n+import sys\n+\n+import pytest\n+\n+from pydicom import dcmread\n+from pydicom.config import debug\n+from pydicom.data import get_testdata_files\n+\n+\n+DS_PATH = get_testdata_files(\"CT_small.dcm\")[0]\n+PYTEST = [int(x) for x in pytest.__version__.split('.')]\n+\n+\n+@pytest.mark.skipif(PYTEST[:2] < [3, 4], reason='no caplog')\n+class TestDebug(object):\n+    \"\"\"Tests for config.debug().\"\"\"\n+    def setup(self):\n+        self.logger = logging.getLogger('pydicom')\n+\n+    def teardown(self):\n+        # Reset to just NullHandler\n+        self.logger.handlers = [self.logger.handlers[0]]\n+\n+    def test_default(self, caplog):\n+        \"\"\"Test that the default logging handler is a NullHandler.\"\"\"\n+        assert 1 == len(self.logger.handlers)\n+        assert isinstance(self.logger.handlers[0], logging.NullHandler)\n+\n+        with caplog.at_level(logging.DEBUG, logger='pydicom'):\n+            ds = dcmread(DS_PATH)\n+\n+            assert \"Call to dcmread()\" not in caplog.text\n+            assert \"Reading File Meta Information preamble...\" in caplog.text\n+            assert \"Reading File Meta Information prefix...\" in caplog.text\n+            assert \"00000080: 'DICM' prefix found\" in caplog.text\n+\n+    def test_debug_on_handler_null(self, caplog):\n+        \"\"\"Test debug(True, False).\"\"\"\n+        debug(True, False)\n+        assert 1 == len(self.logger.handlers)\n+        assert isinstance(self.logger.handlers[0], logging.NullHandler)\n+\n+        with caplog.at_level(logging.DEBUG, logger='pydicom'):\n+            ds = dcmread(DS_PATH)\n+\n+            assert \"Call to dcmread()\" in caplog.text\n+            assert \"Reading File Meta Information preamble...\" in caplog.text\n+            assert \"Reading File Meta Information prefix...\" in caplog.text\n+            assert \"00000080: 'DICM' prefix found\" in caplog.text\n+            msg = (\n+                \"00009848: fc ff fc ff 4f 42 00 00 7e 00 00 00    \"\n+                \"(fffc, fffc) OB Length: 126\"\n+            )\n+            assert msg in caplog.text\n+\n+    def test_debug_off_handler_null(self, caplog):\n+        \"\"\"Test debug(False, False).\"\"\"\n+        debug(False, False)\n+        assert 1 == len(self.logger.handlers)\n+        assert isinstance(self.logger.handlers[0], logging.NullHandler)\n+\n+        with caplog.at_level(logging.DEBUG, logger='pydicom'):\n+            ds = dcmread(DS_PATH)\n+\n+            assert \"Call to dcmread()\" not in caplog.text\n+            assert \"Reading File Meta Information preamble...\" in caplog.text\n+            assert \"Reading File Meta Information prefix...\" in caplog.text\n+            assert \"00000080: 'DICM' prefix found\" in caplog.text\n+\n+    def test_debug_on_handler_stream(self, caplog):\n+        \"\"\"Test debug(True, True).\"\"\"\n+        debug(True, True)\n+        assert 2 == len(self.logger.handlers)\n+        assert isinstance(self.logger.handlers[0], logging.NullHandler)\n+        assert isinstance(self.logger.handlers[1], logging.StreamHandler)\n+\n+        with caplog.at_level(logging.DEBUG, logger='pydicom'):\n+            ds = dcmread(DS_PATH)\n+\n+            assert \"Call to dcmread()\" in caplog.text\n+            assert \"Reading File Meta Information preamble...\" in caplog.text\n+            assert \"Reading File Meta Information prefix...\" in caplog.text\n+            assert \"00000080: 'DICM' prefix found\" in caplog.text\n+            msg = (\n+                \"00009848: fc ff fc ff 4f 42 00 00 7e 00 00 00    \"\n+                \"(fffc, fffc) OB Length: 126\"\n+            )\n+            assert msg in caplog.text\n+\n+    def test_debug_off_handler_stream(self, caplog):\n+        \"\"\"Test debug(False, True).\"\"\"\n+        debug(False, True)\n+        assert 2 == len(self.logger.handlers)\n+        assert isinstance(self.logger.handlers[0], logging.NullHandler)\n+        assert isinstance(self.logger.handlers[1], logging.StreamHandler)\n+\n+        with caplog.at_level(logging.DEBUG, logger='pydicom'):\n+            ds = dcmread(DS_PATH)\n+\n+            assert \"Call to dcmread()\" not in caplog.text\n+            assert \"Reading File Meta Information preamble...\" in caplog.text\n+            assert \"Reading File Meta Information prefix...\" in caplog.text\n+            assert \"00000080: 'DICM' prefix found\" in caplog.text\n", "problem_statement": "pydicom should not define handler, formatter and log level.\nThe `config` module (imported when pydicom is imported) defines a handler and set the log level for the pydicom logger. This should not be the case IMO. It should be the responsibility of the client code of pydicom to configure the logging module to its convenience. Otherwise one end up having multiple logs record as soon as pydicom is imported:\r\n\r\nExample:\r\n```\r\nCould not import pillow\r\n2018-03-25 15:27:29,744 :: DEBUG :: pydicom \r\n  Could not import pillow\r\nCould not import jpeg_ls\r\n2018-03-25 15:27:29,745 :: DEBUG :: pydicom \r\n  Could not import jpeg_ls\r\nCould not import gdcm\r\n2018-03-25 15:27:29,745 :: DEBUG :: pydicom \r\n  Could not import gdcm\r\n``` \r\nOr am I missing something?\n", "hints_text": "In addition, I don't understand what the purpose of the `config.debug` function since the default behavor of the logging module in absence of configuartion seems to already be the one you want.\r\n\r\nFrom https://docs.python.org/3/howto/logging.html#configuring-logging-for-a-library:\r\n\r\n> If the using application does not use logging, and library code makes logging calls, then (as described in the previous section) events of severity WARNING and greater will be printed to sys.stderr. This is regarded as the best default behaviour.\r\n\r\nand\r\n\r\n>**It is strongly advised that you do not add any handlers other than NullHandler to your library\u2019s loggers.** This is because the configuration of handlers is the prerogative of the application developer who uses your library. The application developer knows their target audience and what handlers are most appropriate for their application: if you add handlers \u2018under the hood\u2019, you might well interfere with their ability to carry out unit tests and deliver logs which suit their requirements. \r\n\nI think you make good points here.  I support changing the logging to comply with python's suggested behavior.\r\n\r\n> In addition, I don't understand what the purpose of the config.debug function\r\n\r\nOne reason is that the core loop in pydicom (data_element_generator in filereader.py) is extremely optimized for speed - it checks the `debugging` flag set by config.debug, to avoid composing messages and doing function calls to logger when not needed.", "created_at": "2019-07-27T00:18:11Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_config.py::TestDebug::test_default", "pydicom/tests/test_config.py::TestDebug::test_debug_on_handler_null", "pydicom/tests/test_config.py::TestDebug::test_debug_off_handler_null", "pydicom/tests/test_config.py::TestDebug::test_debug_on_handler_stream", "pydicom/tests/test_config.py::TestDebug::test_debug_off_handler_stream"], "PASS_TO_PASS": [], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1469", "base_commit": "04e3ffd5ed2b8504e45ee0dc6bc7194d2440012d", "patch": "diff --git a/pvlib/clearsky.py b/pvlib/clearsky.py\n--- a/pvlib/clearsky.py\n+++ b/pvlib/clearsky.py\n@@ -960,8 +960,8 @@ def bird(zenith, airmass_relative, aod380, aod500, precipitable_water,\n         Extraterrestrial radiation [W/m^2], defaults to 1364[W/m^2]\n     asymmetry : numeric\n         Asymmetry factor, defaults to 0.85\n-    albedo : numeric\n-        Albedo, defaults to 0.2\n+    albedo : numeric, default 0.2\n+        Ground surface albedo. [unitless]\n \n     Returns\n     -------\ndiff --git a/pvlib/irradiance.py b/pvlib/irradiance.py\n--- a/pvlib/irradiance.py\n+++ b/pvlib/irradiance.py\n@@ -304,7 +304,7 @@ def beam_component(surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n def get_total_irradiance(surface_tilt, surface_azimuth,\n                          solar_zenith, solar_azimuth,\n                          dni, ghi, dhi, dni_extra=None, airmass=None,\n-                         albedo=.25, surface_type=None,\n+                         albedo=0.25, surface_type=None,\n                          model='isotropic',\n                          model_perez='allsitescomposite1990'):\n     r\"\"\"\n@@ -344,7 +344,7 @@ def get_total_irradiance(surface_tilt, surface_azimuth,\n     airmass : None or numeric, default None\n         Relative airmass (not adjusted for pressure). [unitless]\n     albedo : numeric, default 0.25\n-        Surface albedo. [unitless]\n+        Ground surface albedo. [unitless]\n     surface_type : None or str, default None\n         Surface type. See :py:func:`~pvlib.irradiance.get_ground_diffuse` for\n         the list of accepted values.\n@@ -1872,7 +1872,7 @@ def gti_dirint(poa_global, aoi, solar_zenith, solar_azimuth, times,\n         applied.\n \n     albedo : numeric, default 0.25\n-        Surface albedo\n+        Ground surface albedo. [unitless]\n \n     model : String, default 'perez'\n         Irradiance model.  See :py:func:`get_sky_diffuse` for allowed values.\ndiff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -1339,6 +1339,16 @@ def _prep_inputs_solar_pos(self, weather):\n             **kwargs)\n         return self\n \n+    def _prep_inputs_albedo(self, weather):\n+        \"\"\"\n+        Get albedo from weather\n+        \"\"\"\n+        try:\n+            self.results.albedo = _tuple_from_dfs(weather, 'albedo')\n+        except KeyError:\n+            self.results.albedo = None\n+        return self\n+\n     def _prep_inputs_airmass(self):\n         \"\"\"\n         Assign airmass\n@@ -1471,11 +1481,17 @@ def prepare_inputs(self, weather):\n \n         Parameters\n         ----------\n-        weather : DataFrame, or tuple or list of DataFrame\n+        weather : DataFrame, or tuple or list of DataFrames\n             Required column names include ``'dni'``, ``'ghi'``, ``'dhi'``.\n-            Optional column names are ``'wind_speed'``, ``'temp_air'``; if not\n+            Optional column names are ``'wind_speed'``, ``'temp_air'``,\n+            ``'albedo'``.\n+\n+            If optional columns ``'wind_speed'``, ``'temp_air'`` are not\n             provided, air temperature of 20 C and wind speed\n-            of 0 m/s will be added to the DataFrame.\n+            of 0 m/s will be added to the `weather` DataFrame.\n+\n+            If optional column ``'albedo'`` is provided, albedo values in the\n+            ModelChain's PVSystem.arrays are ignored.\n \n             If `weather` is a tuple or list, it must be of the same length and\n             order as the Arrays of the ModelChain's PVSystem.\n@@ -1494,7 +1510,7 @@ def prepare_inputs(self, weather):\n         Notes\n         -----\n         Assigns attributes to ``results``: ``times``, ``weather``,\n-        ``solar_position``, ``airmass``, ``total_irrad``, ``aoi``\n+        ``solar_position``, ``airmass``, ``total_irrad``, ``aoi``, ``albedo``.\n \n         See also\n         --------\n@@ -1507,6 +1523,7 @@ def prepare_inputs(self, weather):\n \n         self._prep_inputs_solar_pos(weather)\n         self._prep_inputs_airmass()\n+        self._prep_inputs_albedo(weather)\n \n         # PVSystem.get_irradiance and SingleAxisTracker.get_irradiance\n         # and PVSystem.get_aoi and SingleAxisTracker.get_aoi\n@@ -1531,6 +1548,7 @@ def prepare_inputs(self, weather):\n             _tuple_from_dfs(self.results.weather, 'dni'),\n             _tuple_from_dfs(self.results.weather, 'ghi'),\n             _tuple_from_dfs(self.results.weather, 'dhi'),\n+            albedo=self.results.albedo,\n             airmass=self.results.airmass['airmass_relative'],\n             model=self.transposition_model\n         )\n@@ -1724,16 +1742,32 @@ def run_model(self, weather):\n         Parameters\n         ----------\n         weather : DataFrame, or tuple or list of DataFrame\n-            Irradiance column names must include ``'dni'``, ``'ghi'``, and\n-            ``'dhi'``. If optional columns ``'temp_air'`` and ``'wind_speed'``\n+            Column names must include:\n+\n+            - ``'dni'``\n+            - ``'ghi'``\n+            - ``'dhi'``\n+\n+            Optional columns are:\n+\n+            - ``'temp_air'``\n+            - ``'cell_temperature'``\n+            - ``'module_temperature'``\n+            - ``'wind_speed'``\n+            - ``'albedo'``\n+\n+            If optional columns ``'temp_air'`` and ``'wind_speed'``\n             are not provided, air temperature of 20 C and wind speed of 0 m/s\n             are added to the DataFrame. If optional column\n             ``'cell_temperature'`` is provided, these values are used instead\n-            of `temperature_model`. If optional column `module_temperature`\n+            of `temperature_model`. If optional column ``'module_temperature'``\n             is provided, `temperature_model` must be ``'sapm'``.\n \n-            If list or tuple, must be of the same length and order as the\n-            Arrays of the ModelChain's PVSystem.\n+            If optional column ``'albedo'`` is provided, ``'albedo'`` may not\n+            be present on the ModelChain's PVSystem.Arrays.\n+\n+            If weather is a list or tuple, it must be of the same length and\n+            order as the Arrays of the ModelChain's PVSystem.\n \n         Returns\n         -------\ndiff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\n--- a/pvlib/pvsystem.py\n+++ b/pvlib/pvsystem.py\n@@ -134,7 +134,7 @@ class PVSystem:\n         a single array is created from the other parameters (e.g.\n         `surface_tilt`, `surface_azimuth`). Must contain at least one Array,\n         if length of arrays is 0 a ValueError is raised. If `arrays` is\n-        specified the following parameters are ignored:\n+        specified the following PVSystem parameters are ignored:\n \n         - `surface_tilt`\n         - `surface_azimuth`\n@@ -157,13 +157,16 @@ class PVSystem:\n         North=0, East=90, South=180, West=270.\n \n     albedo : None or float, default None\n-        The ground albedo. If ``None``, will attempt to use\n-        ``surface_type`` and ``irradiance.SURFACE_ALBEDOS``\n-        to lookup albedo.\n+        Ground surface albedo. If ``None``, then ``surface_type`` is used\n+        to look up a value in ``irradiance.SURFACE_ALBEDOS``.\n+        If ``surface_type`` is also None then a ground surface albedo\n+        of 0.25 is used. For time-dependent albedos, add ``'albedo'`` to\n+        the input ``'weather'`` DataFrame for\n+        :py:class:`pvlib.modelchain.ModelChain` methods.\n \n     surface_type : None or string, default None\n-        The ground surface type. See ``irradiance.SURFACE_ALBEDOS``\n-        for valid values.\n+        The ground surface type. See ``irradiance.SURFACE_ALBEDOS`` for\n+        valid values.\n \n     module : None or string, default None\n         The model name of the modules.\n@@ -333,30 +336,32 @@ def get_aoi(self, solar_zenith, solar_azimuth):\n \n     @_unwrap_single_value\n     def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n-                       **kwargs):\n+                       albedo=None, dni_extra=None, airmass=None,\n+                       model='haydavies', **kwargs):\n         \"\"\"\n         Uses the :py:func:`irradiance.get_total_irradiance` function to\n         calculate the plane of array irradiance components on a tilted\n-        surface defined by ``self.surface_tilt``,\n-        ``self.surface_azimuth``, and ``self.albedo``.\n+        surface defined by ``self.surface_tilt`` and ``self.surface_azimuth```.\n \n         Parameters\n         ----------\n-        solar_zenith : float or Series.\n+        solar_zenith : float or Series\n             Solar zenith angle.\n-        solar_azimuth : float or Series.\n+        solar_azimuth : float or Series\n             Solar azimuth angle.\n         dni : float or Series or tuple of float or Series\n-            Direct Normal Irradiance\n+            Direct Normal Irradiance. [W/m2]\n         ghi : float or Series or tuple of float or Series\n-            Global horizontal irradiance\n+            Global horizontal irradiance. [W/m2]\n         dhi : float or Series or tuple of float or Series\n-            Diffuse horizontal irradiance\n-        dni_extra : None, float or Series, default None\n-            Extraterrestrial direct normal irradiance\n+            Diffuse horizontal irradiance. [W/m2]\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n+        dni_extra : None, float, Series or tuple of float or Series,\n+            default None\n+            Extraterrestrial direct normal irradiance. [W/m2]\n         airmass : None, float or Series, default None\n-            Airmass\n+            Airmass. [unitless]\n         model : String, default 'haydavies'\n             Irradiance model.\n \n@@ -376,17 +381,26 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         poa_irradiance : DataFrame or tuple of DataFrame\n             Column names are: ``'poa_global', 'poa_direct', 'poa_diffuse',\n             'poa_sky_diffuse', 'poa_ground_diffuse'``.\n+\n+        See also\n+        --------\n+        :py:func:`pvlib.irradiance.get_total_irradiance`\n         \"\"\"\n         dni = self._validate_per_array(dni, system_wide=True)\n         ghi = self._validate_per_array(ghi, system_wide=True)\n         dhi = self._validate_per_array(dhi, system_wide=True)\n+\n+        albedo = self._validate_per_array(albedo, system_wide=True)\n+\n         return tuple(\n             array.get_irradiance(solar_zenith, solar_azimuth,\n                                  dni, ghi, dhi,\n-                                 dni_extra, airmass, model,\n+                                 albedo=albedo,\n+                                 dni_extra=dni_extra, airmass=airmass,\n+                                 model=model,\n                                  **kwargs)\n-            for array, dni, ghi, dhi in zip(\n-                self.arrays, dni, ghi, dhi\n+            for array, dni, ghi, dhi, albedo in zip(\n+                self.arrays, dni, ghi, dhi, albedo\n             )\n         )\n \n@@ -1258,14 +1272,14 @@ class Array:\n         If not provided, a FixedMount with zero tilt is used.\n \n     albedo : None or float, default None\n-        The ground albedo. If ``None``, will attempt to use\n-        ``surface_type`` to look up an albedo value in\n-        ``irradiance.SURFACE_ALBEDOS``. If a surface albedo\n-        cannot be found then 0.25 is used.\n+        Ground surface albedo. If ``None``, then ``surface_type`` is used\n+        to look up a value in ``irradiance.SURFACE_ALBEDOS``.\n+        If ``surface_type`` is also None then a ground surface albedo\n+        of 0.25 is used.\n \n     surface_type : None or string, default None\n-        The ground surface type. See ``irradiance.SURFACE_ALBEDOS``\n-        for valid values.\n+        The ground surface type. See ``irradiance.SURFACE_ALBEDOS`` for valid\n+        values.\n \n     module : None or string, default None\n         The model name of the modules.\n@@ -1425,15 +1439,14 @@ def get_aoi(self, solar_zenith, solar_azimuth):\n                               solar_zenith, solar_azimuth)\n \n     def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n-                       **kwargs):\n+                       albedo=None, dni_extra=None, airmass=None,\n+                       model='haydavies', **kwargs):\n         \"\"\"\n         Get plane of array irradiance components.\n \n         Uses the :py:func:`pvlib.irradiance.get_total_irradiance` function to\n         calculate the plane of array irradiance components for a surface\n-        defined by ``self.surface_tilt`` and ``self.surface_azimuth`` with\n-        albedo ``self.albedo``.\n+        defined by ``self.surface_tilt`` and ``self.surface_azimuth``.\n \n         Parameters\n         ----------\n@@ -1442,15 +1455,17 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         solar_azimuth : float or Series.\n             Solar azimuth angle.\n         dni : float or Series\n-            Direct Normal Irradiance\n-        ghi : float or Series\n+            Direct normal irradiance. [W/m2]\n+        ghi : float or Series. [W/m2]\n             Global horizontal irradiance\n         dhi : float or Series\n-            Diffuse horizontal irradiance\n+            Diffuse horizontal irradiance. [W/m2]\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n         dni_extra : None, float or Series, default None\n-            Extraterrestrial direct normal irradiance\n+            Extraterrestrial direct normal irradiance. [W/m2]\n         airmass : None, float or Series, default None\n-            Airmass\n+            Airmass. [unitless]\n         model : String, default 'haydavies'\n             Irradiance model.\n \n@@ -1463,7 +1478,14 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n         poa_irradiance : DataFrame\n             Column names are: ``'poa_global', 'poa_direct', 'poa_diffuse',\n             'poa_sky_diffuse', 'poa_ground_diffuse'``.\n+\n+        See also\n+        --------\n+        :py:func:`pvlib.irradiance.get_total_irradiance`\n         \"\"\"\n+        if albedo is None:\n+            albedo = self.albedo\n+\n         # not needed for all models, but this is easier\n         if dni_extra is None:\n             dni_extra = irradiance.get_extra_radiation(solar_zenith.index)\n@@ -1476,10 +1498,10 @@ def get_irradiance(self, solar_zenith, solar_azimuth, dni, ghi, dhi,\n                                                orientation['surface_azimuth'],\n                                                solar_zenith, solar_azimuth,\n                                                dni, ghi, dhi,\n+                                               albedo=albedo,\n                                                dni_extra=dni_extra,\n                                                airmass=airmass,\n                                                model=model,\n-                                               albedo=self.albedo,\n                                                **kwargs)\n \n     def get_iam(self, aoi, iam_model='physical'):\ndiff --git a/pvlib/tracking.py b/pvlib/tracking.py\n--- a/pvlib/tracking.py\n+++ b/pvlib/tracking.py\n@@ -187,7 +187,8 @@ def get_aoi(self, surface_tilt, surface_azimuth, solar_zenith,\n     @_unwrap_single_value\n     def get_irradiance(self, surface_tilt, surface_azimuth,\n                        solar_zenith, solar_azimuth, dni, ghi, dhi,\n-                       dni_extra=None, airmass=None, model='haydavies',\n+                       albedo=None, dni_extra=None, airmass=None,\n+                       model='haydavies',\n                        **kwargs):\n         \"\"\"\n         Uses the :func:`irradiance.get_total_irradiance` function to\n@@ -214,6 +215,8 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n             Global horizontal irradiance\n         dhi : float or Series\n             Diffuse horizontal irradiance\n+        albedo : None, float or Series, default None\n+            Ground surface albedo. [unitless]\n         dni_extra : float or Series, default None\n             Extraterrestrial direct normal irradiance\n         airmass : float or Series, default None\n@@ -244,6 +247,13 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n         ghi = self._validate_per_array(ghi, system_wide=True)\n         dhi = self._validate_per_array(dhi, system_wide=True)\n \n+        if albedo is None:\n+            # assign default albedo here because SingleAxisTracker\n+            # initializes albedo to None\n+            albedo = 0.25\n+\n+        albedo = self._validate_per_array(albedo, system_wide=True)\n+\n         return tuple(\n             irradiance.get_total_irradiance(\n                 surface_tilt,\n@@ -254,10 +264,10 @@ def get_irradiance(self, surface_tilt, surface_azimuth,\n                 dni_extra=dni_extra,\n                 airmass=airmass,\n                 model=model,\n-                albedo=self.arrays[0].albedo,\n+                albedo=albedo,\n                 **kwargs)\n-            for array, dni, ghi, dhi in zip(\n-                self.arrays, dni, ghi, dhi\n+            for array, dni, ghi, dhi, albedo in zip(\n+                self.arrays, dni, ghi, dhi, albedo\n             )\n         )\n \n", "test_patch": "diff --git a/pvlib/tests/test_clearsky.py b/pvlib/tests/test_clearsky.py\n--- a/pvlib/tests/test_clearsky.py\n+++ b/pvlib/tests/test_clearsky.py\n@@ -756,6 +756,30 @@ def test_bird():\n     assert np.allclose(\n         testdata['Dif Hz'].where(dusk, 0.), diffuse_horz[1:48], rtol=1e-3\n     )\n+    # repeat test with albedo as a Series\n+    alb_series = pd.Series(0.2, index=times)\n+    irrads = clearsky.bird(\n+        zenith, airmass, aod_380nm, aod_500nm, h2o_cm, o3_cm, press_mB * 100.,\n+        etr, b_a, alb_series\n+    )\n+    Eb, Ebh, Gh, Dh = (irrads[_] for _ in field_names)\n+    direct_beam = pd.Series(np.where(dawn, Eb, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Direct Beam'].where(dusk, 0.), direct_beam[1:48], rtol=1e-3\n+    )\n+    direct_horz = pd.Series(np.where(dawn, Ebh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Direct Hz'].where(dusk, 0.), direct_horz[1:48], rtol=1e-3\n+    )\n+    global_horz = pd.Series(np.where(dawn, Gh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Global Hz'].where(dusk, 0.), global_horz[1:48], rtol=1e-3\n+    )\n+    diffuse_horz = pd.Series(np.where(dawn, Dh, 0.), index=times).fillna(0.)\n+    assert np.allclose(\n+        testdata['Dif Hz'].where(dusk, 0.), diffuse_horz[1:48], rtol=1e-3\n+    )\n+\n     # test keyword parameters\n     irrads2 = clearsky.bird(\n         zenith, airmass, aod_380nm, aod_500nm, h2o_cm, dni_extra=etr\ndiff --git a/pvlib/tests/test_irradiance.py b/pvlib/tests/test_irradiance.py\n--- a/pvlib/tests/test_irradiance.py\n+++ b/pvlib/tests/test_irradiance.py\n@@ -120,29 +120,38 @@ def test_get_extra_radiation_invalid():\n         irradiance.get_extra_radiation(300, method='invalid')\n \n \n-def test_grounddiffuse_simple_float():\n+def test_get_ground_diffuse_simple_float():\n     result = irradiance.get_ground_diffuse(40, 900)\n     assert_allclose(result, 26.32000014911496)\n \n \n-def test_grounddiffuse_simple_series(irrad_data):\n+def test_get_ground_diffuse_simple_series(irrad_data):\n     ground_irrad = irradiance.get_ground_diffuse(40, irrad_data['ghi'])\n     assert ground_irrad.name == 'diffuse_ground'\n \n \n-def test_grounddiffuse_albedo_0(irrad_data):\n+def test_get_ground_diffuse_albedo_0(irrad_data):\n     ground_irrad = irradiance.get_ground_diffuse(\n         40, irrad_data['ghi'], albedo=0)\n     assert 0 == ground_irrad.all()\n \n \n+def test_get_ground_diffuse_albedo_series(times):\n+    albedo = pd.Series(0.2, index=times)\n+    ground_irrad = irradiance.get_ground_diffuse(\n+        45, pd.Series(1000, index=times), albedo)\n+    expected = albedo * 0.5 * (1 - np.sqrt(2) / 2.) * 1000\n+    expected.name = 'diffuse_ground'\n+    assert_series_equal(ground_irrad, expected)\n+\n+\n def test_grounddiffuse_albedo_invalid_surface(irrad_data):\n     with pytest.raises(KeyError):\n         irradiance.get_ground_diffuse(\n             40, irrad_data['ghi'], surface_type='invalid')\n \n \n-def test_grounddiffuse_albedo_surface(irrad_data):\n+def test_get_ground_diffuse_albedo_surface(irrad_data):\n     result = irradiance.get_ground_diffuse(40, irrad_data['ghi'],\n                                            surface_type='sand')\n     assert_allclose(result, [0, 3.731058, 48.778813, 12.035025], atol=1e-4)\n@@ -387,6 +396,26 @@ def test_get_total_irradiance(irrad_data, ephem_data, dni_et,\n                                           'poa_ground_diffuse']\n \n \n+def test_get_total_irradiance_albedo(\n+        irrad_data, ephem_data, dni_et, relative_airmass):\n+    models = ['isotropic', 'klucher',\n+              'haydavies', 'reindl', 'king', 'perez']\n+    albedo = pd.Series(0.2, index=ephem_data.index)\n+    for model in models:\n+        total = irradiance.get_total_irradiance(\n+            32, 180,\n+            ephem_data['apparent_zenith'], ephem_data['azimuth'],\n+            dni=irrad_data['dni'], ghi=irrad_data['ghi'],\n+            dhi=irrad_data['dhi'],\n+            dni_extra=dni_et, airmass=relative_airmass,\n+            model=model,\n+            albedo=albedo)\n+\n+        assert total.columns.tolist() == ['poa_global', 'poa_direct',\n+                                          'poa_diffuse', 'poa_sky_diffuse',\n+                                          'poa_ground_diffuse']\n+\n+\n @pytest.mark.parametrize('model', ['isotropic', 'klucher',\n                                    'haydavies', 'reindl', 'king', 'perez'])\n def test_get_total_irradiance_scalars(model):\n@@ -698,6 +727,14 @@ def test_gti_dirint():\n \n     assert_frame_equal(output, expected)\n \n+    # test with albedo as a Series\n+    albedo = pd.Series(0.05, index=times)\n+    output = irradiance.gti_dirint(\n+        poa_global, aoi, zenith, azimuth, times, surface_tilt, surface_azimuth,\n+        albedo=albedo)\n+\n+    assert_frame_equal(output, expected)\n+\n     # test temp_dew input\n     temp_dew = np.array([70, 80, 20])\n     output = irradiance.gti_dirint(\ndiff --git a/pvlib/tests/test_modelchain.py b/pvlib/tests/test_modelchain.py\n--- a/pvlib/tests/test_modelchain.py\n+++ b/pvlib/tests/test_modelchain.py\n@@ -497,6 +497,24 @@ def test_prepare_inputs_multi_weather(\n     assert len(mc.results.total_irrad) == num_arrays\n \n \n+@pytest.mark.parametrize(\"input_type\", [tuple, list])\n+def test_prepare_inputs_albedo_in_weather(\n+        sapm_dc_snl_ac_system_Array, location, input_type):\n+    times = pd.date_range(start='20160101 1200-0700',\n+                          end='20160101 1800-0700', freq='6H')\n+    mc = ModelChain(sapm_dc_snl_ac_system_Array, location)\n+    weather = pd.DataFrame({'ghi': 1, 'dhi': 1, 'dni': 1, 'albedo': 0.5},\n+                           index=times)\n+    # weather as a single DataFrame\n+    mc.prepare_inputs(weather)\n+    num_arrays = sapm_dc_snl_ac_system_Array.num_arrays\n+    assert len(mc.results.total_irrad) == num_arrays\n+    # repeat with tuple of weather\n+    mc.prepare_inputs(input_type((weather, weather)))\n+    num_arrays = sapm_dc_snl_ac_system_Array.num_arrays\n+    assert len(mc.results.total_irrad) == num_arrays\n+\n+\n def test_prepare_inputs_no_irradiance(sapm_dc_snl_ac_system, location):\n     mc = ModelChain(sapm_dc_snl_ac_system, location)\n     weather = pd.DataFrame()\ndiff --git a/pvlib/tests/test_pvsystem.py b/pvlib/tests/test_pvsystem.py\n--- a/pvlib/tests/test_pvsystem.py\n+++ b/pvlib/tests/test_pvsystem.py\n@@ -1689,14 +1689,37 @@ def test_PVSystem_get_irradiance():\n                                        irrads['dhi'])\n \n     expected = pd.DataFrame(data=np.array(\n-        [[ 883.65494055,  745.86141676,  137.79352379,  126.397131  ,\n-              11.39639279],\n-           [   0.        ,   -0.        ,    0.        ,    0.        ,    0.        ]]),\n+        [[883.65494055, 745.86141676, 137.79352379, 126.397131, 11.39639279],\n+         [0., -0., 0., 0., 0.]]),\n                             columns=['poa_global', 'poa_direct',\n                                      'poa_diffuse', 'poa_sky_diffuse',\n                                      'poa_ground_diffuse'],\n                             index=times)\n+    assert_frame_equal(irradiance, expected, check_less_precise=2)\n+\n \n+def test_PVSystem_get_irradiance_albedo():\n+    system = pvsystem.PVSystem(surface_tilt=32, surface_azimuth=135)\n+    times = pd.date_range(start='20160101 1200-0700',\n+                          end='20160101 1800-0700', freq='6H')\n+    location = Location(latitude=32, longitude=-111)\n+    solar_position = location.get_solarposition(times)\n+    irrads = pd.DataFrame({'dni': [900, 0], 'ghi': [600, 0], 'dhi': [100, 0],\n+                           'albedo': [0.5, 0.5]},\n+                          index=times)\n+    # albedo as a Series\n+    irradiance = system.get_irradiance(solar_position['apparent_zenith'],\n+                                       solar_position['azimuth'],\n+                                       irrads['dni'],\n+                                       irrads['ghi'],\n+                                       irrads['dhi'],\n+                                       albedo=irrads['albedo'])\n+    expected = pd.DataFrame(data=np.array(\n+        [[895.05134334, 745.86141676, 149.18992658, 126.397131, 22.79279558],\n+         [0., -0., 0., 0., 0.]]),\n+        columns=['poa_global', 'poa_direct', 'poa_diffuse', 'poa_sky_diffuse',\n+                 'poa_ground_diffuse'],\n+        index=times)\n     assert_frame_equal(irradiance, expected, check_less_precise=2)\n \n \ndiff --git a/pvlib/tests/test_tracking.py b/pvlib/tests/test_tracking.py\n--- a/pvlib/tests/test_tracking.py\n+++ b/pvlib/tests/test_tracking.py\n@@ -393,6 +393,25 @@ def test_get_irradiance():\n \n     assert_frame_equal(irradiance, expected, check_less_precise=2)\n \n+    # test with albedo as a Series\n+    irrads['albedo'] = [0.5, 0.5]\n+    with np.errstate(invalid='ignore'):\n+        irradiance = system.get_irradiance(tracker_data['surface_tilt'],\n+                                           tracker_data['surface_azimuth'],\n+                                           solar_zenith,\n+                                           solar_azimuth,\n+                                           irrads['dni'],\n+                                           irrads['ghi'],\n+                                           irrads['dhi'],\n+                                           albedo=irrads['albedo'])\n+\n+    expected = pd.Series(data=[21.05514984, nan], index=times,\n+                         name='poa_ground_diffuse')\n+\n+    assert_series_equal(irradiance['poa_ground_diffuse'], expected,\n+                        check_less_precise=2)\n+\n+\n \n def test_SingleAxisTracker___repr__():\n     with pytest.warns(pvlibDeprecationWarning):\n", "problem_statement": "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", "hints_text": "I think I agree that it would make more sense to pass around ground albedo with the weather data instead of treating it as a characteristic of the array.  \r\n\r\n> Albedo is treated as a scalar constant in pvlib\r\n\r\n> An 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. \r\n\r\nIt is true that the docs for `Array` and `PVSystem` say that `albedo` is a float, but I think it also works to set albedo to a time series that matches the weather you pass to `ModelChain.run_model()`.  At least that saves you from looping w/ scalar albedo values.  \nI agree that albedo can change with e.g., precipitation or season, but it is also a property of the bare ground surface, and it is that perspective that put albedo with the system parameters. One use case for not having albedo in the weather data would be to evaluate the effect of different ground cover on array output.\r\n\r\nI am only away of the NSRDB offering albedo with weather data; are there other sources?\r\n\r\nI'm +1 on allowing `albedo` to be a Series. I'm neutral on bundling albedo with weather data, but I don't see a better option.  We only have two data structures that supply `ModelChain`: the `weather` DataFrame, and the `PVSystem` instance. I don't think it is practical to create a third just for `albedo`, and it isn't any more work to add or modify `albedo` to `weather` than it is to extract `albedo` from downloaded weather data and add it to `PVSystem`.\r\n\r\n\nTo clarify my above message, I think it *already* works to set `PVSystem.albedo` or `Array.albedo` to a Series, despite the docs saying it must be float.  \r\n\r\n> are there other sources?\r\n\r\nA non-exhaustive list of examples: [SolarAnywhere](https://www.solaranywhere.com/support/data-fields/albedo/), [SolarGIS](https://solargis.com/docs/getting-started/data-parameters), [MERRA2](https://disc.gsfc.nasa.gov/datasets/M2TMNXRAD_5.12.4/summary)\nDoes anyone know if including time-specific albedo this has been shown to be even more important with bifacials?\r\n\r\n(I would think yes.)\n> Does anyone know if including time-specific albedo this has been shown to be even more important with bifacials?\r\n> \r\n> (I would think yes.)\r\n\r\nYes, it is more important than for single-sided modules. There are ground surfaces where the albedo depends on the solar elevation and hence time of day.\r\n\r\nOne caution about albedo from satellite-derived irradiance: those values are at least km^2 scale, and are observed from space, whereas a PV model is assuming that the albedo is localized (m^2) and has been determined from the irradiance reaching the ground. [SolarAnywhere ](https://www.solaranywhere.com/support/data-fields/albedo/)provides an informative list of caveats.\r\n\r\nThe good news is that the uncertainty in albedo is typically secondary to uncertainty in other data such as GHI, when considering uncertainty in energy production.\r\n\nWhich is the better course of action?\r\n\r\n1. Leave `albedo` on `PVsystem` or `Array`. Edit the docstrings and add tests to make it explicit that `PVSystem.albedo` or `Array.albedo` can be a Series. Advantages: avoids deprecating and removing `PVSystem.albedo`. Downside: users obtaining albedo from weather data sources have an extra step to perform when using `ModelChain` methods.\r\n2. Add `albedo` as an optional column in `weather`, and have `ModelChain` methods use `weather['albedo']` instead of `PVSystem.albedo` when `weather['albedo']` is present. Advantages: convenient for ModelChain users, and avoids deprecating `PVsystem.albedo`. Disadvatanges: potential for confusion when a user also assigns `PVSystem.albedo`. \r\n3. Move `albedo` from `PVSystem` to `weather`. Same advantages as 2, but requires deprecation, and adds an extra step for users who aren't getting albedo with weather data, and would probably lead to adding code to `ModelChain` similar to the code that infers albedo from `PVSystem.surface_type`.\n@cwhanse I lean toward option 1. However, it seems that for option 1, a user might have to duplicate computations already done in the `ModelChain` in the preliminary step of using the weather to compute the existing `PVSystem.albedo` (as some formulaic combination of ground conditions and weather). Do you know if this is really an issue?\nThanks for the discussion around this feature.  I favor Option 2, noting that PVSystem.albedo is already an optional argument.  Option 1 is prone to bugs and cases need to be handled checking for mismatches in series/dataframe lengths or perhaps same series lengths but different indices.  I would discourage Option 3, as it seems there is both a basis for and utility in retaining albedo as a property of PVSystem.\nFor option 2, I would say raise an exception (or at very least a warning) if two albedos are specified. This could definitely be the source of a subtle computational bug when users don't realize a weather file they pulled is doing something they didn't intend.\n+1 for option 2, and I agree with @campanelli-sunpower that an exception should be raised if two albedos are specified\n+1 for option 2, but I like keeping the old method for the case where you just want to set either a single annual albedo or monthly albedos (tho not sure if this option currently exists). I agree also raise exception if both are specified - we can always change it later.", "created_at": "2022-06-09T01:46:04Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance_albedo", "pvlib/tests/test_tracking.py::test_get_irradiance"], "PASS_TO_PASS": ["pvlib/tests/test_clearsky.py::test_ineichen_series", "pvlib/tests/test_clearsky.py::test_ineichen_series_perez_enhancement", "pvlib/tests/test_clearsky.py::test_ineichen_scalar_input", "pvlib/tests/test_clearsky.py::test_ineichen_nans", "pvlib/tests/test_clearsky.py::test_ineichen_arrays", "pvlib/tests/test_clearsky.py::test_ineichen_dni_extra", "pvlib/tests/test_clearsky.py::test_ineichen_altitude", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_leapyear", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_nointerp", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_months", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_months_leapyear", "pvlib/tests/test_clearsky.py::test_lookup_linke_turbidity_nointerp_months", "pvlib/tests/test_clearsky.py::test_haurwitz", "pvlib/tests/test_clearsky.py::test_simplified_solis_scalar_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_scalar_neg_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_series_elevation", "pvlib/tests/test_clearsky.py::test_simplified_solis_dni_extra", "pvlib/tests/test_clearsky.py::test_simplified_solis_pressure", "pvlib/tests/test_clearsky.py::test_simplified_solis_aod700", "pvlib/tests/test_clearsky.py::test_simplified_solis_precipitable_water", "pvlib/tests/test_clearsky.py::test_simplified_solis_small_scalar_pw", "pvlib/tests/test_clearsky.py::test_simplified_solis_return_arrays", "pvlib/tests/test_clearsky.py::test_simplified_solis_nans_arrays", "pvlib/tests/test_clearsky.py::test_simplified_solis_nans_series", "pvlib/tests/test_clearsky.py::test_linke_turbidity_corners", "pvlib/tests/test_clearsky.py::test_degrees_to_index_1", "pvlib/tests/test_clearsky.py::test_detect_clearsky", "pvlib/tests/test_clearsky.py::test_detect_clearsky_defaults", "pvlib/tests/test_clearsky.py::test_detect_clearsky_components", "pvlib/tests/test_clearsky.py::test_detect_clearsky_iterations", "pvlib/tests/test_clearsky.py::test_detect_clearsky_kwargs", "pvlib/tests/test_clearsky.py::test_detect_clearsky_window", "pvlib/tests/test_clearsky.py::test_detect_clearsky_time_interval", "pvlib/tests/test_clearsky.py::test_detect_clearsky_arrays", "pvlib/tests/test_clearsky.py::test_detect_clearsky_irregular_times", "pvlib/tests/test_clearsky.py::test_detect_clearsky_missing_index", "pvlib/tests/test_clearsky.py::test__line_length_windowed", "pvlib/tests/test_clearsky.py::test__max_diff_windowed", "pvlib/tests/test_clearsky.py::test__calc_stats", "pvlib/tests/test_clearsky.py::test_bird", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[asce-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[spencer-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[nrel-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-300-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-300.0-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval2-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval3-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval4-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval5-expected5]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval6-expected6]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval7-expected7]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation[pyephem-testval8-1383.636203]", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_epoch_year", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_nrel_numba", "pvlib/tests/test_irradiance.py::test_get_extra_radiation_invalid", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_simple_float", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_simple_series", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_0", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_series", "pvlib/tests/test_irradiance.py::test_grounddiffuse_albedo_invalid_surface", "pvlib/tests/test_irradiance.py::test_get_ground_diffuse_albedo_surface", "pvlib/tests/test_irradiance.py::test_isotropic_float", "pvlib/tests/test_irradiance.py::test_isotropic_series", "pvlib/tests/test_irradiance.py::test_klucher_series_float", "pvlib/tests/test_irradiance.py::test_klucher_series", "pvlib/tests/test_irradiance.py::test_haydavies", "pvlib/tests/test_irradiance.py::test_reindl", "pvlib/tests/test_irradiance.py::test_king", "pvlib/tests/test_irradiance.py::test_perez", "pvlib/tests/test_irradiance.py::test_perez_components", "pvlib/tests/test_irradiance.py::test_perez_negative_horizon", "pvlib/tests/test_irradiance.py::test_perez_arrays", "pvlib/tests/test_irradiance.py::test_perez_scalar", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[isotropic]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[klucher]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[haydavies]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[reindl]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[king]", "pvlib/tests/test_irradiance.py::test_sky_diffuse_zenith_close_to_90[perez]", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_model_invalid", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_missing_dni_extra", "pvlib/tests/test_irradiance.py::test_get_sky_diffuse_missing_airmass", "pvlib/tests/test_irradiance.py::test_campbell_norman", "pvlib/tests/test_irradiance.py::test_get_total_irradiance", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_albedo", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[isotropic]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[klucher]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[haydavies]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[reindl]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[king]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_scalars[perez]", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_missing_dni_extra", "pvlib/tests/test_irradiance.py::test_get_total_irradiance_missing_airmass", "pvlib/tests/test_irradiance.py::test_poa_components", "pvlib/tests/test_irradiance.py::test_disc_value[93193-expected0]", "pvlib/tests/test_irradiance.py::test_disc_value[None-expected1]", "pvlib/tests/test_irradiance.py::test_disc_value[101325-expected2]", "pvlib/tests/test_irradiance.py::test_disc_overirradiance", "pvlib/tests/test_irradiance.py::test_disc_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_dirint_value", "pvlib/tests/test_irradiance.py::test_dirint_nans", "pvlib/tests/test_irradiance.py::test_dirint_tdew", "pvlib/tests/test_irradiance.py::test_dirint_no_delta_kt", "pvlib/tests/test_irradiance.py::test_dirint_coeffs", "pvlib/tests/test_irradiance.py::test_dirint_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_gti_dirint", "pvlib/tests/test_irradiance.py::test_erbs", "pvlib/tests/test_irradiance.py::test_erbs_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_erbs_all_scalar", "pvlib/tests/test_irradiance.py::test_dirindex", "pvlib/tests/test_irradiance.py::test_dirindex_min_cos_zenith_max_zenith", "pvlib/tests/test_irradiance.py::test_dni", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[0-0-0-0-0-1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[30-180-30-180-0-1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[30-180-150-0-180--1]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[90-0-30-60-75.5224878-0.25]", "pvlib/tests/test_irradiance.py::test_aoi_and_aoi_projection[90-0-30-170-119.4987042--0.4924038]", "pvlib/tests/test_irradiance.py::test_aoi_projection_precision", "pvlib/tests/test_irradiance.py::test_kt_kt_prime_factor", "pvlib/tests/test_irradiance.py::test_clearsky_index", "pvlib/tests/test_irradiance.py::test_clearness_index", "pvlib/tests/test_irradiance.py::test_clearness_index_zenith_independent", "pvlib/tests/test_modelchain.py::test_ModelChain_creation", "pvlib/tests/test_modelchain.py::test_with_sapm", "pvlib/tests/test_modelchain.py::test_with_pvwatts", "pvlib/tests/test_modelchain.py::test_run_model_with_irradiance", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_irradiance_arrays_no_loss_input_type[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_invalid_inverter_params_arrays[adr]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_multi_weather[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_albedo_in_weather[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_albedo_in_weather[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_no_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_arrays_one_missing_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_weather_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_ModelChain_times_error_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_times_arrays", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dhi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[ghi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dni]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[tuple-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-sandia]", "pvlib/tests/test_modelchain.py::test_run_model_arrays_weather[list-pvwatts]", "pvlib/tests/test_modelchain.py::test_run_model_perez", "pvlib/tests/test_modelchain.py::test_run_model_gueymard_perez", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_sapm_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_pvsyst_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_faiman_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_fuentes_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_noct_sam_temp", "pvlib/tests/test_modelchain.py::test_run_model_tracker", "pvlib/tests/test_modelchain.py::test_run_model_tracker_list", "pvlib/tests/test_modelchain.py::test__assign_total_irrad", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_multi_data[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_wrong_number_arrays[list]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_different_indices", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa_arrays_missing_column", "pvlib/tests/test_modelchain.py::test__prepare_temperature", "pvlib/tests/test_modelchain.py::test__prepare_temperature_len1_weather_tuple", "pvlib/tests/test_modelchain.py::test__prepare_temperature_arrays_weather", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params0-sapm_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params1-pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params2-faiman_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params3-fuentes_temp]", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params4-noct_sam_temp]", "pvlib/tests/test_modelchain.py::test_run_model_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_arrays_solar_position_weather", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_tracking", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_multi_array[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[<lambda>]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_no_poa_global[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_poa_global_differs", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays_error[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_minimal_input", "pvlib/tests/test_modelchain.py::test_run_model_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_singleton_weather_single_array", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance_weather_single_array", "pvlib/tests/test_modelchain.py::test_infer_dc_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[desoto]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[singlediode]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvwatts_dc]", "pvlib/tests/test_modelchain.py::test_infer_dc_model_incomplete", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[cec]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[desoto]", "pvlib/tests/test_modelchain.py::test_singlediode_dc_arrays[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec_native]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[sapm_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[faiman_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[fuentes_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[noct_sam_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model_invalid", "pvlib/tests/test_modelchain.py::test_temperature_model_inconsistent", "pvlib/tests/test_modelchain.py::test_dc_model_user_func", "pvlib/tests/test_modelchain.py::test_pvwatts_dc_multiple_strings", "pvlib/tests/test_modelchain.py::test_ac_models[sandia]", "pvlib/tests/test_modelchain.py::test_ac_models[adr]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts]", "pvlib/tests/test_modelchain.py::test_ac_models[sandia_multi]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts_multi]", "pvlib/tests/test_modelchain.py::test_ac_model_user_func", "pvlib/tests/test_modelchain.py::test_ac_model_not_a_model", "pvlib/tests/test_modelchain.py::test_infer_ac_model_invalid_params", "pvlib/tests/test_modelchain.py::test_aoi_models[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models_singleon_weather_single_array[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_model_no_loss", "pvlib/tests/test_modelchain.py::test_aoi_model_user_func", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[ashrae]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[physical]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model_invalid", "pvlib/tests/test_modelchain.py::test_spectral_models[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models_singleton_weather_single_array[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_dc_ohmic_model_ohms_from_percent", "pvlib/tests/test_modelchain.py::test_dc_ohmic_model_no_dc_ohmic_loss", "pvlib/tests/test_modelchain.py::test_dc_ohmic_ext_def", "pvlib/tests/test_modelchain.py::test_dc_ohmic_not_a_model", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts_arrays", "pvlib/tests/test_modelchain.py::test_losses_models_ext_def", "pvlib/tests/test_modelchain.py::test_losses_models_no_loss", "pvlib/tests/test_modelchain.py::test_invalid_dc_model_params", "pvlib/tests/test_modelchain.py::test_invalid_models[dc_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[ac_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[aoi_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[spectral_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[temperature_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[losses_model]", "pvlib/tests/test_modelchain.py::test_bad_get_orientation", "pvlib/tests/test_modelchain.py::test_with_sapm_pvsystem_arrays", "pvlib/tests/test_modelchain.py::test_ModelChain_no_extra_kwargs", "pvlib/tests/test_modelchain.py::test_ModelChain_attributes_deprecated_10", "pvlib/tests/test_modelchain.py::test_basic_chain_alt_az", "pvlib/tests/test_modelchain.py::test_basic_chain_altitude_pressure", "pvlib/tests/test_modelchain.py::test_complete_irradiance_clean_run", "pvlib/tests/test_modelchain.py::test_complete_irradiance", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays[list]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[tuple]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_arrays_wrong_length[list]", "pvlib/tests/test_modelchain.py::test_unknown_attribute", "pvlib/tests/test_modelchain.py::test_inconsistent_array_params", "pvlib/tests/test_modelchain.py::test_modelchain__common_keys", "pvlib/tests/test_modelchain.py::test__irrad_for_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[ashrae-model_params0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[physical-model_params1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam[martin_ruiz-model_params2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_iam", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_interp", "pvlib/tests/test_pvsystem.py::test__normalize_sam_product_names", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_iam_invalid", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_raise_no_parameters", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_cecmod", "pvlib/tests/test_pvsystem.py::test_retrieve_sam_cecinverter", "pvlib/tests/test_pvsystem.py::test_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[1.5-1.00028714375]", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[airmass1-expected1]", "pvlib/tests/test_pvsystem.py::test_sapm_spectral_loss[airmass2-expected2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_spectral_loss", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_spectral_loss", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters0-multisi-None]", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters1-multisi-None]", "pvlib/tests/test_pvsystem.py::test_PVSystem_first_solar_spectral_loss[module_parameters2-None-coefficients2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_first_solar_spectral_loss", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input0-1140.0510967821876]", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input1-expected1]", "pvlib/tests/test_pvsystem.py::test_sapm_effective_irradiance[test_input2-expected2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_effective_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[20-poa_diffuse0-aoi0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[poa_direct1-poa_diffuse1-aoi1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_effective_irradiance_value_error[poa_direct2-poa_diffuse2-20]", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_celltemp_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_sapm_celltemp_different_arrays", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvsyst_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_faiman_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp_error", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_cell_temperature_invalid", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[faiman]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[sapm]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[fuentes]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[noct_sam]", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_module_height", "pvlib/tests/test_pvsystem.py::test_Array__infer_temperature_model_params", "pvlib/tests/test_pvsystem.py::test_Array__infer_cell_type", "pvlib/tests/test_pvsystem.py::test_calcparams_desoto", "pvlib/tests/test_pvsystem.py::test_calcparams_cec", "pvlib/tests/test_pvsystem.py::test_calcparams_cec_extra_params_propagation", "pvlib/tests/test_pvsystem.py::test_calcparams_pvsyst", "pvlib/tests/test_pvsystem.py::test_PVSystem_calcparams_desoto", "pvlib/tests/test_pvsystem.py::test_PVSystem_calcparams_pvsyst", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_pvsyst]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_desoto]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams[calcparams_cec]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_desoto-1-celltemp0]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_desoto-irrad1-1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_cec-1-celltemp2]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_cec-irrad3-1]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_pvsyst-1-celltemp4]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_calcparams_value_error[calcparams_pvsyst-irrad5-1]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i0-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i1-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i2-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i3-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i4-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i5-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i6-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i7-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i8-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i9-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_v_from_i[fixture_v_from_i10-newton-1e-08]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i0]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i1]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i2]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i3]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i4]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i5]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i6]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i7]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i8]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i9]", "pvlib/tests/test_pvsystem.py::test_i_from_v_from_i[fixture_v_from_i10]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v0-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v1-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v2-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v3-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v4-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v5-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-lambertw-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-brentq-1e-11]", "pvlib/tests/test_pvsystem.py::test_i_from_v[fixture_i_from_v6-newton-1e-11]", "pvlib/tests/test_pvsystem.py::test_PVSystem_i_from_v", "pvlib/tests/test_pvsystem.py::test_i_from_v_size", "pvlib/tests/test_pvsystem.py::test_v_from_i_size", "pvlib/tests/test_pvsystem.py::test_mpp_floats", "pvlib/tests/test_pvsystem.py::test_mpp_array", "pvlib/tests/test_pvsystem.py::test_mpp_series", "pvlib/tests/test_pvsystem.py::test_singlediode_series", "pvlib/tests/test_pvsystem.py::test_singlediode_array", "pvlib/tests/test_pvsystem.py::test_singlediode_floats", "pvlib/tests/test_pvsystem.py::test_singlediode_floats_ivcurve", "pvlib/tests/test_pvsystem.py::test_singlediode_series_ivcurve", "pvlib/tests/test_pvsystem.py::test_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_scale_voltage_current_power", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_sandia", "pvlib/tests/test_pvsystem.py::test_PVSystem_snlinverter", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_sandia_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_pvwatts_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[sandia]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[adr]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_single_array_tuple_input[pvwatts]", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_adr", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_adr_multi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_ac_invalid", "pvlib/tests/test_pvsystem.py::test_PVSystem_creation", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_creation", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_aoi", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_get_aoi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance_model", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_get_irradiance_multi_irrad", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module_parameters]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[module_type]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[temperature_model_parameters]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[albedo]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[surface_tilt]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[surface_azimuth]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[racking_model]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[modules_per_string]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_attributes[strings_per_inverter]", "pvlib/tests/test_pvsystem.py::test_PVSystem___repr__", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array___repr__", "pvlib/tests/test_pvsystem.py::test_Array___repr__", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_scalars", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_arrays", "pvlib/tests/test_pvsystem.py::test_pvwatts_dc_series", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_default", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_arrays", "pvlib/tests/test_pvsystem.py::test_pvwatts_losses_series", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_dc", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_dc_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_pvwatts_dc", "pvlib/tests/test_pvsystem.py::test_PVSystem_multiple_array_pvwatts_dc_value_error", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_losses", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_ac", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvwatts_ac_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_num_arrays", "pvlib/tests/test_pvsystem.py::test_PVSystem_at_least_one_array", "pvlib/tests/test_pvsystem.py::test_combine_loss_factors", "pvlib/tests/test_pvsystem.py::test_no_extra_kwargs", "pvlib/tests/test_pvsystem.py::test_AbstractMount_constructor", "pvlib/tests/test_pvsystem.py::test_FixedMount_constructor", "pvlib/tests/test_pvsystem.py::test_FixedMount_get_orientation", "pvlib/tests/test_pvsystem.py::test_SingleAxisTrackerMount_constructor", "pvlib/tests/test_pvsystem.py::test_SingleAxisTrackerMount_get_orientation", "pvlib/tests/test_pvsystem.py::test_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_PVSystem_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_dc_ohmic_losses", "pvlib/tests/test_pvsystem.py::test_Array_dc_ohms_from_percent", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_temperature_deprecated[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[sapm-keys0]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[fuentes-keys1]", "pvlib/tests/test_pvsystem.py::test_Array_temperature_missing_parameters[noct_sam-keys2]", "pvlib/tests/test_tracking.py::test_solar_noon", "pvlib/tests/test_tracking.py::test_scalars", "pvlib/tests/test_tracking.py::test_arrays", "pvlib/tests/test_tracking.py::test_nans", "pvlib/tests/test_tracking.py::test_arrays_multi", "pvlib/tests/test_tracking.py::test_azimuth_north_south", "pvlib/tests/test_tracking.py::test_max_angle", "pvlib/tests/test_tracking.py::test_backtrack", "pvlib/tests/test_tracking.py::test_axis_tilt", "pvlib/tests/test_tracking.py::test_axis_azimuth", "pvlib/tests/test_tracking.py::test_horizon_flat", "pvlib/tests/test_tracking.py::test_horizon_tilted", "pvlib/tests/test_tracking.py::test_low_sun_angles", "pvlib/tests/test_tracking.py::test_SingleAxisTracker_tracking", "pvlib/tests/test_tracking.py::test_get_aoi", "pvlib/tests/test_tracking.py::test_SingleAxisTracker___repr__", "pvlib/tests/test_tracking.py::test_calc_axis_tilt", "pvlib/tests/test_tracking.py::test_slope_aware_backtracking", "pvlib/tests/test_tracking.py::test_singleaxis_aoi_gh1221"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-800", "base_commit": "2f3586b6f67383b1ec0c24c4772e65119c3f5261", "patch": "diff --git a/pydicom/uid.py b/pydicom/uid.py\n--- a/pydicom/uid.py\n+++ b/pydicom/uid.py\n@@ -250,19 +250,19 @@ def generate_uid(prefix=PYDICOM_ROOT_UID, entropy_srcs=None):\n     ----------\n     prefix : str or None\n         The UID prefix to use when creating the UID. Default is the pydicom\n-        root UID '1.2.826.0.1.3680043.8.498.'. If None then a value of '2.25.'\n-        will be used (as described on `David Clunie's website\n-        <http://www.dclunie.com/medical-image-faq/html/part2.html#UID>`_).\n+        root UID '1.2.826.0.1.3680043.8.498.'. If None then a prefix of '2.25.'\n+        will be used with the integer form of a UUID generated using the\n+        UUID4 algorithm.\n     entropy_srcs : list of str or None\n-        If a list of str, the prefix will be appended with a SHA512 hash of the\n-        list which means the result is deterministic and should make the\n-        original data unrecoverable. If None random data will be used\n-        (default).\n+        If `prefix` is not None, then the prefix will be appended with a\n+        SHA512 hash of the list which means the result is deterministic and\n+        should make the original data unrecoverable. If None random data will\n+        be used (default).\n \n     Returns\n     -------\n     pydicom.uid.UID\n-        A 64 character DICOM UID.\n+        A DICOM UID of up to 64 characters.\n \n     Raises\n     ------\n@@ -275,17 +275,17 @@ def generate_uid(prefix=PYDICOM_ROOT_UID, entropy_srcs=None):\n     >>> generate_uid()\n     1.2.826.0.1.3680043.8.498.22463838056059845879389038257786771680\n     >>> generate_uid(prefix=None)\n-    2.25.12586835699909622925962004639368649121731805922235633382942\n+    2.25.167161297070865690102504091919570542144\n     >>> generate_uid(entropy_srcs=['lorem', 'ipsum'])\n     1.2.826.0.1.3680043.8.498.87507166259346337659265156363895084463\n     >>> generate_uid(entropy_srcs=['lorem', 'ipsum'])\n     1.2.826.0.1.3680043.8.498.87507166259346337659265156363895084463\n     \"\"\"\n-    max_uid_len = 64\n-\n     if prefix is None:\n-        prefix = '2.25.'\n+        # UUID -> as 128-bit int -> max 39 characters long\n+        return UID('2.25.{}'.format(uuid.uuid4().int))\n \n+    max_uid_len = 64\n     if len(prefix) > max_uid_len - 1:\n         raise ValueError(\"The prefix must be less than 63 chars\")\n     if not re.match(RE_VALID_UID_PREFIX, prefix):\n", "test_patch": "diff --git a/pydicom/tests/test_uid.py b/pydicom/tests/test_uid.py\n--- a/pydicom/tests/test_uid.py\n+++ b/pydicom/tests/test_uid.py\n@@ -1,6 +1,8 @@\n # Copyright 2008-2018 pydicom authors. See LICENSE file for details.\n \"\"\"Test suite for uid.py\"\"\"\n \n+import uuid\n+\n import pytest\n \n from pydicom.uid import UID, generate_uid, PYDICOM_ROOT_UID, JPEGLSLossy\n@@ -57,6 +59,24 @@ def test_entropy_src_custom(self):\n         assert uid == rf\n         assert len(uid) == 64\n \n+    def test_none(self):\n+        \"\"\"Test generate_uid(None).\"\"\"\n+        uid = generate_uid(prefix=None)\n+        # Check prefix\n+        assert '2.25.' == uid[:5]\n+        # Check UUID suffix\n+        as_uuid = uuid.UUID(int=int(uid[5:]))\n+        assert isinstance(as_uuid, uuid.UUID)\n+        assert as_uuid.version == 4\n+        assert as_uuid.variant == uuid.RFC_4122\n+\n+    def test_none_iterate(self):\n+        \"\"\"Test generate_uid(None) generates valid UIDs.\"\"\"\n+        # Generate random UIDs, if a bad method then should eventually fail\n+        for ii in range(100000):\n+            uid = generate_uid(None)\n+            assert uid.is_valid\n+\n \n class TestUID(object):\n     \"\"\"Test DICOM UIDs\"\"\"\n", "problem_statement": "The function generate_uid() generates non-conforming \u201c2.25 .\u201d DICOM UIDs\n<!-- Instructions For Filing a Bug: https://github.com/pydicom/pydicom/blob/master/CONTRIBUTING.md#filing-bugs -->\r\n\r\n#### Description\r\nIt seems there was already a discussion about this function in the past (#125), but the current implementation generates non-conforming DICOM UIDs when called with prefix \u2018none\u2019 to trigger that the function generate_uid() should generate a UUID derived UID.\r\n\r\nThe DICOM Standard requires (see DICOM PS 3.5, B.2 that when a UUID derived UID is constructed it should be in the format \u201c2.25.\u201d + uuid(in its decimal representation string representation)\r\nFor example a UUID of f81d4fae-7dec-11d0-a765-00a0c91e6bf6 should become 2.25.329800735698586629295641978511506172918\r\n\r\nThe current implementation extends the uuid part to the remaining 59 characters. By not following the DICOM formatting rule, receiving systems that are processing DICOM instances created with this library are not capable of converting the generated \u201c2.25\u201d UID back to a UUID. Due to the extra sha512 operation on the UUID, the variant and version info of the UUID are also lost.\r\n\r\n#### Steps/Code to Reproduce\r\n- call generate_uid() to generate a \"2.25.\" DICOM UID\r\n\r\n#### Expected Results\r\nA conforming unique DICOM UID is returned.\r\n\r\n#### Actual Results\r\nNon conforming UID is returned.\n", "hints_text": "Thanks for pointing this out - I wasn't aware of that section of the standard.  Do you know of a solution that respects the privacy issues pointed out in #125? \r\n\r\n> By not following the DICOM formatting rule, receiving systems that are processing DICOM instances created with this library are not capable of converting the generated \u201c2.25\u201d UID back to a UUID\r\n\r\nWhy would it be necessary for the receiving software to generate a \"real UUID\" from this (and therefore the variant and the version)?  Just curious, as if we want to be standard compliant it doesn't really matter.\r\n\r\n\r\n\nI think the solution proposed by @cancan101 in #125 to use the uuid v4 algorithm is fine. The uuid v1 algorithm leaks the MAC address. The open source dcm4che (Java) implementation for example also uses the uuid v4 algorithm, the two C# implementation I known of use the .NET Guid.NewGuid() method, which will also returns v4 uuids. \r\n\r\nDICOM is all about interoperability, there may be receiving systems that (implicitly) depend on it. A Level 2 (Full) C-STORE SCP may, (but is not required) validate the attributes of an incoming SOP instance. Personally, I have never encountered a DICOM system that had trouble with it, most systems just threat a UID as 64 bytes and are happy with it as long as it is unique.\r\n\r\nPossible (performance) scenario\r\nLarge DICOM archives need to maintain a relational database to maintain which images are stored in the system. To ensure data integrity, these systems put often a constraint on the uniqueness of the SOP Instance UID column. Most DBMS systems will create a non-cluster index to ensure that this constraint can be met. A uuid is only 16 bytes, compared to a UID that is 64 bytes, which can make the difference of keeping the index in memory or not. Some DBMS systems have a native data type to support uuid columns. The complication is of course that these systems also need to support images with \u201cOrganizationally Derived\u201d UIDs and this optimization only makes sense if a majority of UIDs are uuid derived UIDs.", "created_at": "2019-02-01T07:12:27Z", "version": "1.2", "FAIL_TO_PASS": ["pydicom/tests/test_uid.py::TestGenerateUID::test_none"], "PASS_TO_PASS": ["pydicom/tests/test_uid.py::TestGenerateUID::test_generate_uid", "pydicom/tests/test_uid.py::TestGenerateUID::test_entropy_src", "pydicom/tests/test_uid.py::TestGenerateUID::test_entropy_src_custom", "pydicom/tests/test_uid.py::TestGenerateUID::test_none_iterate", "pydicom/tests/test_uid.py::TestUID::test_equality", "pydicom/tests/test_uid.py::TestUID::test_inequality", "pydicom/tests/test_uid.py::TestUID::test_hash", "pydicom/tests/test_uid.py::TestUID::test_str", "pydicom/tests/test_uid.py::TestUID::test_is_implicit_vr", "pydicom/tests/test_uid.py::TestUID::test_is_little_endian", "pydicom/tests/test_uid.py::TestUID::test_is_deflated", "pydicom/tests/test_uid.py::TestUID::test_is_transfer_syntax", "pydicom/tests/test_uid.py::TestUID::test_is_compressed", "pydicom/tests/test_uid.py::TestUID::test_is_encapsulated", "pydicom/tests/test_uid.py::TestUID::test_name", "pydicom/tests/test_uid.py::TestUID::test_name_with_equal_hash", "pydicom/tests/test_uid.py::TestUID::test_type", "pydicom/tests/test_uid.py::TestUID::test_info", "pydicom/tests/test_uid.py::TestUID::test_is_retired", "pydicom/tests/test_uid.py::TestUID::test_is_valid", "pydicom/tests/test_uid.py::TestUID::test_is_private", "pydicom/tests/test_uid.py::TestUID::test_raises", "pydicom/tests/test_uid.py::TestUID::test_transitive", "pydicom/tests/test_uid.py::TestUIDPrivate::test_equality", "pydicom/tests/test_uid.py::TestUIDPrivate::test_inequality", "pydicom/tests/test_uid.py::TestUIDPrivate::test_hash", "pydicom/tests/test_uid.py::TestUIDPrivate::test_str", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_implicit_vr", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_little_endian", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_deflated", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_transfer_syntax", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_compressed", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_encapsulated", "pydicom/tests/test_uid.py::TestUIDPrivate::test_name", "pydicom/tests/test_uid.py::TestUIDPrivate::test_type", "pydicom/tests/test_uid.py::TestUIDPrivate::test_info", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_retired", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_valid", "pydicom/tests/test_uid.py::TestUIDPrivate::test_is_private"], "environment_setup_commit": "b4b44acbf1ddcaf03df16210aac46cb3a8acd6b9"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1050", "base_commit": "00c248441ffb8b7d46c6d855b723e696a8f5aada", "patch": "diff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -741,9 +741,13 @@ def DataElement_from_raw(raw_data_element, encoding=None):\n     except NotImplementedError as e:\n         raise NotImplementedError(\"{0:s} in tag {1!r}\".format(str(e), raw.tag))\n \n-    if raw.tag in _LUT_DESCRIPTOR_TAGS and value[0] < 0:\n+    if raw.tag in _LUT_DESCRIPTOR_TAGS and value:\n         # We only fix the first value as the third value is 8 or 16\n-        value[0] += 65536\n+        try:\n+            if value[0] < 0:\n+                value[0] += 65536\n+        except TypeError:\n+            pass\n \n     return DataElement(raw.tag, VR, value, raw.value_tell,\n                        raw.length == 0xFFFFFFFF, already_converted=True)\n", "test_patch": "diff --git a/pydicom/tests/test_filereader.py b/pydicom/tests/test_filereader.py\n--- a/pydicom/tests/test_filereader.py\n+++ b/pydicom/tests/test_filereader.py\n@@ -674,6 +674,27 @@ def test_lut_descriptor(self):\n             assert elem.VR == 'SS'\n             assert elem.value == [62720, -2048, 16]\n \n+    def test_lut_descriptor_empty(self):\n+        \"\"\"Regression test for #1049: LUT empty raises.\"\"\"\n+        bs = DicomBytesIO(b'\\x28\\x00\\x01\\x11\\x53\\x53\\x00\\x00')\n+        bs.is_little_endian = True\n+        bs.is_implicit_VR = False\n+        ds = dcmread(bs, force=True)\n+        elem = ds[0x00281101]\n+        assert elem.value is None\n+        assert elem.VR == 'SS'\n+\n+    def test_lut_descriptor_singleton(self):\n+        \"\"\"Test LUT Descriptor with VM = 1\"\"\"\n+        bs = DicomBytesIO(b'\\x28\\x00\\x01\\x11\\x53\\x53\\x02\\x00\\x00\\xf5')\n+        bs.is_little_endian = True\n+        bs.is_implicit_VR = False\n+        ds = dcmread(bs, force=True)\n+        elem = ds[0x00281101]\n+        # No conversion to US if not a triplet\n+        assert elem.value == -2816\n+        assert elem.VR == 'SS'\n+\n \n class TestIncorrectVR(object):\n     def setup(self):\n", "problem_statement": "LUT Descriptor tag with no value yields TypeError\n**Describe the bug**\r\nI have a DICOM image with the following tag (copied from ImageJ)\r\n\r\n```\r\n0028,1101  Red Palette Color Lookup Table Descriptor: \r\n```\r\n\r\nwhich corresponds to the raw data element, produced by [`DataElement_from_raw`](https://github.com/pydicom/pydicom/blob/v1.4.1/pydicom/dataelem.py#L699):\r\n```\r\nRawDataElement(tag=(0028, 1101), VR='US', length=0, value=None, value_tell=1850, is_implicit_VR=False, is_little_endian=True)\r\n```\r\n\r\nBecause this tag is matched by the [LUT Descriptor tags](https://github.com/pydicom/pydicom/blob/v1.4.1/pydicom/dataelem.py#L696) and the value is empty (`None`), the [following line](https://github.com/pydicom/pydicom/blob/v1.4.1/pydicom/dataelem.py#L761):\r\n```\r\nif raw.tag in _LUT_DESCRIPTOR_TAGS and value[0] < 0:\r\n```\r\nresults in \r\n```\r\nTypeError: 'NoneType' object is not subscriptable\r\n```\r\n\r\n**Expected behavior**\r\n\r\nGiven that I discovered this by parsing what seems to be a set of faulty DICOMs (mangled pixel data), I'm not sure if an error should be raised if the colour attribute value is not provided.\r\n\r\nHowever, given that `value` can be `None` for other tags, the simple fix is\r\n\r\n```python\r\ntry:\r\n    if raw.tag in _LUT_DESCRIPTOR_TAGS and value[0] < 0:\r\n        # We only fix the first value as the third value is 8 or 16\r\n        value[0] += 65536\r\nexcept TypeError:\r\n    pass\r\n```\r\n\r\n(or test if `value` is iterable).\r\n\r\n**Your environment**\r\n```\r\nDarwin-19.3.0-x86_64-i386-64bit\r\nPython  3.7.6 | packaged by conda-forge | (default, Jan  7 2020, 22:05:27)\r\n[Clang 9.0.1 ]\r\npydicom  1.4.1\r\n```\r\n\r\nMany thanks!\n", "hints_text": "", "created_at": "2020-02-10T10:54:28Z", "version": "1.4", "FAIL_TO_PASS": ["pydicom/tests/test_filereader.py::TestReader::test_lut_descriptor_empty", "pydicom/tests/test_filereader.py::TestReader::test_lut_descriptor_singleton"], "PASS_TO_PASS": ["pydicom/tests/test_filereader.py::TestReader::test_empty_numbers_tag", "pydicom/tests/test_filereader.py::TestReader::test_UTF8_filename", "pydicom/tests/test_filereader.py::TestReader::test_RTPlan", "pydicom/tests/test_filereader.py::TestReader::test_RTDose", "pydicom/tests/test_filereader.py::TestReader::test_CT", "pydicom/tests/test_filereader.py::TestReader::test_CT_PixelData", "pydicom/tests/test_filereader.py::TestReader::test_no_force", "pydicom/tests/test_filereader.py::TestReader::test_RTStruct", "pydicom/tests/test_filereader.py::TestReader::test_dir", "pydicom/tests/test_filereader.py::TestReader::test_MR", "pydicom/tests/test_filereader.py::TestReader::test_deflate", "pydicom/tests/test_filereader.py::TestReader::test_no_pixels_read", "pydicom/tests/test_filereader.py::TestReader::test_specific_tags", "pydicom/tests/test_filereader.py::TestReader::test_specific_tags_with_unknown_length_SQ", "pydicom/tests/test_filereader.py::TestReader::test_specific_tags_with_unknown_length_tag", "pydicom/tests/test_filereader.py::TestReader::test_tag_with_unknown_length_tag_too_short", "pydicom/tests/test_filereader.py::TestReader::test_private_SQ", "pydicom/tests/test_filereader.py::TestReader::test_nested_private_SQ", "pydicom/tests/test_filereader.py::TestReader::test_no_meta_group_length", "pydicom/tests/test_filereader.py::TestReader::test_no_transfer_syntax_in_meta", "pydicom/tests/test_filereader.py::TestReader::test_explicit_VR_little_endian_no_meta", "pydicom/tests/test_filereader.py::TestReader::test_explicit_VR_big_endian_no_meta", "pydicom/tests/test_filereader.py::TestReader::test_planar_config", "pydicom/tests/test_filereader.py::TestReader::test_correct_ambiguous_vr", "pydicom/tests/test_filereader.py::TestReader::test_correct_ambiguous_explicit_vr", "pydicom/tests/test_filereader.py::TestReader::test_correct_ambiguous_vr_compressed", "pydicom/tests/test_filereader.py::TestReader::test_long_specific_char_set", "pydicom/tests/test_filereader.py::TestReader::test_no_preamble_file_meta_dataset", "pydicom/tests/test_filereader.py::TestReader::test_no_preamble_command_group_dataset", "pydicom/tests/test_filereader.py::TestReader::test_group_length_wrong", "pydicom/tests/test_filereader.py::TestReader::test_preamble_command_meta_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_preamble_meta_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_preamble_commandset_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_meta_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_commandset_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_file_meta_dataset_implicit_vr", "pydicom/tests/test_filereader.py::TestReader::test_no_dataset", "pydicom/tests/test_filereader.py::TestReader::test_empty_file", "pydicom/tests/test_filereader.py::TestReader::test_empty_specific_character_set", "pydicom/tests/test_filereader.py::TestReader::test_dcmread_does_not_raise", "pydicom/tests/test_filereader.py::TestReader::test_lut_descriptor", "pydicom/tests/test_filereader.py::TestIncorrectVR::test_implicit_vr_expected_explicit_used", "pydicom/tests/test_filereader.py::TestIncorrectVR::test_implicit_vr_expected_explicit_used_strict", "pydicom/tests/test_filereader.py::TestIncorrectVR::test_explicit_vr_expected_implicit_used", "pydicom/tests/test_filereader.py::TestIncorrectVR::test_explicit_vr_expected_implicit_used_strict", "pydicom/tests/test_filereader.py::TestIncorrectVR::test_seq_item_looks_like_explicit_VR", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[\\x00A-0x00", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[@A-0x40", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[[A-0x5b", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[`A-0x60", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[{A-0x7b", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[\\xffA-0xff", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[A\\x00-0x41", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[Z\\x00-0x5a", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[a\\x00-0x61", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[z\\x00-0x7a", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[\\x00Z-0x00", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[\\x00a-0x00", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[\\x00z-0x00", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[AA-AA]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[AZ-AZ]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[ZA-ZA]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[ZZ-ZZ]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[Aa-Aa]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[Az-Az]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[aA-aA]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[aZ-aZ]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[aa-aa]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[az-az]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[Za-Za]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[Zz-Zz]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[zA-zA]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[zZ-zZ]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[za-za]", "pydicom/tests/test_filereader.py::TestUnknownVR::test_fail_decode_msg[zz-zz]", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OD_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OD_explicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OL_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OL_explicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UC_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UC_explicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UR_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UR_explicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_AE", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OV_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_OV_explicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_SV_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UV_implicit_little", "pydicom/tests/test_filereader.py::TestReadDataElement::test_read_UV_explicit_little", "pydicom/tests/test_filereader.py::TestDeferredRead::test_time_check", "pydicom/tests/test_filereader.py::TestDeferredRead::test_file_exists", "pydicom/tests/test_filereader.py::TestDeferredRead::test_values_identical", "pydicom/tests/test_filereader.py::TestDeferredRead::test_zipped_deferred", "pydicom/tests/test_filereader.py::TestDeferredRead::test_filelike_deferred", "pydicom/tests/test_filereader.py::TestReadTruncatedFile::testReadFileWithMissingPixelData", "pydicom/tests/test_filereader.py::TestReadTruncatedFile::testReadFileWithMissingPixelDataArray", "pydicom/tests/test_filereader.py::TestFileLike::test_read_file_given_file_object", "pydicom/tests/test_filereader.py::TestFileLike::test_read_file_given_file_like_object", "pydicom/tests/test_filereader.py::TestDataElementGenerator::test_little_endian_explicit", "pydicom/tests/test_filereader.py::TestDataElementGenerator::test_little_endian_implicit", "pydicom/tests/test_filereader.py::TestDataElementGenerator::test_big_endian_explicit"], "environment_setup_commit": "5098c9147fadcb3e5918487036867931435adeb8"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1093", "base_commit": "11c356f9a89fc88b4d3ff368ce1aae170a97ebd7", "patch": "diff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -1133,7 +1133,7 @@ def prepare_inputs(self, weather):\n         ModelChain.complete_irradiance\n         \"\"\"\n \n-        self._verify_df(weather, required=['ghi', 'dni', 'ghi'])\n+        self._verify_df(weather, required=['ghi', 'dni', 'dhi'])\n         self._assign_weather(weather)\n \n         self.times = self.weather.index\n", "test_patch": "diff --git a/pvlib/tests/test_modelchain.py b/pvlib/tests/test_modelchain.py\n--- a/pvlib/tests/test_modelchain.py\n+++ b/pvlib/tests/test_modelchain.py\n@@ -249,6 +249,16 @@ def test_prepare_inputs_no_irradiance(sapm_dc_snl_ac_system, location):\n         mc.prepare_inputs(weather)\n \n \n+@pytest.mark.parametrize(\"missing\", ['dhi', 'ghi', 'dni'])\n+def test_prepare_inputs_missing_irrad_component(\n+        sapm_dc_snl_ac_system, location, missing):\n+    mc = ModelChain(sapm_dc_snl_ac_system, location)\n+    weather = pd.DataFrame({'dhi': [1, 2], 'dni': [1, 2], 'ghi': [1, 2]})\n+    weather.drop(columns=missing, inplace=True)\n+    with pytest.raises(ValueError):\n+        mc.prepare_inputs(weather)\n+\n+\n def test_run_model_perez(sapm_dc_snl_ac_system, location):\n     mc = ModelChain(sapm_dc_snl_ac_system, location,\n                     transposition_model='perez')\n", "problem_statement": "ModelChain.prepare_inputs can succeed with missing dhi\nFrom the docstring for `ModelChain.prepare_inputs()` I believe the method should fail if `weather` does not have a `dhi` column.\r\n\r\nThe validation checks for `'ghi'` twice, but not `'dhi`'\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/11c356f9a89fc88b4d3ff368ce1aae170a97ebd7/pvlib/modelchain.py#L1136\n", "hints_text": "", "created_at": "2020-11-20T22:36:43Z", "version": "0.7", "FAIL_TO_PASS": ["pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dhi]"], "PASS_TO_PASS": ["pvlib/tests/test_modelchain.py::test_ModelChain_creation", "pvlib/tests/test_modelchain.py::test_with_sapm", "pvlib/tests/test_modelchain.py::test_with_pvwatts", "pvlib/tests/test_modelchain.py::test_orientation_strategy[None-expected0]", "pvlib/tests/test_modelchain.py::test_orientation_strategy[None-expected1]", "pvlib/tests/test_modelchain.py::test_orientation_strategy[flat-expected2]", "pvlib/tests/test_modelchain.py::test_orientation_strategy[south_at_latitude_tilt-expected3]", "pvlib/tests/test_modelchain.py::test_run_model_with_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_no_irradiance", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[ghi]", "pvlib/tests/test_modelchain.py::test_prepare_inputs_missing_irrad_component[dni]", "pvlib/tests/test_modelchain.py::test_run_model_perez", "pvlib/tests/test_modelchain.py::test_run_model_gueymard_perez", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_sapm_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_pvsyst_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_faiman_temp", "pvlib/tests/test_modelchain.py::test_run_model_with_weather_fuentes_temp", "pvlib/tests/test_modelchain.py::test_run_model_tracker", "pvlib/tests/test_modelchain.py::test__assign_total_irrad", "pvlib/tests/test_modelchain.py::test_prepare_inputs_from_poa", "pvlib/tests/test_modelchain.py::test__prepare_temperature", "pvlib/tests/test_modelchain.py::test_run_model_from_poa", "pvlib/tests/test_modelchain.py::test_run_model_from_poa_tracking", "pvlib/tests/test_modelchain.py::test_run_model_from_effective_irradiance", "pvlib/tests/test_modelchain.py::test_infer_dc_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[desoto]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvsyst]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[singlediode]", "pvlib/tests/test_modelchain.py::test_infer_dc_model[pvwatts_dc]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec]", "pvlib/tests/test_modelchain.py::test_infer_spectral_model[cec_native]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[sapm_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[faiman_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[pvsyst_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[fuentes_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model_invalid", "pvlib/tests/test_modelchain.py::test_infer_temp_model_no_params", "pvlib/tests/test_modelchain.py::test_temperature_model_inconsistent", "pvlib/tests/test_modelchain.py::test_dc_model_user_func", "pvlib/tests/test_modelchain.py::test_ac_models[sandia]", "pvlib/tests/test_modelchain.py::test_ac_models[adr]", "pvlib/tests/test_modelchain.py::test_ac_models[pvwatts]", "pvlib/tests/test_modelchain.py::test_ac_models_deprecated[snlinverter]", "pvlib/tests/test_modelchain.py::test_ac_models_deprecated[adrinverter]", "pvlib/tests/test_modelchain.py::test_ac_model_user_func", "pvlib/tests/test_modelchain.py::test_ac_model_not_a_model", "pvlib/tests/test_modelchain.py::test_aoi_models[sapm]", "pvlib/tests/test_modelchain.py::test_aoi_models[ashrae]", "pvlib/tests/test_modelchain.py::test_aoi_models[physical]", "pvlib/tests/test_modelchain.py::test_aoi_models[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_aoi_model_no_loss", "pvlib/tests/test_modelchain.py::test_aoi_model_user_func", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[sapm]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[ashrae]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[physical]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model[martin_ruiz]", "pvlib/tests/test_modelchain.py::test_infer_aoi_model_invalid", "pvlib/tests/test_modelchain.py::test_spectral_models[sapm]", "pvlib/tests/test_modelchain.py::test_spectral_models[first_solar]", "pvlib/tests/test_modelchain.py::test_spectral_models[no_loss]", "pvlib/tests/test_modelchain.py::test_spectral_models[constant_spectral_loss]", "pvlib/tests/test_modelchain.py::test_losses_models_pvwatts", "pvlib/tests/test_modelchain.py::test_losses_models_ext_def", "pvlib/tests/test_modelchain.py::test_losses_models_no_loss", "pvlib/tests/test_modelchain.py::test_invalid_dc_model_params", "pvlib/tests/test_modelchain.py::test_invalid_models[dc_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[ac_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[aoi_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[spectral_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[temperature_model]", "pvlib/tests/test_modelchain.py::test_invalid_models[losses_model]", "pvlib/tests/test_modelchain.py::test_bad_get_orientation", "pvlib/tests/test_modelchain.py::test_deprecated_09[snlinverter]", "pvlib/tests/test_modelchain.py::test_deprecated_09[adrinverter]", "pvlib/tests/test_modelchain.py::test_ModelChain_kwargs_deprecated_09", "pvlib/tests/test_modelchain.py::test_basic_chain_required", "pvlib/tests/test_modelchain.py::test_basic_chain_alt_az", "pvlib/tests/test_modelchain.py::test_basic_chain_strategy", "pvlib/tests/test_modelchain.py::test_basic_chain_altitude_pressure", "pvlib/tests/test_modelchain.py::test_ModelChain___repr__[south_at_latitude_tilt-south_at_latitude_tilt]", "pvlib/tests/test_modelchain.py::test_ModelChain___repr__[None-None]", "pvlib/tests/test_modelchain.py::test_complete_irradiance_clean_run", "pvlib/tests/test_modelchain.py::test_complete_irradiance"], "environment_setup_commit": "6e5148f59c5050e8f7a0084b7ae39e93b80f72e6"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4084", "base_commit": "181918e9c2840dc3f5ff1c713bf6b5a00d0725b5", "patch": "diff --git a/src/sqlfluff/core/linter/linter.py b/src/sqlfluff/core/linter/linter.py\n--- a/src/sqlfluff/core/linter/linter.py\n+++ b/src/sqlfluff/core/linter/linter.py\n@@ -1103,17 +1103,39 @@ def lint_path(\n         processes: Optional[int] = None,\n     ) -> LintedDir:\n         \"\"\"Lint a path.\"\"\"\n-        linted_path = LintedDir(path)\n-        if self.formatter:\n-            self.formatter.dispatch_path(path)\n-        fnames = list(\n-            self.paths_from_path(\n+        return self.lint_paths(\n+            (path,), fix, ignore_non_existent_files, ignore_files, processes\n+        ).paths[0]\n+\n+    def lint_paths(\n+        self,\n+        paths: Tuple[str, ...],\n+        fix: bool = False,\n+        ignore_non_existent_files: bool = False,\n+        ignore_files: bool = True,\n+        processes: Optional[int] = None,\n+    ) -> LintingResult:\n+        \"\"\"Lint an iterable of paths.\"\"\"\n+        # If no paths specified - assume local\n+        if not paths:  # pragma: no cover\n+            paths = (os.getcwd(),)\n+        # Set up the result to hold what we get back\n+        result = LintingResult()\n+\n+        expanded_paths: List[str] = []\n+        expanded_path_to_linted_dir = {}\n+        for path in paths:\n+            linted_dir = LintedDir(path)\n+            result.add(linted_dir)\n+            for fname in self.paths_from_path(\n                 path,\n                 ignore_non_existent_files=ignore_non_existent_files,\n                 ignore_files=ignore_files,\n-            )\n-        )\n+            ):\n+                expanded_paths.append(fname)\n+                expanded_path_to_linted_dir[fname] = linted_dir\n \n+        files_count = len(expanded_paths)\n         if processes is None:\n             processes = self.config.get(\"processes\", default=1)\n \n@@ -1131,72 +1153,29 @@ def lint_path(\n             self.formatter.dispatch_processing_header(effective_processes)\n \n         # Show files progress bar only when there is more than one.\n-        files_count = len(fnames)\n+        first_path = expanded_paths[0] if expanded_paths else \"\"\n         progress_bar_files = tqdm(\n             total=files_count,\n-            desc=f\"file {os.path.basename(fnames[0] if fnames else '')}\",\n+            desc=f\"file {first_path}\",\n             leave=False,\n             disable=files_count <= 1 or progress_bar_configuration.disable_progress_bar,\n         )\n \n-        for i, linted_file in enumerate(runner.run(fnames, fix), start=1):\n-            linted_path.add(linted_file)\n+        for i, linted_file in enumerate(runner.run(expanded_paths, fix), start=1):\n+            linted_dir = expanded_path_to_linted_dir[linted_file.path]\n+            linted_dir.add(linted_file)\n             # If any fatal errors, then stop iteration.\n             if any(v.fatal for v in linted_file.violations):  # pragma: no cover\n                 linter_logger.error(\"Fatal linting error. Halting further linting.\")\n                 break\n \n             # Progress bar for files is rendered only when there is more than one file.\n-            # Additionally as it's updated after each loop, we need to get file name\n+            # Additionally, as it's updated after each loop, we need to get file name\n             # from the next loop. This is why `enumerate` starts with `1` and there\n             # is `i < len` to not exceed files list length.\n             progress_bar_files.update(n=1)\n-            if i < len(fnames):\n-                progress_bar_files.set_description(\n-                    f\"file {os.path.basename(fnames[i])}\"\n-                )\n-\n-        return linted_path\n-\n-    def lint_paths(\n-        self,\n-        paths: Tuple[str, ...],\n-        fix: bool = False,\n-        ignore_non_existent_files: bool = False,\n-        ignore_files: bool = True,\n-        processes: Optional[int] = None,\n-    ) -> LintingResult:\n-        \"\"\"Lint an iterable of paths.\"\"\"\n-        paths_count = len(paths)\n-\n-        # If no paths specified - assume local\n-        if not paths_count:  # pragma: no cover\n-            paths = (os.getcwd(),)\n-        # Set up the result to hold what we get back\n-        result = LintingResult()\n-\n-        progress_bar_paths = tqdm(\n-            total=paths_count,\n-            desc=\"path\",\n-            leave=False,\n-            disable=paths_count <= 1 or progress_bar_configuration.disable_progress_bar,\n-        )\n-        for path in paths:\n-            progress_bar_paths.set_description(f\"path {path}\")\n-\n-            # Iterate through files recursively in the specified directory (if it's a\n-            # directory) or read the file directly if it's not\n-            result.add(\n-                self.lint_path(\n-                    path,\n-                    fix=fix,\n-                    ignore_non_existent_files=ignore_non_existent_files,\n-                    ignore_files=ignore_files,\n-                    processes=processes,\n-                )\n-            )\n-\n-            progress_bar_paths.update(1)\n+            if i < len(expanded_paths):\n+                progress_bar_files.set_description(f\"file {expanded_paths[i]}\")\n \n         result.stop_timer()\n         return result\n", "test_patch": "diff --git a/test/cli/commands_test.py b/test/cli/commands_test.py\n--- a/test/cli/commands_test.py\n+++ b/test/cli/commands_test.py\n@@ -1272,15 +1272,16 @@ def test__cli__command_lint_nocolor(isatty, should_strip_ansi, capsys, tmpdir):\n )\n @pytest.mark.parametrize(\"write_file\", [None, \"outfile\"])\n def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_path):\n-    \"\"\"Test the output output formats for multiple files.\n+    \"\"\"Test the output formats for multiple files.\n \n     This tests runs both stdout checking and file checking.\n     \"\"\"\n-    fpath = \"test/fixtures/linter/indentation_errors.sql\"\n+    fpath1 = \"test/fixtures/linter/indentation_errors.sql\"\n+    fpath2 = \"test/fixtures/linter/multiple_sql_errors.sql\"\n \n     cmd_args = (\n-        fpath,\n-        fpath,\n+        fpath1,\n+        fpath2,\n         \"--format\",\n         serialize,\n         \"--disable-progress-bar\",\n@@ -1313,7 +1314,7 @@ def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_\n     print(\"## End Payload\")\n \n     if serialize == \"human\":\n-        assert payload_length == 31 if write_file else 32\n+        assert payload_length == 26 if write_file else 32\n     elif serialize == \"json\":\n         result = json.loads(result_payload)\n         assert len(result) == 2\n@@ -1323,13 +1324,13 @@ def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_\n     elif serialize == \"github-annotation\":\n         result = json.loads(result_payload)\n         filepaths = {r[\"file\"] for r in result}\n-        assert len(filepaths) == 1\n+        assert len(filepaths) == 2\n     elif serialize == \"github-annotation-native\":\n         result = result_payload.split(\"\\n\")\n         # SQLFluff produces trailing newline\n         if result[-1] == \"\":\n             del result[-1]\n-        assert len(result) == 24\n+        assert len(result) == 17\n     else:\n         raise Exception\n \n@@ -1721,8 +1722,16 @@ def test_cli_lint_enabled_progress_bar_multiple_paths(\n         )\n         raw_output = repr(result.output)\n \n-        assert r\"\\rpath test/fixtures/linter/passing.sql:\" in raw_output\n-        assert r\"\\rpath test/fixtures/linter/indentation_errors.sql:\" in raw_output\n+        sep = os.sep\n+        if sys.platform == \"win32\":\n+            sep *= 2\n+        assert (\n+            r\"\\rfile test/fixtures/linter/passing.sql:\".replace(\"/\", sep) in raw_output\n+        )\n+        assert (\n+            r\"\\rfile test/fixtures/linter/indentation_errors.sql:\".replace(\"/\", sep)\n+            in raw_output\n+        )\n         assert r\"\\rlint by rules:\" in raw_output\n         assert r\"\\rrule L001:\" in raw_output\n         assert r\"\\rrule L049:\" in raw_output\n@@ -1741,9 +1750,27 @@ def test_cli_lint_enabled_progress_bar_multiple_files(\n         )\n         raw_output = repr(result.output)\n \n-        assert r\"\\rfile passing.1.sql:\" in raw_output\n-        assert r\"\\rfile passing.2.sql:\" in raw_output\n-        assert r\"\\rfile passing.3.sql:\" in raw_output\n+        sep = os.sep\n+        if sys.platform == \"win32\":\n+            sep *= 2\n+        assert (\n+            r\"\\rfile test/fixtures/linter/multiple_files/passing.1.sql:\".replace(\n+                \"/\", sep\n+            )\n+            in raw_output\n+        )\n+        assert (\n+            r\"\\rfile test/fixtures/linter/multiple_files/passing.2.sql:\".replace(\n+                \"/\", sep\n+            )\n+            in raw_output\n+        )\n+        assert (\n+            r\"\\rfile test/fixtures/linter/multiple_files/passing.3.sql:\".replace(\n+                \"/\", sep\n+            )\n+            in raw_output\n+        )\n         assert r\"\\rlint by rules:\" in raw_output\n         assert r\"\\rrule L001:\" in raw_output\n         assert r\"\\rrule L049:\" in raw_output\n", "problem_statement": "Multiple processes not used when list of explicit filenames is passed\n### Search before asking\n\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\n\n\n### What Happened\n\nWhen providing a long list of file names to `sqlfluff lint -p -1`, only a single CPU is used. This seems to stem from the fact that https://github.com/sqlfluff/sqlfluff/blob/a006378af8b670f9235653694dbcddd4c62d1ab9/src/sqlfluff/core/linter/linter.py#L1190 is iterating over the list of files. For each listed path there, it would run the found files in parallel. As we are inputting whole filenames here, a path equals a single file and thus `sqlfluff` would only process one file at a time.\r\n\r\nThe context here is the execution of `sqlfluff lint` inside a `pre-commit` hook.\n\n### Expected Behaviour\n\nAll CPU cores are used as `-p -1` is passed on the commandline.\n\n### Observed Behaviour\n\nOnly a single CPU core is used.\n\n### How to reproduce\n\nRun `sqlfluff lint -p -1` with a long list of files.\n\n### Dialect\n\nAffects all. \n\n### Version\n\n1.4.2\n\n### Configuration\n\nNone.\n\n### Are you willing to work on and submit a PR to address the issue?\n\n- [ ] Yes I am willing to submit a PR!\n\n### Code of Conduct\n\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\n\n", "hints_text": "I have been wondering for some time why sqlfluff never manages to use 100% of CPU. Running it on my Code base takes about 90 minutes. Though never more than 30% of cpu is used\u2026 maybe this sis the reason\u2026\nYeah - this looks like an accurate diagnosis. Most of the testing for the multiprocessing feature was done on large projects of multiple files, but _where a single path was passed_ e.g. `sqlfluff lint .`.\r\n\r\nThis seems like a very sensible improvement for people using the commit hook.\r\n\r\n@barrywhart - you did a lot of the original multiprocessing work. Reckon you could take this one on?\nI'll take a look, sure!", "created_at": "2022-11-18T22:49:21Z", "version": "1.3", "FAIL_TO_PASS": ["test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_paths", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_files"], "PASS_TO_PASS": ["test/cli/commands_test.py::test__cli__command_directed", "test/cli/commands_test.py::test__cli__command_dialect", "test/cli/commands_test.py::test__cli__command_no_dialect", "test/cli/commands_test.py::test__cli__command_parse_error_dialect_explicit_warning", "test/cli/commands_test.py::test__cli__command_parse_error_dialect_implicit_warning", "test/cli/commands_test.py::test__cli__command_dialect_legacy", "test/cli/commands_test.py::test__cli__command_extra_config_fail", "test/cli/commands_test.py::test__cli__command_lint_stdin[command0]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command1]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command2]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command3]", "test/cli/commands_test.py::test__cli__command_render_stdin", "test/cli/commands_test.py::test__cli__command_lint_parse[command0]", "test/cli/commands_test.py::test__cli__command_lint_parse[command1]", "test/cli/commands_test.py::test__cli__command_lint_parse[command2]", "test/cli/commands_test.py::test__cli__command_lint_parse[command3]", "test/cli/commands_test.py::test__cli__command_lint_parse[command4]", "test/cli/commands_test.py::test__cli__command_lint_parse[command5]", "test/cli/commands_test.py::test__cli__command_lint_parse[command6]", "test/cli/commands_test.py::test__cli__command_lint_parse[command7]", "test/cli/commands_test.py::test__cli__command_lint_parse[command8]", "test/cli/commands_test.py::test__cli__command_lint_parse[command9]", "test/cli/commands_test.py::test__cli__command_lint_parse[command10]", "test/cli/commands_test.py::test__cli__command_lint_parse[command11]", "test/cli/commands_test.py::test__cli__command_lint_parse[command12]", "test/cli/commands_test.py::test__cli__command_lint_parse[command13]", "test/cli/commands_test.py::test__cli__command_lint_parse[command14]", "test/cli/commands_test.py::test__cli__command_lint_parse[command15]", "test/cli/commands_test.py::test__cli__command_lint_parse[command16]", "test/cli/commands_test.py::test__cli__command_lint_parse[command17]", "test/cli/commands_test.py::test__cli__command_lint_parse[command18]", "test/cli/commands_test.py::test__cli__command_lint_parse[command19]", "test/cli/commands_test.py::test__cli__command_lint_parse[command20]", "test/cli/commands_test.py::test__cli__command_lint_parse[command21]", "test/cli/commands_test.py::test__cli__command_lint_parse[command22]", "test/cli/commands_test.py::test__cli__command_lint_parse[command23]", "test/cli/commands_test.py::test__cli__command_lint_parse[command24]", "test/cli/commands_test.py::test__cli__command_lint_parse[command25]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command0-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command1-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command2-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command3-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command4-1]", "test/cli/commands_test.py::test__cli__command_lint_warning_explicit_file_ignored", "test/cli/commands_test.py::test__cli__command_lint_skip_ignore_files", "test/cli/commands_test.py::test__cli__command_lint_ignore_local_config", "test/cli/commands_test.py::test__cli__command_lint_warning", "test/cli/commands_test.py::test__cli__command_versioning", "test/cli/commands_test.py::test__cli__command_version", "test/cli/commands_test.py::test__cli__command_rules", "test/cli/commands_test.py::test__cli__command_dialects", "test/cli/commands_test.py::test__cli__command__fix[L001-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/whitespace_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L003-test/fixtures/linter/indentation_error_hard.sql]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_templating_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_suppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[0_lint_errors_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[0_lint_errors_1_suppressed_parse_error]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[1_lint_error_1_unsuppressed_parse_error_FIX_EVEN_UNPARSABLE]", "test/cli/commands_test.py::test__cli__fix_error_handling_behavior[2_files_with_lint_errors_1_unsuppressed_parse_error]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[command-line-False]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[command-line-True]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[config-file-False]", "test/cli/commands_test.py::test_cli_fix_even_unparsable[config-file-True]", "test/cli/commands_test.py::test__cli__fix_loop_limit_behavior[--", "test/cli/commands_test.py::test__cli__command_fix_stdin[select", "test/cli/commands_test.py::test__cli__command_fix_stdin[", "test/cli/commands_test.py::test__cli__command_fix_stdin[SELECT", "test/cli/commands_test.py::test__cli__command_fix_stdin_logging_to_stderr", "test/cli/commands_test.py::test__cli__command_fix_stdin_safety", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[create", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[select", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-y-0-0]", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-n-1-1]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[None-yaml]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[None-json]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[outfile-yaml]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[outfile-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[select", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[SElect", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command0]", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command1]", "test/cli/commands_test.py::test__cli__command_lint_nocolor", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-human]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-yaml]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[None-github-annotation-native]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-human]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-yaml]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[outfile-github-annotation-native]", "test/cli/commands_test.py::test__cli__command_lint_serialize_github_annotation", "test/cli/commands_test.py::test__cli__command_lint_serialize_github_annotation_native", "test/cli/commands_test.py::test__cli__command_lint_serialize_annotation_level_error_failure_equivalent[github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_annotation_level_error_failure_equivalent[github-annotation-native]", "test/cli/commands_test.py::test___main___help", "test/cli/commands_test.py::test_encoding[utf-8-ascii]", "test/cli/commands_test.py::test_encoding[utf-8-sig-UTF-8-SIG]", "test/cli/commands_test.py::test_encoding[utf-32-UTF-32]", "test/cli/commands_test.py::test_cli_encoding[utf-8-command-line-False]", "test/cli/commands_test.py::test_cli_encoding[utf-8-SIG-command-line-True]", "test/cli/commands_test.py::test_cli_encoding[utf-8-config-file-False]", "test/cli/commands_test.py::test_cli_encoding[utf-8-SIG-config-file-True]", "test/cli/commands_test.py::test_cli_no_disable_noqa_flag", "test/cli/commands_test.py::test_cli_disable_noqa_flag", "test/cli/commands_test.py::test_cli_get_default_config", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_disabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_disabled_progress_bar_deprecated_option", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar", "test/cli/commands_test.py::test__cli__fix_multiple_errors_no_show_errors", "test/cli/commands_test.py::test__cli__fix_multiple_errors_show_errors", "test/cli/commands_test.py::test__cli__multiple_files__fix_multiple_errors_show_errors", "test/cli/commands_test.py::test__cli__render_fail", "test/cli/commands_test.py::test__cli__render_pass"], "environment_setup_commit": "dc59c2a5672aacedaf91f0e6129b467eefad331b"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-3648", "base_commit": "e56fc6002dac0fb7eb446d58bd8aa7a839908535", "patch": "diff --git a/src/sqlfluff/core/templaters/slicers/tracer.py b/src/sqlfluff/core/templaters/slicers/tracer.py\n--- a/src/sqlfluff/core/templaters/slicers/tracer.py\n+++ b/src/sqlfluff/core/templaters/slicers/tracer.py\n@@ -206,7 +206,8 @@ def __init__(self, raw_str: str, env: Environment):\n \n         # Internal bookkeeping\n         self.slice_id: int = 0\n-        self.inside_set_or_macro: bool = False  # {% set %} or {% macro %}\n+        # {% set %} or {% macro %} or {% call %}\n+        self.inside_set_macro_or_call: bool = False\n         self.inside_block = False  # {% block %}\n         self.stack: List[int] = []\n         self.idx_raw: int = 0\n@@ -236,12 +237,18 @@ def slice_info_for_literal(self, length, prefix=\"\") -> RawSliceInfo:\n             unique_alternate_id, alternate_code, inside_block=self.inside_block\n         )\n \n-    def update_inside_set_or_macro_or_block(\n-        self, block_type: str, trimmed_parts: List[str]\n-    ) -> None:\n-        \"\"\"Based on block tag, update whether we're in a set/macro section.\"\"\"\n+    def update_inside_set_call_macro_or_block(\n+        self,\n+        block_type: str,\n+        trimmed_parts: List[str],\n+        m_open: Optional[regex.Match],\n+        m_close: Optional[regex.Match],\n+        tag_contents: List[str],\n+    ) -> Optional[RawSliceInfo]:\n+        \"\"\"Based on block tag, update whether in a set/call/macro/block section.\"\"\"\n         if block_type == \"block_start\" and trimmed_parts[0] in (\n             \"block\",\n+            \"call\",\n             \"macro\",\n             \"set\",\n         ):\n@@ -274,16 +281,22 @@ def update_inside_set_or_macro_or_block(\n                     if trimmed_parts[0] == \"block\":\n                         self.inside_block = True\n                     else:\n-                        self.inside_set_or_macro = True\n+                        result = None\n+                        if trimmed_parts[0] == \"call\":\n+                            assert m_open and m_close\n+                            result = self.track_call(m_open, m_close, tag_contents)\n+                        self.inside_set_macro_or_call = True\n+                        return result\n                 else:\n                     raise  # pragma: no cover\n         elif block_type == \"block_end\":\n-            if trimmed_parts[0] in (\"endmacro\", \"endset\"):\n-                # Exiting a set or macro.\n-                self.inside_set_or_macro = False\n+            if trimmed_parts[0] in (\"endcall\", \"endmacro\", \"endset\"):\n+                # Exiting a set or macro or block.\n+                self.inside_set_macro_or_call = False\n             elif trimmed_parts[0] == \"endblock\":\n                 # Exiting a {% block %} block.\n                 self.inside_block = False\n+        return None\n \n     def make_raw_slice_info(\n         self,\n@@ -292,7 +305,7 @@ def make_raw_slice_info(\n         inside_block: bool = False,\n     ) -> RawSliceInfo:\n         \"\"\"Create RawSliceInfo as given, or \"empty\" if in set/macro block.\"\"\"\n-        if not self.inside_set_or_macro:\n+        if not self.inside_set_macro_or_call:\n             return RawSliceInfo(unique_alternate_id, alternate_code, [], inside_block)\n         else:\n             return RawSliceInfo(None, None, [], False)\n@@ -355,6 +368,8 @@ def analyze(self, make_template: Callable[[str], Template]) -> JinjaTracer:\n             # raw_end and raw_begin behave a little differently in\n             # that the whole tag shows up in one go rather than getting\n             # parts of the tag at a time.\n+            m_open = None\n+            m_close = None\n             if elem_type.endswith(\"_end\") or elem_type == \"raw_begin\":\n                 block_type = self.block_types[elem_type]\n                 block_subtype = None\n@@ -376,7 +391,11 @@ def analyze(self, make_template: Callable[[str], Template]) -> JinjaTracer:\n                         raw_slice_info = self.track_templated(\n                             m_open, m_close, tag_contents\n                         )\n-                self.update_inside_set_or_macro_or_block(block_type, tag_contents)\n+                raw_slice_info_temp = self.update_inside_set_call_macro_or_block(\n+                    block_type, tag_contents, m_open, m_close, tag_contents\n+                )\n+                if raw_slice_info_temp:\n+                    raw_slice_info = raw_slice_info_temp\n                 m_strip_right = regex.search(\n                     r\"\\s+$\", raw, regex.MULTILINE | regex.DOTALL\n                 )\n@@ -428,6 +447,7 @@ def analyze(self, make_template: Callable[[str], Template]) -> JinjaTracer:\n                     slice_idx = len(self.raw_sliced) - 1\n                     self.idx_raw += len(str_buff)\n                 if block_type.startswith(\"block\"):\n+                    self.track_block_start(block_type, tag_contents[0])\n                     self.track_block_end(block_type, tag_contents[0])\n                     self.update_next_slice_indices(\n                         slice_idx, block_type, tag_contents[0]\n@@ -457,6 +477,21 @@ def track_templated(\n         )\n         return self.make_raw_slice_info(unique_alternate_id, alternate_code)\n \n+    def track_call(\n+        self, m_open: regex.Match, m_close: regex.Match, tag_contents: List[str]\n+    ):\n+        \"\"\"Set up tracking for \"{% call ... %}\".\"\"\"\n+        unique_alternate_id = self.next_slice_id()\n+        open_ = m_open.group(1)\n+        close_ = m_close.group(1)\n+        # Here, we still need to evaluate the original tag contents, e.g. in\n+        # case it has intentional side effects, but also return a slice ID\n+        # for tracking.\n+        alternate_code = (\n+            f\"\\0{unique_alternate_id} {open_} \" f\"{''.join(tag_contents)} {close_}\"\n+        )\n+        return self.make_raw_slice_info(unique_alternate_id, alternate_code)\n+\n     def track_literal(self, raw: str, block_idx: int) -> None:\n         \"\"\"Set up tracking for a Jinja literal.\"\"\"\n         self.raw_sliced.append(\n@@ -517,6 +552,20 @@ def extract_tag_contents(\n             trimmed_parts = trimmed_content.split()\n         return trimmed_parts\n \n+    def track_block_start(self, block_type: str, tag_name: str) -> None:\n+        \"\"\"On starting a 'call' block, set slice_type to \"templated\".\"\"\"\n+        if block_type == \"block_start\" and tag_name == \"call\":\n+            # Replace RawSliceInfo for this slice with one that has block_type\n+            # \"templated\".\n+            old_raw_file_slice = self.raw_sliced[-1]\n+            self.raw_sliced[-1] = old_raw_file_slice._replace(slice_type=\"templated\")\n+\n+            # Move existing raw_slice_info entry since it's keyed by RawFileSlice.\n+            self.raw_slice_info[self.raw_sliced[-1]] = self.raw_slice_info[\n+                old_raw_file_slice\n+            ]\n+            del self.raw_slice_info[old_raw_file_slice]\n+\n     def track_block_end(self, block_type: str, tag_name: str) -> None:\n         \"\"\"On ending a 'for' or 'if' block, set up tracking.\"\"\"\n         if block_type == \"block_end\" and tag_name in (\n@@ -553,7 +602,7 @@ def update_next_slice_indices(\n             \"endfor\",\n             \"endif\",\n         ):\n-            if not self.inside_set_or_macro:\n+            if not self.inside_set_macro_or_call:\n                 # Record potential forward jump over this block.\n                 self.raw_slice_info[\n                     self.raw_sliced[self.stack[-1]]\n", "test_patch": "diff --git a/test/core/templaters/jinja_test.py b/test/core/templaters/jinja_test.py\n--- a/test/core/templaters/jinja_test.py\n+++ b/test/core/templaters/jinja_test.py\n@@ -697,6 +697,14 @@ def test__templater_jinja_slice_template(test, result):\n     ] == result\n \n \n+def _statement(*args, **kwargs):\n+    return \"_statement\"\n+\n+\n+def _load_result(*args, **kwargs):\n+    return \"_load_result\"\n+\n+\n @pytest.mark.parametrize(\n     \"raw_file,override_context,result\",\n     [\n@@ -1118,6 +1126,32 @@ def test__templater_jinja_slice_template(test, result):\n                 (\"literal\", slice(131, 132, None), slice(88, 89, None)),\n             ],\n         ),\n+        (\n+            \"\"\"{{ statement('variables', fetch_result=true) }}\n+\"\"\",\n+            dict(\n+                statement=_statement,\n+                load_result=_load_result,\n+            ),\n+            [\n+                (\"templated\", slice(0, 47, None), slice(0, 10, None)),\n+                (\"literal\", slice(47, 48, None), slice(10, 11, None)),\n+            ],\n+        ),\n+        (\n+            \"\"\"{% call statement('variables', fetch_result=true) %}select 1 as test{% endcall %}\n+\"\"\",\n+            dict(\n+                statement=_statement,\n+                load_result=_load_result,\n+            ),\n+            [\n+                (\"templated\", slice(0, 52, None), slice(0, 10, None)),\n+                (\"literal\", slice(52, 68, None), slice(10, 10, None)),\n+                (\"block_end\", slice(68, 81, None), slice(10, 10, None)),\n+                (\"literal\", slice(81, 82, None), slice(10, 11, None)),\n+            ],\n+        ),\n     ],\n )\n def test__templater_jinja_slice_file(raw_file, override_context, result, caplog):\n", "problem_statement": "dbt & JinjaTracer results in passing invalid query to database (was: DBT Call statement() block causes invalid query generated)\n### Search before asking\n\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\n\n\n### What Happened\n\nWhen using the call statement() to run a query during compile time, the query generated is garbled causing the following sql error:\r\n```\r\n{% call statement('variables', fetch_result=true) %}\r\n\r\nselect 1 as test;\r\n\r\n{% endcall %}\r\n\r\n{% set test = load_result('variables')['table'].columns.TEST.values()[0] %}\r\n```\r\n\r\nThis results in the following error:\r\n\r\ndbt.exceptions.DatabaseException: Database Error\r\n  001003 (42000): SQL compilation error:\r\n  syntax error line 1 at position 0 unexpected '0'.\r\n\r\nThe query ran looks like this when looking at the query runner history in snowflake:\r\n\r\n```\r\n\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a_0\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a_8\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a\u263a_0\r\n```\r\n\r\nWhereas it should show:\r\n```\r\nselect 1 as test;\r\n```\n\n### Expected Behaviour\n\nExpected that the query runs properly.\n\n### Observed Behaviour\n\n```\r\n=== [dbt templater] Compiling dbt project...\r\n=== [dbt templater] Project Compiled.\r\nTraceback (most recent call last):\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/snowflake/connections.py\", line 219, in exception_handler\r\n    yield\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/sql/connections.py\", line 70, in add_query\r\n    cursor.execute(sql, bindings)\r\n  File \"/usr/local/lib/python3.9/site-packages/snowflake/connector/cursor.py\", line 794, in execute\r\n    Error.errorhandler_wrapper(self.connection, self, error_class, errvalue)\r\n  File \"/usr/local/lib/python3.9/site-packages/snowflake/connector/errors.py\", line 273, in errorhandler_wrapper\r\n    handed_over = Error.hand_to_other_handler(\r\n  File \"/usr/local/lib/python3.9/site-packages/snowflake/connector/errors.py\", line 328, in hand_to_other_handler\r\n    cursor.errorhandler(connection, cursor, error_class, error_value)\r\n  File \"/usr/local/lib/python3.9/site-packages/snowflake/connector/errors.py\", line 207, in default_errorhandler\r\n    raise error_class(\r\nsnowflake.connector.errors.ProgrammingError: 001003 (42000): SQL compilation error:\r\nsyntax error line 1 at position 0 unexpected '0'.\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"/usr/local/bin/sqlfluff\", line 8, in <module>\r\n    sys.exit(cli())\r\n  File \"/usr/local/lib/python3.9/site-packages/click/core.py\", line 1130, in __call__\r\n    return self.main(*args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/click/core.py\", line 1055, in main\r\n    rv = self.invoke(ctx)\r\n  File \"/usr/local/lib/python3.9/site-packages/click/core.py\", line 1657, in invoke\r\n    return _process_result(sub_ctx.command.invoke(sub_ctx))\r\n  File \"/usr/local/lib/python3.9/site-packages/click/core.py\", line 1404, in invoke\r\n    return ctx.invoke(self.callback, **ctx.params)\r\n  File \"/usr/local/lib/python3.9/site-packages/click/core.py\", line 760, in invoke\r\n    return __callback(*args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/cli/commands.py\", line 1008, in parse\r\n    parsed_strings = list(\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/linter/linter.py\", line 1171, in parse_path\r\n    yield self.parse_string(\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/linter/linter.py\", line 835, in parse_string\r\n    rendered = self.render_string(in_str, fname, config, encoding)\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/linter/linter.py\", line 784, in render_string\r\n    templated_file, templater_violations = self.templater.process(\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/templaters/base.py\", line 47, in _wrapped\r\n    return func(self, in_str=in_str, fname=fname, config=config, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff_templater_dbt/templater.py\", line 331, in process\r\n    processed_result = self._unsafe_process(fname_absolute_path, in_str, config)\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff_templater_dbt/templater.py\", line 552, in _unsafe_process\r\n    raw_sliced, sliced_file, templated_sql = self.slice_file(\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/templaters/jinja.py\", line 462, in slice_file\r\n    trace = tracer.trace(append_to_templated=kwargs.pop(\"append_to_templated\", \"\"))\r\n  File \"/usr/local/lib/python3.9/site-packages/sqlfluff/core/templaters/slicers/tracer.py\", line 77, in trace\r\n    trace_template_output = trace_template.render()\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/environment.py\", line 1090, in render\r\n    self.environment.handle_exception()\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/environment.py\", line 832, in handle_exception\r\n    reraise(*rewrite_traceback_stack(source=source))\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/_compat.py\", line 28, in reraise\r\n    raise value.with_traceback(tb)\r\n  File \"<template>\", line 16, in top-level template code\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/sandbox.py\", line 462, in call\r\n    return __context.call(__obj, *args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/clients/jinja.py\", line 321, in __call__\r\n    return self.call_macro(*args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/clients/jinja.py\", line 248, in call_macro\r\n    return macro(*args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/runtime.py\", line 679, in _invoke\r\n    rv = self._func(*arguments)\r\n  File \"<template>\", line 10, in template\r\n  File \"/usr/local/lib/python3.9/site-packages/jinja2/sandbox.py\", line 462, in call\r\n    return __context.call(__obj, *args, **kwargs)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/base/impl.py\", line 235, in execute\r\n    return self.connections.execute(sql=sql, auto_begin=auto_begin, fetch=fetch)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/sql/connections.py\", line 122, in execute\r\n    _, cursor = self.add_query(sql, auto_begin)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/snowflake/connections.py\", line 458, in add_query\r\n    connection, cursor = super().add_query(\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/sql/connections.py\", line 78, in add_query\r\n    return connection, cursor\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/contextlib.py\", line 137, in __exit__\r\n    self.gen.throw(typ, value, traceback)\r\n  File \"/usr/local/lib/python3.9/site-packages/dbt/adapters/snowflake/connections.py\", line 238, in exception_handler\r\n    raise DatabaseException(msg)\r\ndbt.exceptions.DatabaseException: Database Error\r\n  001003 (42000): SQL compilation error:\r\n  syntax error line 1 at position 0 unexpected '0'.\r\n```\n\n### How to reproduce\n\nUse the statement() block described here:\r\nhttps://docs.getdbt.com/reference/dbt-jinja-functions/statement-blocks\r\n\r\n\n\n### Dialect\n\nSnowflake\n\n### Version\n\n1.2.0 with the dbt formatter\n\n### Configuration\n\n```\r\n[sqlfluff]\r\n# verbose is an integer (0-2) indicating the level of log output\r\nverbose = 2\r\n# Turn off color formatting of output\r\nnocolor = False\r\n# Supported dialects https://docs.sqlfluff.com/en/stable/dialects.html\r\n# Or run 'sqlfluff dialects'\r\ndialect = snowflake\r\n# One of [raw|jinja|python|placeholder]\r\ntemplater = dbt\r\n# Comma separated list of rules to check, default to all\r\nrules = all\r\n# Comma separated list of rules to exclude, or None\r\nexclude_rules = None\r\n# The depth to recursively parse to (0 for unlimited)\r\nrecurse = 0\r\n# Below controls SQLFluff output, see max_line_length for SQL output\r\noutput_line_length = 80\r\n# Number of passes to run before admitting defeat\r\nrunaway_limit = 10\r\n# Ignore errors by category (one or more of the following, separated by commas: lexing,linting,parsing,templating)\r\nignore = None\r\n# Ignore linting errors found within sections of code coming directly from\r\n# templated code (e.g. from within Jinja curly braces. Note that it does not\r\n# ignore errors from literal code found within template loops.\r\nignore_templated_areas = True\r\n# can either be autodetect or a valid encoding e.g. utf-8, utf-8-sig\r\nencoding = autodetect\r\n# Ignore inline overrides (e.g. to test if still required)\r\ndisable_noqa = False\r\n# Comma separated list of file extensions to lint\r\n# NB: This config will only apply in the root folder\r\nsql_file_exts = .sql,.sql.j2,.dml,.ddl\r\n# Allow fix to run on files, even if they contain parsing errors\r\n# Note altering this is NOT RECOMMENDED as can corrupt SQL\r\nfix_even_unparsable = False\r\n# Very large files can make the parser effectively hang.\r\n# This limit skips files over a certain character length\r\n# and warns the user what has happened.\r\n# Set this to 0 to disable.\r\nlarge_file_skip_char_limit = 20000\r\n\r\n[sqlfluff:indentation]\r\n# See https://docs.sqlfluff.com/en/stable/indentation.html\r\nindented_joins = False\r\nindented_ctes = False\r\nindented_using_on = True\r\nindented_on_contents = True\r\ntemplate_blocks_indent = True\r\n\r\n[sqlfluff:templater]\r\nunwrap_wrapped_queries = True\r\n\r\n[sqlfluff:templater:jinja]\r\napply_dbt_builtins = True\r\nload_macros_from_path = macros/\r\n\r\n[sqlfluff:templater:jinja:macros]\r\n# Macros provided as builtins for dbt projects\r\ndbt_ref = {% macro ref(model_ref) %}{{model_ref}}{% endmacro %}\r\ndbt_source = {% macro source(source_name, table) %}{{source_name}}_{{table}}{% endmacro %}\r\ndbt_config = {% macro config() %}{% for k in kwargs %}{% endfor %}{% endmacro %}\r\ndbt_var = {% macro var(variable, default='') %}item{% endmacro %}\r\ndbt_is_incremental = {% macro is_incremental() %}True{% endmacro %}\r\n\r\n[sqlfluff:templater:dbt]\r\nproject_dir = ./\r\n\r\n# Some rules can be configured directly from the config common to other rules\r\n[sqlfluff:rules]\r\ntab_space_size = 4\r\nmax_line_length = 120\r\nindent_unit = space\r\ncomma_style = trailing\r\nallow_scalar = True\r\nsingle_table_references = consistent\r\nunquoted_identifiers_policy = all\r\n\r\n# Some rules have their own specific config\r\n[sqlfluff:rules:L003]\r\nhanging_indents = True\r\n\r\n[sqlfluff:rules:L007]\r\noperator_new_lines = after\r\n\r\n[sqlfluff:rules:L010]\r\n# Keywords\r\ncapitalisation_policy = lower\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L011]\r\n# Aliasing preference for tables\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:L012]\r\n# Aliasing preference for columns\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:L014]\r\n# Unquoted identifiers\r\nextended_capitalisation_policy = lower\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L016]\r\n# Line length\r\nignore_comment_lines = False\r\nignore_comment_clauses = False\r\n\r\n[sqlfluff:rules:L027]\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L026]\r\n# References must be in FROM clause\r\n# Disabled for some dialects (e.g. bigquery)\r\nforce_enable = False\r\n\r\n[sqlfluff:rules:L028]\r\n# References must be consistently used\r\n# Disabled for some dialects (e.g. bigquery)\r\nforce_enable = False\r\n\r\n[sqlfluff:rules:L029]\r\n# Keywords should not be used as identifiers.\r\nunquoted_identifiers_policy = aliases\r\nquoted_identifiers_policy = none\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L030]\r\n# Function names\r\nextended_capitalisation_policy = lower\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L031]\r\n# Avoid table aliases in from clauses and join conditions.\r\n# Disabled for some dialects (e.g. bigquery)\r\nforce_enable = False\r\n\r\n[sqlfluff:rules:L036]\r\nwildcard_policy = single\r\n\r\n[sqlfluff:rules:L038]\r\n# Trailing commas\r\nselect_clause_trailing_comma = forbid\r\n\r\n[sqlfluff:rules:L040]\r\n# Null & Boolean Literals\r\ncapitalisation_policy = consistent\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L042]\r\n# By default, allow subqueries in from clauses, but not join clauses\r\nforbid_subquery_in = join\r\n\r\n[sqlfluff:rules:L047]\r\n# Consistent syntax to count all rows\r\nprefer_count_1 = False\r\nprefer_count_0 = False\r\n\r\n[sqlfluff:rules:L051]\r\n# Fully qualify JOIN clause\r\nfully_qualify_join_types = inner\r\n\r\n[sqlfluff:rules:L052]\r\n# Semi-colon formatting approach\r\nmultiline_newline = False\r\nrequire_final_semicolon = False\r\n\r\n[sqlfluff:rules:L054]\r\n# GROUP BY/ORDER BY column references\r\ngroup_by_and_order_by_style = consistent\r\n\r\n[sqlfluff:rules:L057]\r\n# Special characters in identifiers\r\nunquoted_identifiers_policy = all\r\nquoted_identifiers_policy = all\r\nallow_space_in_identifier = False\r\nadditional_allowed_characters = \"\"\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L059]\r\n# Policy on quoted and unquoted identifiers\r\nprefer_quoted_identifiers = False\r\nignore_words = None\r\nignore_words_regex = None\r\nforce_enable = False\r\n\r\n[sqlfluff:rules:L062]\r\n# Comma separated list of blocked words that should not be used\r\nblocked_words = None\r\nblocked_regex = None\r\n\r\n[sqlfluff:rules:L063]\r\n# Data Types\r\nextended_capitalisation_policy = consistent\r\n# Comma separated list of words to ignore for this rule\r\nignore_words = None\r\nignore_words_regex = None\r\n\r\n[sqlfluff:rules:L064]\r\n# Consistent usage of preferred quotes for quoted literals\r\npreferred_quoted_literal_style = consistent\r\n# Disabled for dialects that do not support single and double quotes for quoted literals (e.g. Postgres)\r\nforce_enable = False\r\n\r\n[sqlfluff:rules:L066]\r\nmin_alias_length = None\r\nmax_alias_length = None\r\n```\n\n### Are you willing to work on and submit a PR to address the issue?\n\n- [ ] Yes I am willing to submit a PR!\n\n### Code of Conduct\n\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\n\n", "hints_text": "This is a curious error. Can you run `dbt compile` and post what dbt expects the compiled form of this statement to be? I worry that while a query is run at compile time, this query otherwise compiles to an empty file - and that could be causing issues.\ndbt compile doesn't output the call statement blocks since they're interpreted at runtime; however, we can see the output ran on the snowflake query history.\r\n\r\nSource test.sql\r\n```\r\n{% call statement('variables', fetch_result=true) %}\r\n\r\n    select 1\r\n\r\n{% endcall %}\r\n\r\nwith source (\r\n    select 1\r\n)\r\n\r\nselect * from source\r\n```\r\n\r\nCompiled output of test.sql\r\n```\r\n\r\n\r\nwith source (\r\n    select 1\r\n)\r\n\r\nselect * from source\r\n```\nThe dbt [documentation](https://docs.getdbt.com/reference/dbt-jinja-functions/statement-blocks) mentions re: `statement()`:\r\n\r\n>Volatile API\r\n>While the statement and load_result setup works for now, we intend to improve this interface in the future. If you have questions or suggestions, please let us know in GitHub or on Slack.\r\n\r\nSo this might be a relatively lower priority issue. IIUC, it may also be dbt specific (not affecting the `jinja` templater).\r\n\r\n\nI did some preliminary investigation. IIUC, SQLFluff's `JinjaTracer` should treat this:\r\n```\r\n{% call statement('variables', fetch_result=true) %}\r\n\r\nselect 1 as test;\r\n\r\n{% endcall %}\r\n```\r\n\r\nlike this:\r\n```\r\n{{ statement('variables', fetch_result=true) }}\r\n```\r\n\r\nIn both cases, whatever `statement()` returns is passed through to the template output. I think this will be pretty straightforward, other than the usual trickiness of working on this complex area of the code.", "created_at": "2022-07-21T22:42:19Z", "version": "1.2", "FAIL_TO_PASS": ["test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[{%"], "PASS_TO_PASS": ["test/core/templaters/jinja_test.py::test__templater_jinja[simple]", "test/core/templaters/jinja_test.py::test__templater_jinja[unboundlocal_bugfix]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[basic_block]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[strip_left_block]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[strip_both_block]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[basic_data]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[strip_right_data]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[strip_both_data]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[strip_both_comment]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[union_all_loop1]", "test/core/templaters/jinja_test.py::test__templater_jinja_slices[set_multiple_variables_and_define_macro]", "test/core/templaters/jinja_test.py::test_templater_set_block_handling", "test/core/templaters/jinja_test.py::test__templater_jinja_error_variable", "test/core/templaters/jinja_test.py::test__templater_jinja_dynamic_variable_no_violations", "test/core/templaters/jinja_test.py::test__templater_jinja_error_syntax", "test/core/templaters/jinja_test.py::test__templater_jinja_error_catastrophic", "test/core/templaters/jinja_test.py::test__templater_jinja_error_macro_path_does_not_exist", "test/core/templaters/jinja_test.py::test__templater_jinja_lint_empty", "test/core/templaters/jinja_test.py::test__templater_full[jinja_a/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_b/jinja-False-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_config-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_is_incremental-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_ref-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_source-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_this-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/dbt_builtins_var_default-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_e/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_f/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_g_macros/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_h_macros/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_i_raw/raw_tag-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_i_raw/raw_tag_2-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_j_libraries/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_k_config_override_path_macros/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_l_metas/001-False-True]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_l_metas/002-False-True]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_m_libraries_module/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_n_nested_macros/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_o_config_override_dbt_builtins/override_dbt_builtins-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_p_disable_dbt_builtins/disable_dbt_builtins-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_q_multiple_path_macros/jinja-True-False]", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_template[-result0]", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_template[foo-result1]", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_template[foo", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_template[SELECT", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_template[{%", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[-None-result0]", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[foo-None-result1]", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[SELECT", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[{{", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[SELECT\\n", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[{%-", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[select\\n", "test/core/templaters/jinja_test.py::test__templater_jinja_large_file_check"], "environment_setup_commit": "388dd01e05c7dcb880165c7241ed4027d9d0171e"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-905", "base_commit": "62e8dc3a148c40c0c28f62b23e943692a3198846", "patch": "diff --git a/src/sqlfluff/core/rules/std/L036.py b/src/sqlfluff/core/rules/std/L036.py\n--- a/src/sqlfluff/core/rules/std/L036.py\n+++ b/src/sqlfluff/core/rules/std/L036.py\n@@ -90,21 +90,26 @@ def _get_indexes(segment):\n         )\n \n     def _eval_multiple_select_target_elements(self, select_targets_info, segment):\n-        if select_targets_info.first_new_line_idx == -1:\n-            # there are multiple select targets but no new lines\n-\n-            # Find and delete any whitespace between \"SELECT\" and its targets.\n-            ws_to_delete = segment.select_children(\n-                start_seg=segment.segments[select_targets_info.select_idx],\n-                select_if=lambda s: s.is_type(\"whitespace\"),\n-                loop_while=lambda s: s.is_type(\"whitespace\") or s.is_meta,\n+        \"\"\"Multiple select targets. Ensure each is on a separate line.\"\"\"\n+        # Insert newline before every select target.\n+        fixes = []\n+        for i, select_target in enumerate(select_targets_info.select_targets):\n+            base_segment = (\n+                segment if not i else select_targets_info.select_targets[i - 1]\n             )\n-            fixes = [LintFix(\"delete\", ws) for ws in ws_to_delete]\n-            # Insert newline before the first select target.\n-            ins = self.make_newline(\n-                pos_marker=segment.pos_marker.advance_by(segment.raw)\n-            )\n-            fixes.append(LintFix(\"create\", select_targets_info.select_targets[0], ins))\n+            if base_segment.pos_marker.line_no == select_target.pos_marker.line_no:\n+                # Find and delete any whitespace before the select target.\n+                ws_to_delete = segment.select_children(\n+                    start_seg=segment.segments[select_targets_info.select_idx]\n+                    if not i\n+                    else select_targets_info.select_targets[i - 1],\n+                    select_if=lambda s: s.is_type(\"whitespace\"),\n+                    loop_while=lambda s: s.is_type(\"whitespace\", \"comma\") or s.is_meta,\n+                )\n+                fixes += [LintFix(\"delete\", ws) for ws in ws_to_delete]\n+                ins = self.make_newline(pos_marker=select_target.pos_marker)\n+                fixes.append(LintFix(\"create\", select_target, ins))\n+        if fixes:\n             return LintResult(anchor=segment, fixes=fixes)\n \n     def _eval_single_select_target_element(\n", "test_patch": "diff --git a/test/api/simple_test.py b/test/api/simple_test.py\n--- a/test/api/simple_test.py\n+++ b/test/api/simple_test.py\n@@ -121,7 +121,14 @@ def test__api__fix_string():\n     # Check return types.\n     assert isinstance(result, str)\n     # Check actual result\n-    assert result == \"SELECT\\n    *, 1, blah AS foo FROM mytable\\n\"\n+    assert (\n+        result\n+        == \"\"\"SELECT\n+    *,\n+    1,\n+    blah AS foo FROM mytable\n+\"\"\"\n+    )\n \n \n def test__api__fix_string_specific():\ndiff --git a/test/fixtures/rules/std_rule_cases/L036.yml b/test/fixtures/rules/std_rule_cases/L036.yml\n--- a/test/fixtures/rules/std_rule_cases/L036.yml\n+++ b/test/fixtures/rules/std_rule_cases/L036.yml\n@@ -37,7 +37,7 @@ test_multiple_select_targets_all_on_the_same_line:\n   fail_str: |\n     select a, b, c\n     from x\n-  fix_str: \"select\\na, b, c\\nfrom x\\n\"\n+  fix_str: \"select\\na,\\nb,\\nc\\nfrom x\\n\"\n \n test_multiple_select_targets_trailing_whitespace_after_select:\n   # TRICKY: Use explicit newlines to preserve the trailing space after \"SELECT\".\n@@ -57,3 +57,47 @@ test_comment_between_select_and_single_select_target:\n         -- This is the user's ID.\n         FROM\n         safe_user\n+\n+test_multiple_select_targets_some_newlines_missing_1:\n+  fail_str: |\n+    select\n+      a, b, c,\n+      d, e, f, g,\n+      h\n+    from x\n+  # The spaces before a, d, and h look odd, but these are places where the\n+  # select targets were already on a separate line, and the rule made no\n+  # changes.\n+  fix_str: |\n+    select\n+      a,\n+    b,\n+    c,\n+      d,\n+    e,\n+    f,\n+    g,\n+      h\n+    from x\n+\n+\n+test_multiple_select_targets_some_newlines_missing_2:\n+  fail_str: |\n+    select a, b, c,\n+      d, e, f, g,\n+      h\n+    from x\n+  # The spaces before d, and h look odd, but these are places where the\n+  # select targets were already on a separate line, and the rule made no\n+  # changes.\n+  fix_str: |\n+    select\n+    a,\n+    b,\n+    c,\n+      d,\n+    e,\n+    f,\n+    g,\n+      h\n+    from x\n", "problem_statement": "Enhance rule L036 to put all columns on separate lines if any of them are\nThe current description is ambiguous, but after discussion, we decided to update the rule and keep the description at least _similar_ to what it is currently.. See discussion on #769.\n", "hints_text": "FWIW I'm a +1 for this...", "created_at": "2021-03-28T21:22:12Z", "version": "0.4", "FAIL_TO_PASS": ["test/api/simple_test.py::test__api__fix_string"], "PASS_TO_PASS": ["test/api/simple_test.py::test__api__lint_string_without_violations", "test/api/simple_test.py::test__api__lint_string", "test/api/simple_test.py::test__api__lint_file", "test/api/simple_test.py::test__api__lint_string_specific", "test/api/simple_test.py::test__api__fix_string_specific", "test/api/simple_test.py::test__api__parse_string"], "environment_setup_commit": "cbdcfb09feb4883de91de142956c3be6ac7f827d"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1395", "base_commit": "26579bec7e65296223503b9e05da4af914af6777", "patch": "diff --git a/pvlib/iotools/__init__.py b/pvlib/iotools/__init__.py\n--- a/pvlib/iotools/__init__.py\n+++ b/pvlib/iotools/__init__.py\n@@ -15,6 +15,7 @@\n from pvlib.iotools.pvgis import get_pvgis_tmy, read_pvgis_tmy  # noqa: F401\n from pvlib.iotools.pvgis import read_pvgis_hourly  # noqa: F401\n from pvlib.iotools.pvgis import get_pvgis_hourly  # noqa: F401\n+from pvlib.iotools.pvgis import get_pvgis_horizon  # noqa: F401\n from pvlib.iotools.bsrn import get_bsrn  # noqa: F401\n from pvlib.iotools.bsrn import read_bsrn  # noqa: F401\n from pvlib.iotools.bsrn import parse_bsrn  # noqa: F401\ndiff --git a/pvlib/iotools/pvgis.py b/pvlib/iotools/pvgis.py\n--- a/pvlib/iotools/pvgis.py\n+++ b/pvlib/iotools/pvgis.py\n@@ -665,3 +665,57 @@ def read_pvgis_tmy(filename, pvgis_format=None, map_variables=None):\n         data = data.rename(columns=VARIABLE_MAP)\n \n     return data, months_selected, inputs, meta\n+\n+\n+def get_pvgis_horizon(latitude, longitude, url=URL, **kwargs):\n+    \"\"\"Get horizon data from PVGIS.\n+\n+    Parameters\n+    ----------\n+    latitude : float\n+        Latitude in degrees north\n+    longitude : float\n+        Longitude in degrees east\n+    url: str, default: :const:`pvlib.iotools.pvgis.URL`\n+        Base URL for PVGIS\n+    kwargs:\n+        Passed to requests.get\n+\n+    Returns\n+    -------\n+    data : pd.Series\n+        Pandas Series of the retrived horizon elevation angles. Index is the\n+        corresponding horizon azimuth angles.\n+    metadata : dict\n+        Metadata returned by PVGIS.\n+\n+    Notes\n+    -----\n+    The horizon azimuths are specified clockwise from north, e.g., south=180.\n+    This is the standard pvlib convention, although the PVGIS website specifies\n+    south=0.\n+\n+    References\n+    ----------\n+    .. [1] `PVGIS horizon profile tool\n+       <https://ec.europa.eu/jrc/en/PVGIS/tools/horizon>`_\n+    \"\"\"\n+    params = {'lat': latitude, 'lon': longitude, 'outputformat': 'json'}\n+    res = requests.get(url + 'printhorizon', params=params, **kwargs)\n+    if not res.ok:\n+        try:\n+            err_msg = res.json()\n+        except Exception:\n+            res.raise_for_status()\n+        else:\n+            raise requests.HTTPError(err_msg['message'])\n+    json_output = res.json()\n+    metadata = json_output['meta']\n+    data = pd.DataFrame(json_output['outputs']['horizon_profile'])\n+    data.columns = ['horizon_azimuth', 'horizon_elevation']\n+    # Convert azimuth to pvlib convention (north=0, south=180)\n+    data['horizon_azimuth'] += 180\n+    data.set_index('horizon_azimuth', inplace=True)\n+    data = data['horizon_elevation']  # convert to pd.Series\n+    data = data[data.index < 360]  # remove duplicate north point (0 and 360)\n+    return data, metadata\n", "test_patch": "diff --git a/pvlib/data/test_read_pvgis_horizon.csv b/pvlib/data/test_read_pvgis_horizon.csv\nnew file mode 100644\n--- /dev/null\n+++ b/pvlib/data/test_read_pvgis_horizon.csv\n@@ -0,0 +1,49 @@\n+horizon_azimuth,horizon_elevation\n+0,9.9\n+7.5,13\n+15,14.5\n+22.5,15.7\n+30,14.9\n+37.5,15.3\n+45,15.7\n+52.5,15.7\n+60,13\n+67.5,11.5\n+75,11.1\n+82.5,11.5\n+90,10.3\n+97.5,11.5\n+105,10.3\n+112.5,9.5\n+120,10.7\n+127.5,11.8\n+135,11.8\n+142.5,8.8\n+150,8.4\n+157.5,7.3\n+165,5.7\n+172.5,5.7\n+180,4.6\n+187.5,3.4\n+195,0.8\n+202.5,0\n+210,0\n+217.5,0\n+225,0\n+232.5,0\n+240,0\n+247.5,0\n+255,0\n+262.5,0\n+270,0\n+277.5,0\n+285,0\n+292.5,0\n+300,0\n+307.5,0\n+315,1.1\n+322.5,1.9\n+330,3.8\n+337.5,5\n+345,6.5\n+352.5,9.2\ndiff --git a/pvlib/tests/iotools/test_pvgis.py b/pvlib/tests/iotools/test_pvgis.py\n--- a/pvlib/tests/iotools/test_pvgis.py\n+++ b/pvlib/tests/iotools/test_pvgis.py\n@@ -9,8 +9,9 @@\n import requests\n from pvlib.iotools import get_pvgis_tmy, read_pvgis_tmy\n from pvlib.iotools import get_pvgis_hourly, read_pvgis_hourly\n+from pvlib.iotools import get_pvgis_horizon\n from ..conftest import (DATA_DIR, RERUNS, RERUNS_DELAY, assert_frame_equal,\n-                        fail_on_pvlib_version)\n+                        fail_on_pvlib_version, assert_series_equal)\n from pvlib._deprecation import pvlibDeprecationWarning\n \n \n@@ -509,6 +510,23 @@ def test_get_pvgis_map_variables(pvgis_tmy_mapped_columns):\n     assert all([c in pvgis_tmy_mapped_columns for c in actual.columns])\n \n \n+@pytest.mark.remote_data\n+@pytest.mark.flaky(reruns=RERUNS, reruns_delay=RERUNS_DELAY)\n+def test_read_pvgis_horizon():\n+    pvgis_data, _ = get_pvgis_horizon(35.171051, -106.465158)\n+    horizon_data = pd.read_csv(DATA_DIR / 'test_read_pvgis_horizon.csv',\n+                               index_col=0)\n+    horizon_data = horizon_data['horizon_elevation']\n+    assert_series_equal(pvgis_data, horizon_data)\n+\n+\n+@pytest.mark.remote_data\n+@pytest.mark.flaky(reruns=RERUNS, reruns_delay=RERUNS_DELAY)\n+def test_read_pvgis_horizon_invalid_coords():\n+    with pytest.raises(requests.HTTPError, match='lat: Incorrect value'):\n+        _, _ = get_pvgis_horizon(100, 50)  # unfeasible latitude\n+\n+\n def test_read_pvgis_tmy_map_variables(pvgis_tmy_mapped_columns):\n     fn = DATA_DIR / 'tmy_45.000_8.000_2005_2016.json'\n     actual, _, _, _ = read_pvgis_tmy(fn, map_variables=True)\n", "problem_statement": "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", "hints_text": "@mikofski @cwhanse I saw your discussions in #758 and #1290 and figured I'd share the code I had laying around for downloading the local horizon profile from SRTM. Does this have any interest to you?\nI'm lovin' this! Could we also look into retrieving pvgis horizon data, how do they compare to the SRTM from MINES?", "created_at": "2022-01-26T20:24:16Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly[testfile0-expected_radiation_csv-metadata_exp0-inputs_exp0-False-None]", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly[testfile1-expected_radiation_csv_mapped-metadata_exp1-inputs_exp1-True-csv]", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly[testfile2-expected_pv_json-metadata_exp2-inputs_exp2-False-None]", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly[testfile3-expected_pv_json_mapped-metadata_exp3-inputs_exp3-True-json]", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly_bad_extension", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly[testfile0-expected_radiation_csv-args0-False-https://re.jrc.ec.europa.eu/api/seriescalc?lat=45&lon=8&outputformat=csv&angle=30&aspect=0&usehorizon=0&pvtechchoice=crystSi&mountingplace=free&trackingtype=0&components=1&raddatabase=PVGIS-SARAH&startyear=2016&endyear=2016]", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly[testfile1-expected_radiation_csv_mapped-args1-True-https://re.jrc.ec.europa.eu/api/seriescalc?lat=45&lon=8&outputformat=csv&angle=30&aspect=0&usehorizon=0&pvtechchoice=crystSi&mountingplace=free&trackingtype=0&components=1&raddatabase=PVGIS-SARAH&startyear=2016&endyear=2016]", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly[testfile2-expected_pv_json-args2-False-https://re.jrc.ec.europa.eu/api/v5_2/seriescalc?lat=45&lon=8&outputformat=json&angle=30&aspect=0&pvtechchoice=CIS&mountingplace=free&trackingtype=2&components=0&usehorizon=1&raddatabase=PVGIS-SARAH2&startyear=2013&endyear=2014&pvcalculation=1&peakpower=10&loss=5&optimalangles=1]", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly[testfile3-expected_pv_json_mapped-args3-True-https://re.jrc.ec.europa.eu/api/v5_2/seriescalc?lat=45&lon=8&outputformat=json&angle=30&aspect=0&pvtechchoice=CIS&mountingplace=free&trackingtype=2&components=0&usehorizon=1&raddatabase=PVGIS-SARAH2&startyear=2013&endyear=2014&pvcalculation=1&peakpower=10&loss=5&optimalangles=1]", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly_bad_status_code", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly_bad_outputformat", "pvlib/tests/iotools/test_pvgis.py::test_get_pvgis_hourly_additional_inputs", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_hourly_empty_file", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_map_variables", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_json", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_epw", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_csv", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_basic", "pvlib/tests/iotools/test_pvgis.py::test_read_pvgis_tmy_exception"], "PASS_TO_PASS": [], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-1625", "base_commit": "14e1a23a3166b9a645a16de96f694c77a5d4abb7", "patch": "diff --git a/src/sqlfluff/rules/L031.py b/src/sqlfluff/rules/L031.py\n--- a/src/sqlfluff/rules/L031.py\n+++ b/src/sqlfluff/rules/L031.py\n@@ -211,7 +211,7 @@ def _lint_aliases_in_join(\n             violation_buff.append(\n                 LintResult(\n                     anchor=alias_info.alias_identifier_ref,\n-                    description=\"Avoid using aliases in join condition\",\n+                    description=\"Avoid aliases in from clauses and join conditions.\",\n                     fixes=fixes,\n                 )\n             )\n", "test_patch": "diff --git a/test/cli/commands_test.py b/test/cli/commands_test.py\n--- a/test/cli/commands_test.py\n+++ b/test/cli/commands_test.py\n@@ -49,7 +49,7 @@ def invoke_assert_code(\n expected_output = \"\"\"== [test/fixtures/linter/indentation_error_simple.sql] FAIL\n L:   2 | P:   4 | L003 | Indentation not hanging or a multiple of 4 spaces\n L:   5 | P:  10 | L010 | Keywords must be consistently upper case.\n-L:   5 | P:  13 | L031 | Avoid using aliases in join condition\n+L:   5 | P:  13 | L031 | Avoid aliases in from clauses and join conditions.\n \"\"\"\n \n \n", "problem_statement": "TSQL - L031 incorrectly triggers \"Avoid using aliases in join condition\" when no join present\n## Expected Behaviour\r\n\r\nBoth of these queries should pass, the only difference is the addition of a table alias 'a':\r\n\r\n1/ no alias\r\n\r\n```\r\nSELECT [hello]\r\nFROM\r\n    mytable\r\n```\r\n\r\n2/ same query with alias\r\n\r\n```\r\nSELECT a.[hello]\r\nFROM\r\n    mytable AS a\r\n```\r\n\r\n## Observed Behaviour\r\n\r\n1/ passes\r\n2/ fails with: L031: Avoid using aliases in join condition.\r\n\r\nBut there is no join condition :-)\r\n\r\n## Steps to Reproduce\r\n\r\nLint queries above\r\n\r\n## Dialect\r\n\r\nTSQL\r\n\r\n## Version\r\n\r\nsqlfluff 0.6.9\r\nPython 3.6.9\r\n\r\n## Configuration\r\n\r\nN/A\n", "hints_text": "Actually, re-reading the docs I think this is the intended behaviour... closing", "created_at": "2021-10-13T11:35:29Z", "version": "0.6", "FAIL_TO_PASS": ["test/cli/commands_test.py::test__cli__command_directed"], "PASS_TO_PASS": ["test/cli/commands_test.py::test__cli__command_dialect", "test/cli/commands_test.py::test__cli__command_dialect_legacy", "test/cli/commands_test.py::test__cli__command_lint_stdin[command0]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command1]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command2]", "test/cli/commands_test.py::test__cli__command_lint_stdin[command3]", "test/cli/commands_test.py::test__cli__command_lint_parse[command0]", "test/cli/commands_test.py::test__cli__command_lint_parse[command1]", "test/cli/commands_test.py::test__cli__command_lint_parse[command2]", "test/cli/commands_test.py::test__cli__command_lint_parse[command3]", "test/cli/commands_test.py::test__cli__command_lint_parse[command4]", "test/cli/commands_test.py::test__cli__command_lint_parse[command5]", "test/cli/commands_test.py::test__cli__command_lint_parse[command6]", "test/cli/commands_test.py::test__cli__command_lint_parse[command7]", "test/cli/commands_test.py::test__cli__command_lint_parse[command8]", "test/cli/commands_test.py::test__cli__command_lint_parse[command9]", "test/cli/commands_test.py::test__cli__command_lint_parse[command10]", "test/cli/commands_test.py::test__cli__command_lint_parse[command11]", "test/cli/commands_test.py::test__cli__command_lint_parse[command12]", "test/cli/commands_test.py::test__cli__command_lint_parse[command13]", "test/cli/commands_test.py::test__cli__command_lint_parse[command14]", "test/cli/commands_test.py::test__cli__command_lint_parse[command15]", "test/cli/commands_test.py::test__cli__command_lint_parse[command16]", "test/cli/commands_test.py::test__cli__command_lint_parse[command17]", "test/cli/commands_test.py::test__cli__command_lint_parse[command18]", "test/cli/commands_test.py::test__cli__command_lint_parse[command19]", "test/cli/commands_test.py::test__cli__command_lint_parse[command20]", "test/cli/commands_test.py::test__cli__command_lint_parse[command21]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command0-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command1-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command2-1]", "test/cli/commands_test.py::test__cli__command_lint_warning_explicit_file_ignored", "test/cli/commands_test.py::test__cli__command_lint_skip_ignore_files", "test/cli/commands_test.py::test__cli__command_versioning", "test/cli/commands_test.py::test__cli__command_version", "test/cli/commands_test.py::test__cli__command_rules", "test/cli/commands_test.py::test__cli__command_dialects", "test/cli/commands_test.py::test__cli__command__fix[L001-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/whitespace_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L008-test/fixtures/linter/indentation_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[L003-test/fixtures/linter/indentation_error_hard.sql]", "test/cli/commands_test.py::test__cli__command_fix_stdin[select", "test/cli/commands_test.py::test__cli__command_fix_stdin[", "test/cli/commands_test.py::test__cli__command_fix_stdin[SELECT", "test/cli/commands_test.py::test__cli__command_fix_stdin_logging_to_stderr", "test/cli/commands_test.py::test__cli__command_fix_stdin_safety", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[create", "test/cli/commands_test.py::test__cli__command_fix_stdin_error_exit_code[select", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-y-0-0]", "test/cli/commands_test.py::test__cli__command__fix_no_force[L001-test/fixtures/linter/indentation_errors.sql-n-65-1]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[yaml]", "test/cli/commands_test.py::test__cli__command_parse_serialize_from_stdin[json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[select", "test/cli/commands_test.py::test__cli__command_lint_serialize_from_stdin[SElect", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command0]", "test/cli/commands_test.py::test__cli__command_fail_nice_not_found[command1]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[yaml]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[json]", "test/cli/commands_test.py::test__cli__command_lint_serialize_multiple_files[github-annotation]", "test/cli/commands_test.py::test__cli__command_lint_serialize_github_annotation", "test/cli/commands_test.py::test___main___help", "test/cli/commands_test.py::test_encoding[utf-8-ascii]", "test/cli/commands_test.py::test_encoding[utf-8-sig-UTF-8-SIG]", "test/cli/commands_test.py::test_encoding[utf-32-UTF-32]"], "environment_setup_commit": "67023b85c41d23d6c6d69812a41b207c4f8a9331"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1589", "base_commit": "bd86597f62013f576670a452869ea88a47c58c01", "patch": "diff --git a/pvlib/bifacial/infinite_sheds.py b/pvlib/bifacial/infinite_sheds.py\n--- a/pvlib/bifacial/infinite_sheds.py\n+++ b/pvlib/bifacial/infinite_sheds.py\n@@ -216,8 +216,8 @@ def _ground_angle(x, surface_tilt, gcr):\n     #  :         \\  v      *-.\\\n     #  :          \\<-----P---->\\\n \n-    x1 = x * sind(surface_tilt)\n-    x2 = (x * cosd(surface_tilt) + 1 / gcr)\n+    x1 = gcr * x * sind(surface_tilt)\n+    x2 = gcr * x * cosd(surface_tilt) + 1\n     psi = np.arctan2(x1, x2)  # do this first because it handles 0 / 0\n     return np.rad2deg(psi)\n \ndiff --git a/pvlib/shading.py b/pvlib/shading.py\n--- a/pvlib/shading.py\n+++ b/pvlib/shading.py\n@@ -52,8 +52,8 @@ def masking_angle(surface_tilt, gcr, slant_height):\n     # The original equation (8 in [1]) requires pitch and collector width,\n     # but it's easy to non-dimensionalize it to make it a function of GCR\n     # by factoring out B from the argument to arctan.\n-    numerator = (1 - slant_height) * sind(surface_tilt)\n-    denominator = 1/gcr - (1 - slant_height) * cosd(surface_tilt)\n+    numerator = gcr * (1 - slant_height) * sind(surface_tilt)\n+    denominator = 1 - gcr * (1 - slant_height) * cosd(surface_tilt)\n     phi = np.arctan(numerator / denominator)\n     return np.degrees(phi)\n \n", "test_patch": "diff --git a/pvlib/tests/bifacial/test_infinite_sheds.py b/pvlib/tests/bifacial/test_infinite_sheds.py\n--- a/pvlib/tests/bifacial/test_infinite_sheds.py\n+++ b/pvlib/tests/bifacial/test_infinite_sheds.py\n@@ -106,6 +106,14 @@ def test__ground_angle(test_system):\n     assert np.allclose(angles, expected_angles)\n \n \n+def test__ground_angle_zero_gcr():\n+    surface_tilt = 30.0\n+    x = np.array([0.0, 0.5, 1.0])\n+    angles = infinite_sheds._ground_angle(x, surface_tilt, 0)\n+    expected_angles = np.array([0, 0, 0])\n+    assert np.allclose(angles, expected_angles)\n+\n+\n def test__vf_row_ground(test_system):\n     ts, _, _ = test_system\n     x = np.array([0., 0.5, 1.0])\ndiff --git a/pvlib/tests/test_shading.py b/pvlib/tests/test_shading.py\n--- a/pvlib/tests/test_shading.py\n+++ b/pvlib/tests/test_shading.py\n@@ -45,6 +45,13 @@ def test_masking_angle_scalar(surface_tilt, masking_angle):\n         assert np.isclose(masking_angle_actual, angle)\n \n \n+def test_masking_angle_zero_gcr(surface_tilt):\n+    # scalar inputs and outputs, including zero\n+    for tilt in surface_tilt:\n+        masking_angle_actual = shading.masking_angle(tilt, 0, 0.25)\n+        assert np.isclose(masking_angle_actual, 0)\n+\n+\n def test_masking_angle_passias_series(surface_tilt, average_masking_angle):\n     # pandas series inputs and outputs\n     masking_angle_actual = shading.masking_angle_passias(surface_tilt, 0.5)\n", "problem_statement": "ZeroDivisionError when gcr is zero\n**Describe the bug**\r\n\r\nThough maybe not intuitive, setting ground coverage ratio to zero is useful when a plant consists of a single shed, e.g. calculating the irradiance on the backside of the panels. However, e.g., `bifacial.infinite_sheds.get_irradiance_poa` fails with `ZeroDivisionError` whenever `gcr=0`.\r\n\r\n**To Reproduce**\r\n\r\n```python\r\nfrom pvlib.bifacial.infinite_sheds import get_irradiance_poa\r\n\r\nget_irradiance_poa(surface_tilt=160, surface_azimuth=180, solar_zenith=20, solar_azimuth=180, gcr=0, height=1, pitch=1000, ghi=200, dhi=200, dni=0, albedo=0.2)\r\n```\r\nreturns:\r\n```\r\nTraceback (most recent call last):\r\n  File \"C:\\Python\\Python310\\lib\\site-packages\\IPython\\core\\interactiveshell.py\", line 3398, in run_code\r\n    exec(code_obj, self.user_global_ns, self.user_ns)\r\n  File \"<ipython-input-7-0cb583b2b311>\", line 3, in <cell line: 3>\r\n    get_irradiance_poa(surface_tilt=160, surface_azimuth=180, solar_zenith=20, solar_azimuth=180, gcr=0, height=1, pitch=1, ghi=200, dhi=200, dni=0, albedo=0.2)\r\n  File \"C:\\Python\\Python310\\lib\\site-packages\\pvlib\\bifacial\\infinite_sheds.py\", line 522, in get_irradiance_poa\r\n    vf_shade_sky, vf_noshade_sky = _vf_row_sky_integ(\r\n  File \"C:\\Python\\Python310\\lib\\site-packages\\pvlib\\bifacial\\infinite_sheds.py\", line 145, in _vf_row_sky_integ\r\n    psi_t_shaded = masking_angle(surface_tilt, gcr, x)\r\n  File \"C:\\Python\\Python310\\lib\\site-packages\\pvlib\\shading.py\", line 56, in masking_angle\r\n    denominator = 1/gcr - (1 - slant_height) * cosd(surface_tilt)\r\nZeroDivisionError: division by zero\r\n```\r\n\r\n**Expected behavior**\r\n\r\nOne can easily solve this `ZeroDivisionError` by multiplying both numerator and denominator with `gcr` inside `shading.masking_angle` and the same inside `bifacial.infinite_sheds._ground_angle`.\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: '0.9.3'\r\n - ``pandas.__version__``: '1.4.4'\r\n - python: '3.10.4'\r\n\n", "hints_text": "@kdebrab thanks for investigating this case, which wasn't consider in the implementation. @pvlib/pvlib-core I think we should fix this. \r\n\r\nFor consistency we should also extend `shading.masking_angle_passias` for the limiting case of `gcr=0`. That may be more complicated and could be done in a second PR.\r\n\nI think `infinite_sheds.get_irradiance_poa(...)` converges to `irradiance.get_total_irradiance(..., model='isotropic')` as gcr approaches zero, so that's an option for modeling this situation in the meantime.  \nProbably close enough to be useful, but in theory the rear irradiance would be slightly less than GHI * 1/2 (1 - cos(tilt)), due to the row's shadow.\nWith a more realistic model (e.g. pvfactors) that considers shadows individually that's true, but `infinite_sheds` models ground-reflected irradiance using integrated averages which dilute the effect of the row's own shadow to nothing as gcr approaches zero.  By decreasing `gcr` and increasing `npoints` you can get `infinite_sheds` as close to `get_total_irradiance` as you like:\r\n\r\n![image](https://user-images.githubusercontent.com/57452607/198049857-0c844116-3bc0-48dd-b889-8b658f39b4a0.png)\r\n\r\n<details>\r\n  <summary>Source</summary>\r\n\r\n```python\r\nimport pvlib\r\nimport pandas as pd\r\nimport numpy as np\r\nimport matplotlib.pyplot as plt\r\n\r\ntimes = pd.date_range('2019-06-01', '2019-06-02', freq='T', tz='Etc/GMT+5')\r\nlocation = pvlib.location.Location(40, -80)\r\nsp = location.get_solarposition(times)\r\ncs = location.get_clearsky(times, solar_position=sp)\r\n\r\nkwargs = dict(\r\n    surface_tilt=20, surface_azimuth=180,\r\n    solar_zenith=sp.apparent_zenith, solar_azimuth=sp.azimuth,\r\n    ghi=cs.ghi, dhi=cs.dhi, dni=cs.dni, albedo=0.2\r\n)\r\n\r\nlimit = pvlib.irradiance.get_total_irradiance(**kwargs, model='isotropic')['poa_global']\r\n\r\nall_stats = []\r\n\r\nfor gcr in [0.3, 0.1, 1e-2, 1e-3, 1e-4, 1e-5, 1e-6, 1e-7]:\r\n    stats = dict(gcr=gcr)\r\n    for npoints in [100, 1000, 10000]:\r\n        out = pvlib.bifacial.infinite_sheds.get_irradiance_poa(gcr=gcr, height=1, pitch=1/gcr, npoints=npoints, **kwargs)\r\n        stats[f'npoints={npoints}'] = np.max(np.abs(out['poa_global'] - limit))\r\n\r\n    all_stats.append(stats)\r\n\r\npd.DataFrame(all_stats).set_index('gcr').plot(logx=True, logy=True)\r\nplt.ylabel('Max Absolute Difference [W/m2]')\r\n\r\n```\r\n</details>\n@kanderso-nrel Thanks for pointing this out!\r\n\r\nI think for a `surface_tilt` equal to 20\u00b0 (front side of the panel) this result would be fine, as there is no shadow on the front surface from the own row. However, changing `surface_tilt` to 160\u00b0 (back side of the panel) yields basically the same figure and that is indeed not ideal (and unexpected for me).\r\n\r\nIf I understand well, this is due to the calculation of the ground-reflected irradiance, where the infinite_sheds method uses (1) the **average** 'view factor from the ground to the sky' of the ground between the rows, and (2) the **average** 'fraction of ground between rows that is unshaded', both of which approach 1 as the pitch approaches infinity.\r\n\r\nI think an improved method would not take the average, but some kind of weighted average, considering that the reflection from the ground right in front of the surface is more important than the reflection from the ground further away from the surface. In that case, I would assume that the effect of the row's own shadow would no longer dilute when gcr approaches zero.\n> However, changing surface_tilt to 160\u00b0 (back side of the panel) yields basically the same figure and that is indeed not ideal (and unexpected for me).\r\n\r\nOops, I think I had originally started with rear-side, switched to front-side just to check it as well, and forgot to switch back to rear before posting here.  Good catch.\r\n\r\nFYI pvfactors (`pvlib.bifacial.pvfactors.pvfactors_timeseries`) can model irradiance for single rows (`n_pvrows=1, index_observed_pvrow=0`) and does not suffer from the averaging issues you mention. ", "created_at": "2022-11-11T14:19:29Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/bifacial/test_infinite_sheds.py::test__ground_angle_zero_gcr", "pvlib/tests/test_shading.py::test_masking_angle_zero_gcr"], "PASS_TO_PASS": ["pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_ground_sky_integ", "pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_row_sky_integ", "pvlib/tests/bifacial/test_infinite_sheds.py::test__poa_sky_diffuse_pv", "pvlib/tests/bifacial/test_infinite_sheds.py::test__ground_angle", "pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_row_ground", "pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_row_ground_integ", "pvlib/tests/bifacial/test_infinite_sheds.py::test__poa_ground_shadows", "pvlib/tests/bifacial/test_infinite_sheds.py::test__shaded_fraction_floats", "pvlib/tests/bifacial/test_infinite_sheds.py::test__shaded_fraction_array", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance_poa", "pvlib/tests/bifacial/test_infinite_sheds.py::test__backside_tilt", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance_limiting_gcr", "pvlib/tests/test_shading.py::test_masking_angle_series", "pvlib/tests/test_shading.py::test_masking_angle_scalar", "pvlib/tests/test_shading.py::test_masking_angle_passias_series", "pvlib/tests/test_shading.py::test_masking_angle_passias_scalar", "pvlib/tests/test_shading.py::test_sky_diffuse_passias_series", "pvlib/tests/test_shading.py::test_sky_diffuse_passias_scalar"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-958", "base_commit": "40652fc0a18fd9f1204cb3d4b9829e3a8be5cbe0", "patch": "diff --git a/pydicom/charset.py b/pydicom/charset.py\n--- a/pydicom/charset.py\n+++ b/pydicom/charset.py\n@@ -45,7 +45,7 @@\n     'ISO 2022 IR 144': 'iso_ir_144',\n     'ISO 2022 IR 148': 'iso_ir_148',\n     'ISO 2022 IR 149': 'euc_kr',\n-    'ISO 2022 IR 159': 'iso-2022-jp',\n+    'ISO 2022 IR 159': 'iso2022_jp_2',\n     'ISO 2022 IR 166': 'iso_ir_166',\n     'ISO 2022 IR 58': 'iso_ir_58',\n     'ISO_IR 192': 'UTF8',  # from Chinese example, 2008 PS3.5 Annex J p1-4\n@@ -81,7 +81,7 @@\n     ESC + b'-M': 'iso_ir_148',\n     ESC + b'-T': 'iso_ir_166',\n     ESC + b'$)C': 'euc_kr',\n-    ESC + b'$(D': 'iso-2022-jp',\n+    ESC + b'$(D': 'iso2022_jp_2',\n     ESC + b'$)A': 'iso_ir_58',\n }\n \n@@ -92,7 +92,7 @@\n # To decode them, the escape sequence shall be preserved in the input byte\n # string, and will be removed during decoding by Python.\n handled_encodings = ('iso2022_jp',\n-                     'iso-2022-jp',\n+                     'iso2022_jp_2',\n                      'iso_ir_58')\n \n \n@@ -165,10 +165,18 @@ def _encode_to_jis_x_0201(value, errors='strict'):\n \n     return encoded\n \n-\n def _encode_to_jis_x_0208(value, errors='strict'):\n-    \"\"\"Convert a unicode string into JIS X 0208 byte string using iso2022_jp\n-    encodings.\n+    \"\"\"Convert a unicode string into JIS X 0208 byte string.\"\"\"\n+    return _encode_to_given_charset(value, 'ISO 2022 IR 87', errors=errors)\n+\n+\n+def _encode_to_jis_x_0212(value, errors='strict'):\n+    \"\"\"Convert a unicode string into JIS X 0212 byte string.\"\"\"\n+    return _encode_to_given_charset(value, 'ISO 2022 IR 159', errors=errors)\n+\n+\n+def _encode_to_given_charset(value, character_set, errors='strict'):\n+    \"\"\"Convert a unicode string into given character set.\n     The escape sequence which is located at the end of the encoded value has\n     to vary depending on the value 1 of SpecificCharacterSet. So we have to\n     trim it and append the correct escape sequence manually.\n@@ -177,6 +185,8 @@ def _encode_to_jis_x_0208(value, errors='strict'):\n     ----------\n     value : text type\n         The unicode string as presented to the user.\n+    character_set: str:\n+        Character set for result.\n     errors : str\n         The behavior of a character which could not be encoded. This value\n         is passed to errors argument of str.encode().\n@@ -185,28 +195,30 @@ def _encode_to_jis_x_0208(value, errors='strict'):\n     -------\n     byte string\n         The encoded string. If some characters in value could not be encoded to\n-        JIS X 0208, it depends on the behavior of iso2022_jp encoder.\n+        given character_set, it depends on the behavior of corresponding python\n+        encoder.\n \n     Raises\n     ------\n     UnicodeEncodeError\n         If errors is set to 'strict' and `value` could not be encoded with\n-        JIS X 0208.\n+        given character_set.\n     \"\"\"\n \n+    encoding = python_encoding[character_set]\n     # If errors is not strict, this function is used as fallback.\n     # So keep the tail escape sequence of encoded for backward compatibility.\n     if errors != 'strict':\n-        return value.encode('iso2022_jp', errors=errors)\n+        return value.encode(encoding, errors=errors)\n \n-    Encoder = codecs.getincrementalencoder('iso2022-jp')\n+    Encoder = codecs.getincrementalencoder(encoding)\n     encoder = Encoder()\n \n     encoded = encoder.encode(value[0])\n-    if encoded[:3] != ENCODINGS_TO_CODES['iso2022_jp']:\n+    if not encoded.startswith(ENCODINGS_TO_CODES[encoding]):\n         raise UnicodeEncodeError(\n-            'iso2022_jp', value, 0, len(value),\n-            'Given character is out of ISO IR 87')\n+            encoding, value, 0, len(value),\n+            'Given character is out of {}'.format(character_set))\n \n     for i, c in enumerate(value[1:], 1):\n         try:\n@@ -215,10 +227,10 @@ def _encode_to_jis_x_0208(value, errors='strict'):\n             e.start = i\n             e.end = len(value)\n             raise e\n-        if b[:3] == ENCODINGS_TO_CODES['iso8859']:\n+        if b[:1] == ESC:\n             raise UnicodeEncodeError(\n-                'iso2022_jp', value, i, len(value),\n-                'Given character is out of ISO IR 87')\n+                encoding, value, i, len(value),\n+                'Given character is out of {}'.format(character_set))\n         encoded += b\n     return encoded\n \n@@ -262,13 +274,13 @@ def _get_escape_sequence_for_encoding(encoding, encoded=None):\n \n \n # These encodings need escape sequence to handle alphanumeric characters.\n-need_tail_escape_sequence_encodings = ('iso2022_jp', 'iso-2022-jp')\n+need_tail_escape_sequence_encodings = ('iso2022_jp', 'iso2022_jp_2')\n \n \n custom_encoders = {\n     'shift_jis': _encode_to_jis_x_0201,\n     'iso2022_jp': _encode_to_jis_x_0208,\n-    'iso-2022-jp': _encode_to_jis_x_0208\n+    'iso2022_jp_2': _encode_to_jis_x_0212\n }\n \n \n@@ -562,7 +574,7 @@ def _encode_string_parts(value, encodings):\n     # unencoded_part is empty - we are done, return the encoded string\n     if best_encoding in need_tail_escape_sequence_encodings:\n         encoded += _get_escape_sequence_for_encoding(encodings[0])\n-    return encoded\n+    return bytes(encoded)\n \n \n def _encode_string_impl(value, encoding, errors='strict'):\n", "test_patch": "diff --git a/pydicom/tests/test_charset.py b/pydicom/tests/test_charset.py\n--- a/pydicom/tests/test_charset.py\n+++ b/pydicom/tests/test_charset.py\n@@ -9,6 +9,7 @@\n from pydicom.data import get_charset_files, get_testdata_files\n from pydicom.dataelem import DataElement\n from pydicom.filebase import DicomBytesIO\n+from pydicom.valuerep import PersonName3\n \n # The file names (without '.dcm' extension) of most of the character test\n # files, together with the respective decoded PatientName tag values.\n@@ -197,9 +198,9 @@ def test_bad_decoded_multi_byte_encoding(self):\n                            b'\\x1b$(D\\xc4\\xe9\\xef\\xed\\xf5\\xf3\\xe9\\xef\\xf2')\n \n         with pytest.warns(UserWarning, match='Failed to decode byte string '\n-                                             'with encodings: iso-2022-jp'):\n+                                             'with encodings: iso2022_jp_2'):\n             pydicom.charset.decode_element(elem, ['ISO 2022 IR 159'])\n-            assert u'\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd' == elem.value\n+            assert u'\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd\ufffd' == elem.value\n \n     def test_bad_decoded_multi_byte_encoding_enforce_standard(self):\n         \"\"\"Test handling bad encoding for single encoding if\n@@ -207,7 +208,7 @@ def test_bad_decoded_multi_byte_encoding_enforce_standard(self):\n         config.enforce_valid_values = True\n         elem = DataElement(0x00100010, 'PN',\n                            b'\\x1b$(D\\xc4\\xe9\\xef\\xed\\xf5\\xf3\\xe9\\xef\\xf2')\n-        msg = (\"'iso2022_jp' codec can't decode bytes in position 0-3: \"\n+        msg = (\"'iso2022_jp_2' codec can't decode byte 0xc4 in position 4: \"\n                \"illegal multibyte sequence\")\n         with pytest.raises(UnicodeDecodeError, match=msg):\n             pydicom.charset.decode_element(elem, ['ISO 2022 IR 159'])\n@@ -435,11 +436,26 @@ def test_japanese_multi_byte_personname(self):\n             ds_out = dcmread(fp)\n             assert original_string == ds_out.PatientName.original_string\n \n+        japanese_pn = PersonName3(u\"Mori^Ogai=\u68ee^\u9dd7\u5916=\u3082\u308a^\u304a\u3046\u304c\u3044\")\n+        pyencs = pydicom.charset.convert_encodings([\"ISO 2022 IR 6\",\n+                                                    \"ISO 2022 IR 87\",\n+                                                    \"ISO 2022 IR 159\"])\n+        actual_encoded = bytes(japanese_pn.encode(pyencs))\n+        expect_encoded = (\n+            b\"\\x4d\\x6f\\x72\\x69\\x5e\\x4f\\x67\\x61\\x69\\x3d\\x1b\\x24\\x42\\x3f\"\n+            b\"\\x39\\x1b\\x28\\x42\\x5e\\x1b\\x24\\x28\\x44\\x6c\\x3f\\x1b\\x24\\x42\"\n+            b\"\\x33\\x30\\x1b\\x28\\x42\\x3d\\x1b\\x24\\x42\\x24\\x62\\x24\\x6a\\x1b\"\n+            b\"\\x28\\x42\\x5e\\x1b\\x24\\x42\\x24\\x2a\\x24\\x26\\x24\\x2c\\x24\\x24\"\n+            b\"\\x1b\\x28\\x42\"\n+        )\n+        assert expect_encoded == actual_encoded\n+\n     def test_japanese_multi_byte_encoding(self):\n         \"\"\"Test japanese multi byte strings are correctly encoded.\"\"\"\n-        encoded = pydicom.charset.encode_string(u'\u3042a\uff71\u30a2',\n-                                                ['shift_jis', 'iso2022_jp'])\n-        assert b'\\x1b$B$\"\\x1b(Ja\\x1b)I\\xb1\\x1b$B%\"\\x1b(J' == encoded\n+        encoded = pydicom.charset.encode_string(u'\u3042a\uff71\u30a2\u9f69', ['shift_jis',\n+                                                'iso2022_jp', 'iso2022_jp_2'])\n+        expect = b'\\x1b$B$\"\\x1b(Ja\\x1b)I\\xb1\\x1b$B%\"\\x1b$(DmN\\x1b(J'\n+        assert expect == bytes(encoded)\n \n     def test_bad_japanese_encoding(self):\n         \"\"\"Test japanese multi byte strings are not correctly encoded.\"\"\"\n", "problem_statement": "Encoding to ISO 2022 IR 159 doesn't work\n<!-- Instructions For Filing a Bug: https://github.com/pydicom/pydicom/blob/master/CONTRIBUTING.md#filing-bugs -->\r\n\r\n#### Description\r\nEncoding to ISO 2022 IR 159 doesn't work even if 'ISO 2022 IR 159' is passed to pydicom.charset.convert_encodings.\r\n\r\n#### Steps/Code to Reproduce\r\nISO 2022 IR 159 is designed as supplement characters to ISO 2022 IR 87. So these characters are not frequent use. But person name sometimes contains them. In the following example, the letter of \"\u9dd7\" is only in ISO 2022 IR 159. But we cannot encode them correctly. \r\n\r\n```\r\nimport pydicom\r\n\r\njapanese_pn = u\"Mori^Ogai=\u68ee^\u9dd7\u5916=\u3082\u308a^\u304a\u3046\u304c\u3044\"\r\nspecific_character_sets = [\"ISO 2022 IR 6\", \"ISO 2022 IR 87\", \"ISO 2022 IR 159\"]\r\nexpect_encoded = (\r\n    b\"\\x4d\\x6f\\x72\\x69\\x5e\\x4f\\x67\\x61\\x69\\x3d\\x1b\\x24\\x42\\x3f\"\r\n    b\"\\x39\\x1b\\x28\\x42\\x5e\\x1b\\x24\\x28\\x44\\x6c\\x3f\\x1b\\x24\\x42\"\r\n    b\"\\x33\\x30\\x1b\\x28\\x42\\x3d\\x1b\\x24\\x42\\x24\\x62\\x24\\x6a\\x1b\"\r\n    b\"\\x28\\x42\\x5e\\x1b\\x24\\x42\\x24\\x2a\\x24\\x26\\x24\\x2c\\x24\\x24\"\r\n    b\"\\x1b\\x28\\x42\"\r\n)\r\n\r\npython_encodings = pydicom.charset.convert_encodings(specific_character_sets)\r\nactual_encoded = pydicom.charset.encode_string(japanese_pn, python_encodings)\r\n\r\nprint(\"actual:{}\".format(actual_encoded))\r\nprint(\"expect:{}\".format(expect_encoded))\r\n```\r\n#### Expected Results\r\n<!-- Please paste or describe the expected results.\r\nExample: No error is thrown and the name of the patient is printed.-->\r\n```\r\nb'Mori^Ogai=\\x1b$B?9\\x1b(B^\\x1b$(Dl?\\x1b$B30\\x1b(B=\\x1b$B$b$j\\x1b(B^\\x1b$B$*$&$,$$\\x1b(B'\r\n```\r\n#### Actual Results\r\n<!-- Please paste or specifically describe the actual output or traceback.\r\n(Use %xmode to deactivate ipython's trace beautifier)\r\nExample: ```AttributeError: 'FileDataset' object has no attribute 'PatientID'```\r\n-->\r\n```\r\nb'Mori^Ogai=?^??=??^????'\r\n```\r\n\r\nAnd the followin exception occurs.\r\n\r\n```\r\n/PATH/TO/MY/PYTHON/PACKAGES/pydicom/charset.py:488: UserWarning: Failed to encode value with encodings: iso8859, iso2022_jp, iso-2022-jp - using replacement characters in encoded string\r\n  .format(', '.join(encodings)))\r\n```\r\n\r\n#### Versions\r\n<!--\r\nPlease run the following snippet and paste the output below.\r\nimport platform; print(platform.platform())\r\nimport sys; print(\"Python\", sys.version)\r\nimport pydicom; print(\"pydicom\", pydicom.__version__)\r\n-->\r\n```\r\nLinux-4.15.0-55-generic-x86_64-with-debian-buster-sid\r\nPython 3.6.9 |Anaconda, Inc.| (default, Jul 30 2019, 19:07:31)\r\n[GCC 7.3.0]\r\npydicom 1.3.0\r\n```\r\n\r\n\r\n<!-- Thanks for contributing! -->\r\n\n", "hints_text": "In my opinion, this is caused by selecting the wrong python codec corresponding to 'ISO 2022 IR 159'. In the current implementation,  'iso-2022-jp' is used if 'ISO 2022 IR 159' is passed. But 'iso-2022-jp' is alias to 'iso200_jp'. I think we have to use 'iso-2022-jp-2'. It contains all 'iso-2022-jp' characters and 'ISO 2022 IR 159' characters.\r\n\r\nIf you don't mind, please assign this issue to me. I will make a PR for this issue.\nSure, go ahead! And thanks for the support!", "created_at": "2019-10-16T13:06:11Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_personname", "pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_encoding"], "PASS_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_encodings", "pydicom/tests/test_charset.py::TestCharset::test_nested_character_sets", "pydicom/tests/test_charset.py::TestCharset::test_inherited_character_set_in_sequence", "pydicom/tests/test_charset.py::TestCharset::test_standard_file", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set_enforce_valid", "pydicom/tests/test_charset.py::TestCharset::test_decoding_with_specific_tags", "pydicom/tests/test_charset.py::TestCharset::test_bad_charset", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_code_extensions_not_allowed", "pydicom/tests/test_charset.py::TestCharset::test_convert_encodings_warnings", "pydicom/tests/test_charset.py::TestCharset::test_convert_python_encodings", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_patched_charset", "pydicom/tests/test_charset.py::TestCharset::test_patched_code_extension_charset", "pydicom/tests/test_charset.py::TestCharset::test_multi_charset_default_value", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_personname", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_text", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_code_extensions[ISO", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrArab-\\u0642\\u0628\\u0627\\u0646\\u064a^\\u0644\\u0646\\u0632\\u0627\\u0631]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFren-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFrenMulti-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGerm-\\xc4neas^R\\xfcdiger]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGreek-\\u0394\\u03b9\\u03bf\\u03bd\\u03c5\\u03c3\\u03b9\\u03bf\\u03c2]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH31-Yamada^Tarou=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH32-\\uff94\\uff8f\\uff80\\uff9e^\\uff80\\uff9b\\uff73=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrHbrw-\\u05e9\\u05e8\\u05d5\\u05df^\\u05d3\\u05d1\\u05d5\\u05e8\\u05d4]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrI2-Hong^Gildong=\\u6d2a^\\u5409\\u6d1e=\\ud64d^\\uae38\\ub3d9]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMulti-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMultiExplicitIR6-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrKoreanMulti-\\uae40\\ud76c\\uc911]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrRuss-\\u041b\\u044e\\u043ace\\u043c\\u0431yp\\u0433]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX1-Wang^XiaoDong=\\u738b^\\u5c0f\\u6771]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX2-Wang^XiaoDong=\\u738b^\\u5c0f\\u4e1c]", "pydicom/tests/test_charset.py::TestCharset::test_changed_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_second_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_japanese_encoding", "pydicom/tests/test_charset.py::TestCharset::test_deprecated_decode"], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1192", "base_commit": "1f099ae0f75f0e2ed402a21702e584aac54a30ef", "patch": "diff --git a/pydicom/charset.py b/pydicom/charset.py\n--- a/pydicom/charset.py\n+++ b/pydicom/charset.py\n@@ -634,13 +634,16 @@ def convert_encodings(encodings):\n         ``True``.\n     \"\"\"\n \n-    # If a list if passed, we don't want to modify the list in place so copy it\n-    encodings = encodings[:]\n+    encodings = encodings or ['']\n \n     if isinstance(encodings, str):\n         encodings = [encodings]\n-    elif not encodings[0]:\n-        encodings[0] = 'ISO_IR 6'\n+    else:\n+        # If a list if passed, we don't want to modify the list\n+        # in place so copy it\n+        encodings = encodings[:]\n+        if not encodings[0]:\n+            encodings[0] = 'ISO_IR 6'\n \n     py_encodings = []\n     for encoding in encodings:\n", "test_patch": "diff --git a/pydicom/tests/test_charset.py b/pydicom/tests/test_charset.py\n--- a/pydicom/tests/test_charset.py\n+++ b/pydicom/tests/test_charset.py\n@@ -140,6 +140,15 @@ def test_bad_charset(self):\n         pydicom.charset.decode_element(elem, [])\n         assert 'iso8859' in elem.value.encodings\n \n+    def test_empty_charset(self):\n+        \"\"\"Empty charset defaults to ISO IR 6\"\"\"\n+        elem = DataElement(0x00100010, 'PN', 'CITIZEN')\n+        pydicom.charset.decode_element(elem, [''])\n+        assert ('iso8859',) == elem.value.encodings\n+        elem = DataElement(0x00100010, 'PN', 'CITIZEN')\n+        pydicom.charset.decode_element(elem, None)\n+        assert ('iso8859',) == elem.value.encodings\n+\n     def test_bad_encoded_single_encoding(self, allow_invalid_values):\n         \"\"\"Test handling bad encoding for single encoding\"\"\"\n         elem = DataElement(0x00100010, 'PN',\n@@ -189,6 +198,15 @@ def test_convert_python_encodings(self):\n         encodings = ['iso_ir_126', 'iso_ir_144']\n         assert encodings == pydicom.charset.convert_encodings(encodings)\n \n+    def test_convert_empty_encoding(self):\n+        \"\"\"Test that empty encodings are handled as default encoding\"\"\"\n+        encodings = ''\n+        assert ['iso8859'] == pydicom.charset.convert_encodings(encodings)\n+        encodings = ['']\n+        assert ['iso8859'] == pydicom.charset.convert_encodings(encodings)\n+        encodings = None\n+        assert ['iso8859'] == pydicom.charset.convert_encodings(encodings)\n+\n     def test_bad_decoded_multi_byte_encoding(self, allow_invalid_values):\n         \"\"\"Test handling bad encoding for single encoding\"\"\"\n         elem = DataElement(0x00100010, 'PN',\n", "problem_statement": "\"TypeError: 'NoneType' object is not subscriptable\" when reading dcm file with empty string as Chartset and \"use_none_as_empty_text_VR_value=True\"\n**Describe the bug**\r\nOnce thing I noticed is that `convert_encodings` in `charset.py` expects a list of encodings (according to the docstrings) from tag `0008,0005` but it can be just a value. \r\n\r\nThe problem is when reading Dicom files in production environments I noticed that some devices that are capturing the DICOMs are not very DICOM Compliant and is sending empty string , which it should be allowed as `0008,0005` is a 1C type, which means that if present it should have a valid value. \r\n\r\nI enabled `use_none_as_empty_text_VR_value` to make sure other tags whose value should be float or int have None instead of empty string, but if `0008,0005` value is empty string is switched to None and `convert_encodings` fails with `TypeError: 'NoneType' object is not subscriptable`\r\n\r\n**Expected behavior**\r\nThe expected behavior should be that if empty string or not present it should default to:\r\n```\r\n# default encoding if no encoding defined - corresponds to ISO IR 6 / ASCII\r\ndefault_encoding = \"iso8859\"\r\n```\r\n\r\n**Steps To Reproduce**\r\n\r\nout.dcm file if provided for testing with mock data but `Specific Character Set` set to empty string\r\n\r\nIf setting the `(0008, 0005) Specific Character Set` to empty string and setting `pydicom.config.use_none_as_empty_text_VR_value = True`\r\n\r\n```\r\n>>> import pydicom\r\n>>> pydicom.config.datetime_conversion = True\r\n>>> pydicom.config.allow_DS_float = True\r\n>>> pydicom.config.use_none_as_empty_text_VR_value = True\r\n>>> dataset = pydicom.dcmread(\"test.dcm\")\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/Users/bernardo/.virtualenvs/backend-api/lib/python3.7/site-packages/pydicom/filereader.py\", line 871, in dcmread\r\n    force=force, specific_tags=specific_tags)\r\n  File \"/Users/bernardo/.virtualenvs/backend-api/lib/python3.7/site-packages/pydicom/filereader.py\", line 744, in read_partial\r\n    specific_tags=specific_tags)\r\n  File \"/Users/bernardo/.virtualenvs/backend-api/lib/python3.7/site-packages/pydicom/filereader.py\", line 383, in read_dataset\r\n    encoding = convert_encodings(char_set)\r\n  File \"/Users/bernardo/.virtualenvs/backend-api/lib/python3.7/site-packages/pydicom/charset.py\", line 638, in convert_encodings\r\n    encodings = encodings[:]\r\nTypeError: 'NoneType' object is not subscriptable\r\n>>> pydicom.config.use_none_as_empty_text_VR_value = False\r\n>>> dataset = pydicom.dcmread(\"test.dcm\")\r\n```\r\n`(0008, 0005) Specific Character Set              CS: ''`\r\n\r\n**Your environment**\r\n\r\n```bash\r\npython -m pydicom.env_info\r\nmodule       | version\r\n------       | -------\r\nplatform     | Darwin-19.6.0-x86_64-i386-64bit\r\nPython       | 3.7.6 (default, Dec 30 2019, 19:38:26)  [Clang 11.0.0 (clang-1100.0.33.16)]\r\npydicom      | 2.0.0\r\ngdcm         | _module not found_\r\njpeg_ls      | _module not found_\r\nnumpy        | _module not found_\r\nPIL          | 7.0.0\r\n```\r\n\r\n\r\n[out.dcm.zip](https://github.com/pydicom/pydicom/files/5248618/out.dcm.zip)\r\n\n", "hints_text": "As you wrote, an empty string is not allowed as Specific Character Set, but I agree that this is a case that we shall handle gracefully (e.g. just handle as if the tag were absent). I will have a look.", "created_at": "2020-09-19T12:22:29Z", "version": "2.0", "FAIL_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_convert_empty_encoding"], "PASS_TO_PASS": ["pydicom/tests/test_charset.py::TestCharset::test_encodings", "pydicom/tests/test_charset.py::TestCharset::test_nested_character_sets", "pydicom/tests/test_charset.py::TestCharset::test_inherited_character_set_in_sequence", "pydicom/tests/test_charset.py::TestCharset::test_standard_file", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_character_set_enforce_valid", "pydicom/tests/test_charset.py::TestCharset::test_decoding_with_specific_tags", "pydicom/tests/test_charset.py::TestCharset::test_bad_charset", "pydicom/tests/test_charset.py::TestCharset::test_empty_charset", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_encoded_single_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_code_extensions_not_allowed", "pydicom/tests/test_charset.py::TestCharset::test_convert_encodings_warnings", "pydicom/tests/test_charset.py::TestCharset::test_convert_python_encodings", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_decoded_multi_byte_encoding_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence", "pydicom/tests/test_charset.py::TestCharset::test_unknown_escape_sequence_enforce_standard", "pydicom/tests/test_charset.py::TestCharset::test_patched_charset", "pydicom/tests/test_charset.py::TestCharset::test_patched_code_extension_charset", "pydicom/tests/test_charset.py::TestCharset::test_multi_charset_default_value", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_personname", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_multi_charset_text", "pydicom/tests/test_charset.py::TestCharset::test_single_byte_code_extensions[ISO", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrArab-\\u0642\\u0628\\u0627\\u0646\\u064a^\\u0644\\u0646\\u0632\\u0627\\u0631]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFren-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrFrenMulti-Buc^J\\xe9r\\xf4me]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGerm-\\xc4neas^R\\xfcdiger]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrGreek-\\u0394\\u03b9\\u03bf\\u03bd\\u03c5\\u03c3\\u03b9\\u03bf\\u03c2]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH31-Yamada^Tarou=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrH32-\\uff94\\uff8f\\uff80\\uff9e^\\uff80\\uff9b\\uff73=\\u5c71\\u7530^\\u592a\\u90ce=\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrHbrw-\\u05e9\\u05e8\\u05d5\\u05df^\\u05d3\\u05d1\\u05d5\\u05e8\\u05d4]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrI2-Hong^Gildong=\\u6d2a^\\u5409\\u6d1e=\\ud64d^\\uae38\\ub3d9]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMulti-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrJapMultiExplicitIR6-\\u3084\\u307e\\u3060^\\u305f\\u308d\\u3046]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrKoreanMulti-\\uae40\\ud76c\\uc911]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrRuss-\\u041b\\u044e\\u043ace\\u043c\\u0431yp\\u0433]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX1-Wang^XiaoDong=\\u738b^\\u5c0f\\u6771]", "pydicom/tests/test_charset.py::TestCharset::test_charset_patient_names[chrX2-Wang^XiaoDong=\\u738b^\\u5c0f\\u4e1c]", "pydicom/tests/test_charset.py::TestCharset::test_changed_character_set", "pydicom/tests/test_charset.py::TestCharset::test_invalid_second_encoding", "pydicom/tests/test_charset.py::TestCharset::test_invalid_second_encoding_strict", "pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_personname", "pydicom/tests/test_charset.py::TestCharset::test_japanese_multi_byte_encoding", "pydicom/tests/test_charset.py::TestCharset::test_bad_japanese_encoding", "pydicom/tests/test_charset.py::TestCharset::test_deprecated_decode"], "environment_setup_commit": "9d69811e539774f296c2f289839147e741251716"}, {"repo": "pyvista/pyvista", "instance_id": "pyvista__pyvista-3750", "base_commit": "3db44d72070ccd6e8fc5c7f708500424011bcc47", "patch": "diff --git a/pyvista/core/filters/data_set.py b/pyvista/core/filters/data_set.py\n--- a/pyvista/core/filters/data_set.py\n+++ b/pyvista/core/filters/data_set.py\n@@ -937,9 +937,10 @@ def threshold(\n         continuous=False,\n         preference='cell',\n         all_scalars=False,\n-        progress_bar=False,\n-        component_mode=\"all\",\n+        component_mode='all',\n         component=0,\n+        method='upper',\n+        progress_bar=False,\n     ):\n         \"\"\"Apply a ``vtkThreshold`` filter to the input dataset.\n \n@@ -956,43 +957,54 @@ def threshold(\n            thresholding depends on whether that point is part of a cell that\n            is kept after thresholding.\n \n+           Please also note the default ``preference`` choice for CELL data\n+           over POINT data. This is contrary to most other places in PyVista's\n+           API where the preference typically defaults to POINT data. We chose\n+           to prefer CELL data here so that if thresholding by a named array\n+           that exists for both the POINT and CELL data, this filter will\n+           default to the CELL data array while performing the CELL-wise\n+           operation.\n+\n         Parameters\n         ----------\n         value : float or sequence, optional\n-            Single value or (min, max) to be used for the data threshold.  If\n+            Single value or (min, max) to be used for the data threshold. If\n             a sequence, then length must be 2. If no value is specified, the\n             non-NaN data range will be used to remove any NaN values.\n+            Please reference the ``method`` parameter for how single values\n+            are handled.\n \n         scalars : str, optional\n             Name of scalars to threshold on. Defaults to currently active scalars.\n \n-        invert : bool, optional\n-            If value is a single value, when invert is ``True`` cells\n-            are kept when their values are below parameter ``\"value\"``.\n-            When invert is ``False`` cells are kept when their value is\n-            above the threshold ``\"value\"``.  Default is ``False``:\n-            yielding above the threshold ``\"value\"``.\n+        invert : bool, default: False\n+            Invert the threshold results. That is, cells that would have been\n+            in the output with this option off are excluded, while cells that\n+            would have been excluded from the output are included.\n \n-        continuous : bool, optional\n+            .. warning::\n+                This option is only supported for VTK version 9+\n+\n+        continuous : bool, default: False\n             When True, the continuous interval [minimum cell scalar,\n             maximum cell scalar] will be used to intersect the threshold bound,\n             rather than the set of discrete scalar values from the vertices.\n \n-        preference : str, optional\n+        preference : str, default: 'cell'\n             When ``scalars`` is specified, this is the preferred array\n             type to search for in the dataset.  Must be either\n-            ``'point'`` or ``'cell'``.\n+            ``'point'`` or ``'cell'``. Throughout PyVista, the preference\n+            is typically ``'point'`` but since the threshold filter is a\n+            cell-wise operation, we prefer cell data for thresholding\n+            operations.\n \n-        all_scalars : bool, optional\n+        all_scalars : bool, default: False\n             If using scalars from point data, all\n             points in a cell must satisfy the threshold when this\n             value is ``True``.  When ``False``, any point of the cell\n             with a scalar value satisfying the threshold criterion\n             will extract the cell. Has no effect when using cell data.\n \n-        progress_bar : bool, optional\n-            Display a progress bar to indicate progress.\n-\n         component_mode : {'selected', 'all', 'any'}\n             The method to satisfy the criteria for the threshold of\n             multicomponent scalars.  'selected' (default)\n@@ -1004,6 +1016,17 @@ def threshold(\n             When using ``component_mode='selected'``, this sets\n             which component to threshold on.\n \n+        method : str, default: 'upper'\n+            Set the threshold method for single-values, defining which\n+            threshold bounds to use. If the ``value`` is a range, this\n+            parameter will be ignored, extracting data between the two\n+            values. For single values, ``'lower'`` will extract data\n+            lower than the  ``value``. ``'upper'`` will extract data\n+            larger than the ``value``.\n+\n+        progress_bar : bool, default: False\n+            Display a progress bar to indicate progress.\n+\n         Returns\n         -------\n         pyvista.UnstructuredGrid\n@@ -1057,31 +1080,6 @@ def threshold(\n \n         field = get_array_association(self, scalars, preference=preference)\n \n-        # If using an inverted range, merge the result of two filters:\n-        if isinstance(value, (np.ndarray, collections.abc.Sequence)) and invert:\n-            valid_range = [np.nanmin(arr), np.nanmax(arr)]\n-            # Create two thresholds\n-            t1 = self.threshold(\n-                [valid_range[0], value[0]],\n-                scalars=scalars,\n-                continuous=continuous,\n-                preference=preference,\n-                invert=False,\n-            )\n-            t2 = self.threshold(\n-                [value[1], valid_range[1]],\n-                scalars=scalars,\n-                continuous=continuous,\n-                preference=preference,\n-                invert=False,\n-            )\n-            # Use an AppendFilter to merge the two results\n-            appender = _vtk.vtkAppendFilter()\n-            appender.AddInputData(t1)\n-            appender.AddInputData(t2)\n-            _update_alg(appender, progress_bar, 'Thresholding')\n-            return _get_output(appender)\n-\n         # Run a standard threshold algorithm\n         alg = _vtk.vtkThreshold()\n         alg.SetAllScalars(all_scalars)\n@@ -1094,18 +1092,9 @@ def threshold(\n         # use valid range if no value given\n         if value is None:\n             value = self.get_data_range(scalars)\n-        # check if value is a sequence (if so threshold by min max range like ParaView)\n-        if isinstance(value, (np.ndarray, collections.abc.Sequence)):\n-            if len(value) != 2:\n-                raise ValueError(\n-                    f'Value range must be length one for a float value or two for min/max; not ({value}).'\n-                )\n-            alg.ThresholdBetween(value[0], value[1])\n-        elif isinstance(value, collections.abc.Iterable):\n-            raise TypeError('Value must either be a single scalar or a sequence.')\n-        else:\n-            # just a single value\n-            _set_threshold_limit(alg, value, invert)\n+\n+        _set_threshold_limit(alg, value, method, invert)\n+\n         if component_mode == \"component\":\n             alg.SetComponentModeToUseSelected()\n             dim = arr.shape[1]\n@@ -1124,6 +1113,7 @@ def threshold(\n             raise ValueError(\n                 f\"component_mode must be 'component', 'all', or 'any' got: {component_mode}\"\n             )\n+\n         # Run the threshold\n         _update_alg(alg, progress_bar, 'Thresholding')\n         return _get_output(alg)\n@@ -1135,6 +1125,7 @@ def threshold_percent(\n         invert=False,\n         continuous=False,\n         preference='cell',\n+        method='upper',\n         progress_bar=False,\n     ):\n         \"\"\"Threshold the dataset by a percentage of its range on the active scalars array.\n@@ -1155,25 +1146,36 @@ def threshold_percent(\n         scalars : str, optional\n             Name of scalars to threshold on. Defaults to currently active scalars.\n \n-        invert : bool, optional\n-            When invert is ``True`` cells are kept when their values are\n-            below the percentage of the range.  When invert is\n-            ``False``, cells are kept when their value is above the\n-            percentage of the range. Default is ``False``: yielding\n-            above the threshold ``\"value\"``.\n-\n-        continuous : bool, optional\n-            When ``True``, the continuous interval [minimum cell scalar,\n-            maximum cell scalar] will be used to intersect the threshold\n-            bound, rather than the set of discrete scalar values from\n-            the vertices.\n+        invert : bool, default: False\n+            Invert the threshold results. That is, cells that would have been\n+            in the output with this option off are excluded, while cells that\n+            would have been excluded from the output are included.\n \n-        preference : str, optional\n+            .. warning::\n+                This option is only supported for VTK version 9+\n+\n+        continuous : bool, default: False\n+            When True, the continuous interval [minimum cell scalar,\n+            maximum cell scalar] will be used to intersect the threshold bound,\n+            rather than the set of discrete scalar values from the vertices.\n+\n+        preference : str, default: 'cell'\n             When ``scalars`` is specified, this is the preferred array\n             type to search for in the dataset.  Must be either\n-            ``'point'`` or ``'cell'``.\n+            ``'point'`` or ``'cell'``. Throughout PyVista, the preference\n+            is typically ``'point'`` but since the threshold filter is a\n+            cell-wise operation, we prefer cell data for thresholding\n+            operations.\n+\n+        method : str, default: 'upper'\n+            Set the threshold method for single-values, defining which\n+            threshold bounds to use. If the ``value`` is a range, this\n+            parameter will be ignored, extracting data between the two\n+            values. For single values, ``'lower'`` will extract data\n+            lower than the  ``value``. ``'upper'`` will extract data\n+            larger than the ``value``.\n \n-        progress_bar : bool, optional\n+        progress_bar : bool, default: False\n             Display a progress bar to indicate progress.\n \n         Returns\n@@ -1239,6 +1241,7 @@ def _get_val(percent, dmin, dmax):\n             invert=invert,\n             continuous=continuous,\n             preference=preference,\n+            method=method,\n             progress_bar=progress_bar,\n         )\n \n@@ -5482,20 +5485,58 @@ def separate_cells(self):\n         return self.shrink(1.0)\n \n \n-def _set_threshold_limit(alg, value, invert):\n-    \"\"\"Set vtkThreshold limit.\n+def _set_threshold_limit(alg, value, method, invert):\n+    \"\"\"Set vtkThreshold limits and function.\n+\n+    Addresses VTK API deprecations and previous PyVista inconsistencies with ParaView. Reference:\n \n-    Addresses VTK API deprecation as pointed out in\n-    https://github.com/pyvista/pyvista/issues/2850\n+    * https://github.com/pyvista/pyvista/issues/2850\n+    * https://github.com/pyvista/pyvista/issues/3610\n+    * https://discourse.vtk.org/t/unnecessary-vtk-api-change/9929\n \n     \"\"\"\n-    if invert:\n-        if pyvista.vtk_version_info >= (9, 1):\n-            alg.SetUpperThreshold(value)\n-        else:  # pragma: no cover\n-            alg.ThresholdByLower(value)\n-    else:\n-        if pyvista.vtk_version_info >= (9, 1):\n-            alg.SetLowerThreshold(value)\n-        else:  # pragma: no cover\n-            alg.ThresholdByUpper(value)\n+    # Check value\n+    if isinstance(value, (np.ndarray, collections.abc.Sequence)):\n+        if len(value) != 2:\n+            raise ValueError(\n+                f'Value range must be length one for a float value or two for min/max; not ({value}).'\n+            )\n+        # Check range\n+        if value[0] > value[1]:\n+            raise ValueError(\n+                'Value sequence is invalid, please use (min, max). The provided first value is greater than the second.'\n+            )\n+    elif isinstance(value, collections.abc.Iterable):\n+        raise TypeError('Value must either be a single scalar or a sequence.')\n+    if pyvista.vtk_version_info >= (9,):\n+        alg.SetInvert(invert)\n+    elif invert:  # pragma: no cover\n+        raise ValueError('PyVista no longer supports inverted thresholds for VTK<9.')\n+    # Set values and function\n+    if pyvista.vtk_version_info >= (9, 1):\n+        if isinstance(value, (np.ndarray, collections.abc.Sequence)):\n+            alg.SetThresholdFunction(_vtk.vtkThreshold.THRESHOLD_BETWEEN)\n+            alg.SetLowerThreshold(value[0])\n+            alg.SetUpperThreshold(value[1])\n+        else:\n+            # Single value\n+            if method.lower() == 'lower':\n+                alg.SetLowerThreshold(value)\n+                alg.SetThresholdFunction(_vtk.vtkThreshold.THRESHOLD_LOWER)\n+            elif method.lower() == 'upper':\n+                alg.SetUpperThreshold(value)\n+                alg.SetThresholdFunction(_vtk.vtkThreshold.THRESHOLD_UPPER)\n+            else:\n+                raise ValueError('Invalid method choice. Either `lower` or `upper`')\n+    else:  # pragma: no cover\n+        # ThresholdByLower, ThresholdByUpper, ThresholdBetween\n+        if isinstance(value, (np.ndarray, collections.abc.Sequence)):\n+            alg.ThresholdBetween(value[0], value[1])\n+        else:\n+            # Single value\n+            if method.lower() == 'lower':\n+                alg.ThresholdByLower(value)\n+            elif method.lower() == 'upper':\n+                alg.ThresholdByUpper(value)\n+            else:\n+                raise ValueError('Invalid method choice. Either `lower` or `upper`')\ndiff --git a/pyvista/plotting/widgets.py b/pyvista/plotting/widgets.py\n--- a/pyvista/plotting/widgets.py\n+++ b/pyvista/plotting/widgets.py\n@@ -1247,6 +1247,8 @@ def add_mesh_threshold(\n         pointa=(0.4, 0.9),\n         pointb=(0.9, 0.9),\n         continuous=False,\n+        all_scalars=False,\n+        method='upper',\n         **kwargs,\n     ):\n         \"\"\"Apply a threshold on a mesh with a slider.\n@@ -1265,8 +1267,13 @@ def add_mesh_threshold(\n         scalars : str, optional\n             The string name of the scalars on the mesh to threshold and display.\n \n-        invert : bool, optional\n-            Invert (flip) the threshold.\n+        invert : bool, default: False\n+            Invert the threshold results. That is, cells that would have been\n+            in the output with this option off are excluded, while cells that\n+            would have been excluded from the output are included.\n+\n+            .. warning::\n+                This option is only supported for VTK version 9+\n \n         widget_color : color_like, optional\n             Color of the widget.  Either a string, RGB sequence, or\n@@ -1277,7 +1284,7 @@ def add_mesh_threshold(\n             * ``color=[1.0, 1.0, 1.0]``\n             * ``color='#FFFFFF'``\n \n-        preference : str, optional\n+        preference : str, default: 'cell'\n             When ``mesh.n_points == mesh.n_cells`` and setting\n             scalars, this parameter sets how the scalars will be\n             mapped to the mesh.  Default ``'cell'``, causes the\n@@ -1287,20 +1294,35 @@ def add_mesh_threshold(\n         title : str, optional\n             The string label of the slider widget.\n \n-        pointa : sequence, optional\n+        pointa : sequence, default: (0.4, 0.9)\n             The relative coordinates of the left point of the slider\n             on the display port.\n \n-        pointb : sequence, optional\n+        pointb : sequence, default: (0.9, 0.9)\n             The relative coordinates of the right point of the slider\n             on the display port.\n \n-        continuous : bool, optional\n+        continuous : bool, default: False\n             If this is enabled (default is ``False``), use the continuous\n             interval ``[minimum cell scalar, maximum cell scalar]``\n             to intersect the threshold bound, rather than the set of\n             discrete scalar values from the vertices.\n \n+        all_scalars : bool, default: False\n+            If using scalars from point data, all\n+            points in a cell must satisfy the threshold when this\n+            value is ``True``.  When ``False``, any point of the cell\n+            with a scalar value satisfying the threshold criterion\n+            will extract the cell. Has no effect when using cell data.\n+\n+        method : str, default: 'upper'\n+            Set the threshold method for single-values, defining which\n+            threshold bounds to use. If the ``value`` is a range, this\n+            parameter will be ignored, extracting data between the two\n+            values. For single values, ``'lower'`` will extract data\n+            lower than the  ``value``. ``'upper'`` will extract data\n+            larger than the ``value``.\n+\n         **kwargs : dict, optional\n             All additional keyword arguments are passed to ``add_mesh`` to\n             control how the mesh is displayed.\n@@ -1338,12 +1360,13 @@ def add_mesh_threshold(\n             0, 0, 0, field.value, scalars\n         )  # args: (idx, port, connection, field, name)\n         alg.SetUseContinuousCellRange(continuous)\n+        alg.SetAllScalars(all_scalars)\n \n         threshold_mesh = pyvista.wrap(alg.GetOutput())\n         self.threshold_meshes.append(threshold_mesh)\n \n         def callback(value):\n-            _set_threshold_limit(alg, value, invert)\n+            _set_threshold_limit(alg, value, method, invert)\n             alg.Update()\n             threshold_mesh.shallow_copy(alg.GetOutput())\n \n", "test_patch": "diff --git a/tests/plotting/test_widgets.py b/tests/plotting/test_widgets.py\n--- a/tests/plotting/test_widgets.py\n+++ b/tests/plotting/test_widgets.py\n@@ -150,10 +150,12 @@ def test_widget_slider(uniform):\n     p.add_slider_widget(callback=func, rng=[0, 10], style=\"modern\", pass_widget=True)\n     p.close()\n \n-    p = pyvista.Plotter()\n-    p.add_mesh_threshold(uniform, invert=True)\n-    p.add_mesh(uniform.outline())\n-    p.close()\n+    if pyvista.vtk_version_info >= (9,):\n+        # Invert not support for VTK8.1.2\n+        p = pyvista.Plotter()\n+        p.add_mesh_threshold(uniform, invert=True)\n+        p.add_mesh(uniform.outline())\n+        p.close()\n \n     p = pyvista.Plotter()\n     p.add_mesh_threshold(uniform, invert=False)\ndiff --git a/tests/test_filters.py b/tests/test_filters.py\n--- a/tests/test_filters.py\n+++ b/tests/test_filters.py\n@@ -348,9 +348,10 @@ def test_threshold(datasets):\n     thresh = dataset.threshold([100, 500], invert=False, progress_bar=True)\n     assert thresh is not None\n     assert isinstance(thresh, pyvista.UnstructuredGrid)\n-    thresh = dataset.threshold([100, 500], invert=True, progress_bar=True)\n-    assert thresh is not None\n-    assert isinstance(thresh, pyvista.UnstructuredGrid)\n+    if pyvista.vtk_version_info >= (9,):\n+        thresh = dataset.threshold([100, 500], invert=True, progress_bar=True)\n+        assert thresh is not None\n+        assert isinstance(thresh, pyvista.UnstructuredGrid)\n     # allow Sequence but not Iterable\n     with pytest.raises(TypeError):\n         dataset.threshold({100, 500}, progress_bar=True)\n@@ -364,6 +365,16 @@ def test_threshold(datasets):\n     with pytest.raises(ValueError):\n         dataset.threshold([10, 100, 300], progress_bar=True)\n \n+    if pyvista.vtk_version_info < (9,):\n+        with pytest.raises(ValueError):\n+            dataset.threshold([100, 500], invert=True)\n+\n+    with pytest.raises(ValueError):\n+        dataset.threshold(100, method='between')\n+\n+    with pytest.raises(ValueError):\n+        dataset.threshold((2, 1))\n+\n \n def test_threshold_all_scalars():\n     mesh = pyvista.Sphere()\n@@ -434,6 +445,8 @@ def test_threshold_percent(datasets):\n     inverts = [False, True, False, True, False]\n     # Only test data sets that have arrays\n     for i, dataset in enumerate(datasets[0:3]):\n+        if inverts[i] and pyvista.vtk_version_info < (9,):\n+            continue\n         thresh = dataset.threshold_percent(\n             percent=percents[i], invert=inverts[i], progress_bar=True\n         )\n@@ -450,6 +463,106 @@ def test_threshold_percent(datasets):\n         dataset.threshold_percent({18.0, 85.0})\n \n \n+@pytest.mark.skipif(\n+    pyvista.vtk_version_info < (9,),\n+    reason='The invert parameter is not supported for VTK<9. The general logic for the API differences is tested for VTK<9.1 though.',\n+)\n+def test_threshold_paraview_consistency():\n+    \"\"\"Validate expected results that match ParaView.\"\"\"\n+    x = np.arange(5, dtype=float)\n+    y = np.arange(6, dtype=float)\n+    z = np.arange(2, dtype=float)\n+    xx, yy, zz = np.meshgrid(x, y, z)\n+    mesh = pyvista.StructuredGrid(xx, yy, zz)\n+    mesh.cell_data.set_scalars(np.repeat(range(5), 4))\n+\n+    # Input mesh\n+    #   [[0, 0, 0, 0, 1],\n+    #    [1, 1, 1, 2, 2],\n+    #    [2, 2, 3, 3, 3],\n+    #    [3, 4, 4, 4, 4]]\n+\n+    # upper(0): extract all\n+    thresh = mesh.threshold(0, invert=False, method='upper')\n+    assert thresh.n_cells == mesh.n_cells\n+    assert np.allclose(thresh.active_scalars, mesh.active_scalars)\n+    # upper(0),invert: extract none\n+    thresh = mesh.threshold(0, invert=True, method='upper')\n+    assert thresh.n_cells == 0\n+\n+    # lower(0)\n+    #   [[0, 0, 0, 0   ]]\n+    thresh = mesh.threshold(0, invert=False, method='lower')\n+    assert thresh.n_cells == 4\n+    assert np.allclose(thresh.active_scalars, np.array([0, 0, 0, 0]))\n+    # lower(0),invert\n+    #   [[            1],\n+    #    [1, 1, 1, 2, 2],\n+    #    [2, 2, 3, 3, 3],\n+    #    [3, 4, 4, 4, 4]]\n+    thresh = mesh.threshold(0, invert=True, method='lower')\n+    assert thresh.n_cells == 16\n+    assert thresh.get_data_range() == (1, 4)\n+\n+    # upper(2)\n+    #   [[         2, 2],\n+    #    [2, 2, 3, 3, 3],\n+    #    [3, 4, 4, 4, 4]]\n+    thresh = mesh.threshold(2, invert=False, method='upper')\n+    assert thresh.n_cells == 12\n+    assert thresh.get_data_range() == (2, 4)\n+    # upper(2),invert\n+    #   [[0, 0, 0, 0, 1],\n+    #    [1, 1, 1,     ]]\n+    thresh = mesh.threshold(2, invert=True, method='upper')\n+    assert thresh.n_cells == 8\n+    assert thresh.get_data_range() == (0, 1)\n+\n+    # lower(2)\n+    #   [[0, 0, 0, 0, 1],\n+    #    [1, 1, 1, 2, 2],\n+    #    [2, 2,        ]]\n+    thresh = mesh.threshold(2, invert=False, method='lower')\n+    assert thresh.n_cells == 12\n+    assert thresh.get_data_range() == (0, 2)\n+    # lower(2),invert\n+    #   [[      3, 3, 3],\n+    #    [3, 4, 4, 4, 4]]\n+    thresh = mesh.threshold(2, invert=True, method='lower')\n+    assert thresh.n_cells == 8\n+    assert thresh.get_data_range() == (3, 4)\n+\n+    # between(0, 0)\n+    #   [[0, 0, 0, 0   ]]\n+    thresh = mesh.threshold((0, 0), invert=False)\n+    assert thresh.n_cells == 4\n+    assert np.allclose(thresh.active_scalars, np.array([0, 0, 0, 0]))\n+    # between(0,0),invert\n+    #   [[            1],\n+    #    [1, 1, 1, 2, 2],\n+    #    [2, 2, 3, 3, 3],\n+    #    [3, 4, 4, 4, 4]]\n+    thresh = mesh.threshold((0, 0), invert=True)\n+    assert thresh.n_cells == 16\n+    assert thresh.get_data_range() == (1, 4)\n+\n+    # between(2,3)\n+    #   [[         2, 2],\n+    #    [2, 2, 3, 3, 3],\n+    #    [3,           ]]\n+    thresh = mesh.threshold((2, 3), invert=False)\n+    assert thresh.n_cells == 8\n+    assert thresh.get_data_range() == (2, 3)\n+    # between(2,3),invert\n+    #   [[0, 0, 0, 0, 1],\n+    #    [1, 1, 1,     ],\n+    #    [             ],\n+    #    [   4, 4, 4, 4]]\n+    thresh = mesh.threshold((2, 3), invert=True)\n+    assert thresh.n_cells == 12\n+    assert thresh.get_data_range() == (0, 4)\n+\n+\n def test_outline(datasets):\n     for dataset in datasets:\n         outline = dataset.outline(progress_bar=True)\n@@ -911,6 +1024,10 @@ def test_glyph_orient_and_scale():\n     assert glyph4.bounds[0] == geom.bounds[0] and glyph4.bounds[1] == geom.bounds[1]\n \n \n+@pytest.mark.skipif(\n+    pyvista.vtk_version_info < (9,),\n+    reason='The invert parameter is not supported for VTK<9.',\n+)\n def test_split_and_connectivity():\n     # Load a simple example mesh\n     dataset = examples.load_uniform()\n", "problem_statement": "Unexpected threshold behavior\n### Describe the bug, what's wrong, and what you expected.\n\nI'm using simple structed grids of cells, and need to filter-out some \"nodata\" cells. To do this, I'm setting scalar values to the cell data, then using [threshold](https://docs.pyvista.org/api/core/_autosummary/pyvista.DataSetFilters.threshold.html) with the nodata value with `invert=True`. However, I'm getting confusing and inconsistent results compared to ParaView.\n\n### Steps to reproduce the bug.\n\n```python\r\nimport numpy as np\r\nimport pyvista\r\n\r\nx = np.arange(5, dtype=float)\r\ny = np.arange(6, dtype=float)\r\nz = np.arange(2, dtype=float)\r\nxx, yy, zz = np.meshgrid(x, y, z)\r\nmesh = pyvista.StructuredGrid(xx, yy, zz)\r\nmesh.cell_data.set_scalars(np.repeat(range(5), 4))\r\n\r\n# All data\r\nmesh.plot(show_edges=True)\r\n# output is normal\r\n\r\n# Filtering out nodata (zero) values\r\nmesh.threshold(0, invert=True).plot(show_edges=True)\r\n# output does not look normal, only 0-value cells are shown\r\n```\n\n### System Information\n\n```shell\n--------------------------------------------------------------------------------\r\n  Date: Thu Nov 17 15:23:57 2022 New Zealand Daylight Time\r\n\r\n                OS : Windows\r\n            CPU(s) : 12\r\n           Machine : AMD64\r\n      Architecture : 64bit\r\n               RAM : 31.7 GiB\r\n       Environment : IPython\r\n        GPU Vendor : NVIDIA Corporation\r\n      GPU Renderer : NVIDIA RTX A4000/PCIe/SSE2\r\n       GPU Version : 4.5.0 NVIDIA 472.39\r\n\r\n  Python 3.9.13 | packaged by conda-forge | (main, May 27 2022, 16:50:36) [MSC\r\n  v.1929 64 bit (AMD64)]\r\n\r\n           pyvista : 0.37.0\r\n               vtk : 9.1.0\r\n             numpy : 1.22.3\r\n           imageio : 2.22.0\r\n            scooby : 0.7.0\r\n             pooch : v1.6.0\r\n        matplotlib : 3.6.2\r\n             PyQt5 : 5.12.3\r\n           IPython : 8.6.0\r\n          colorcet : 3.0.1\r\n             scipy : 1.8.0\r\n              tqdm : 4.63.0\r\n            meshio : 5.3.4\r\n--------------------------------------------------------------------------------\n```\n\n\n### Screenshots\n\nNormal looking whole grid:\r\n![image](https://user-images.githubusercontent.com/895458/202339692-5046b23f-c3c8-4b2c-aaa7-4aa06afbae9f.png)\r\n\r\nOdd-looking threshold attempt with pyvista, showing only 0-values:\r\n![image](https://user-images.githubusercontent.com/895458/202339879-b2270e4c-a71b-4d43-86f8-4f67445b7b69.png)\r\n\r\nExpected result with ParaView theshold filter with upper/lower set to 0 and invert selected:\r\n![image](https://user-images.githubusercontent.com/895458/202340379-fea26838-b0f4-4828-b510-825f53522e87.png)\r\n\r\nApologies for any \"user error\", as I'm new to this package.\n", "hints_text": "Can confirm, and I am seeing quite a few inconsistent results with the threshold filter. So this is not a user error!\r\n\r\n For example `inverter=True/False` should produce two logical inverses for this mesh, but it doesn't:\r\n<img width=\"624\" alt=\"Screen Shot 2022-11-16 at 11 18 48 PM\" src=\"https://user-images.githubusercontent.com/22067021/202371190-7dcd64df-1882-4876-b4c3-43fe614913a6.png\">\r\n\r\n\r\nSecond, depending on if `value` is a single value `0` or a range `[0, 0]` yields completely different results from the PyVista filter (but not in ParaView):\r\n\r\n\r\n```py\r\np = pv.Plotter(notebook=0, shape=(1,2))\r\np.add_mesh(mesh.threshold(0, invert=False))\r\np.subplot(0,1)\r\np.add_mesh(mesh.threshold(0, invert=True))\r\np.link_views()\r\np.view_isometric()\r\np.show()\r\n```\r\n\r\n<img width=\"624\" alt=\"Screen Shot 2022-11-16 at 11 20 01 PM\" src=\"https://user-images.githubusercontent.com/22067021/202371391-393280c1-1091-4c61-82b2-e95e76b49327.png\">\r\n\r\n\r\nvs.\r\n\r\n```py\r\np = pv.Plotter(notebook=0, shape=(1,2))\r\np.add_mesh(mesh.threshold([0, 0], invert=False))\r\np.subplot(0,1)\r\np.add_mesh(mesh.threshold([0, 0], invert=True))\r\np.link_views()\r\np.view_isometric()\r\np.show()\r\n```\r\n\r\n<img width=\"624\" alt=\"Screen Shot 2022-11-16 at 11 20 34 PM\" src=\"https://user-images.githubusercontent.com/22067021/202371476-cfd0fabb-daad-47db-acc5-855b43504f21.png\">\r\n\r\n\r\nThis is not good... I'll start digging into this and see if I can fix the `threshold` filter such that I has consistency with itself and with ParaView\r\n\nThanks for taking a closer look. I should have mentioned that I get the same behavior on linux using a similar conda-forge setup.\r\n\r\nI've found that ranges like `[1, 1]` work as expected to filter on values == 1, but to filter on zero, it needs to span a very small range:\r\n```python\r\np = pv.Plotter(notebook=0, shape=(1,2))\r\np.add_mesh(mesh.threshold([-1e-30, 1e-30], invert=False))\r\np.subplot(0,1)\r\np.add_mesh(mesh.threshold([-1e-30, 1e-30], invert=True))\r\np.link_views()\r\np.view_isometric()\r\np.show()\r\n```\r\nwhich also logs this message three times:\r\n> 2022-11-17 22:30:30.021 ( 559.089s) [        20F0C740]       vtkThreshold.cxx:96    WARN| vtkThreshold::ThresholdBetween was deprecated for VTK 9.1 and will be removed in a future version.\r\n\nAnother solution is to use [`extract_cells`](https://docs.pyvista.org/api/core/_autosummary/pyvista.StructuredGrid.extract_cells.html):\r\n```python\r\nmesh.extract_cells(mesh.cell_data.active_scalars != 0).plot()\r\nmesh.extract_cells(mesh.cell_data.active_scalars != 1).plot()\r\n```", "created_at": "2022-12-25T02:44:32Z", "version": "0.38", "FAIL_TO_PASS": ["tests/test_filters.py::test_threshold", "tests/test_filters.py::test_threshold_paraview_consistency"], "PASS_TO_PASS": ["tests/test_filters.py::test_datasetfilters_init", "tests/test_filters.py::test_clip_filter", "tests/test_filters.py::test_clip_by_scalars_filter[False-False]", "tests/test_filters.py::test_clip_by_scalars_filter[False-True]", "tests/test_filters.py::test_clip_by_scalars_filter[True-False]", "tests/test_filters.py::test_clip_by_scalars_filter[True-True]", "tests/test_filters.py::test_clip_filter_no_active", "tests/test_filters.py::test_clip_filter_scalar_multiple", "tests/test_filters.py::test_clip_filter_composite", "tests/test_filters.py::test_clip_box", "tests/test_filters.py::test_clip_box_composite", "tests/test_filters.py::test_clip_surface", "tests/test_filters.py::test_clip_closed_surface", "tests/test_filters.py::test_implicit_distance", "tests/test_filters.py::test_slice_filter", "tests/test_filters.py::test_slice_filter_composite", "tests/test_filters.py::test_slice_orthogonal_filter", "tests/test_filters.py::test_slice_orthogonal_filter_composite", "tests/test_filters.py::test_slice_along_axis", "tests/test_filters.py::test_slice_along_axis_composite", "tests/test_filters.py::test_threshold_all_scalars", "tests/test_filters.py::test_threshold_multicomponent", "tests/test_filters.py::test_threshold_percent", "tests/test_filters.py::test_outline", "tests/test_filters.py::test_outline_composite", "tests/test_filters.py::test_outline_corners", "tests/test_filters.py::test_outline_corners_composite", "tests/test_filters.py::test_extract_geometry", "tests/test_filters.py::test_extract_geometry_extent", "tests/test_filters.py::test_extract_all_edges", "tests/test_filters.py::test_wireframe_composite", "tests/test_filters.py::test_delaunay_2d_unstructured", "tests/test_filters.py::test_contour[contour]", "tests/test_filters.py::test_contour[marching_cubes]", "tests/test_filters.py::test_contour[flying_edges]", "tests/test_filters.py::test_contour_errors", "tests/test_filters.py::test_elevation", "tests/test_filters.py::test_elevation_composite", "tests/test_filters.py::test_texture_map_to_plane", "tests/test_filters.py::test_texture_map_to_sphere", "tests/test_filters.py::test_compute_cell_sizes", "tests/test_filters.py::test_compute_cell_sizes_composite", "tests/test_filters.py::test_cell_centers", "tests/test_filters.py::test_cell_centers_composite", "tests/test_filters.py::test_glyph", "tests/test_filters.py::test_glyph_cell_point_data", "tests/test_filters.py::test_glyph_settings", "tests/test_filters.py::test_glyph_orient_and_scale", "tests/test_filters.py::test_split_and_connectivity", "tests/test_filters.py::test_warp_by_scalar", "tests/test_filters.py::test_warp_by_vector", "tests/test_filters.py::test_invalid_warp_scalar", "tests/test_filters.py::test_invalid_warp_scalar_inplace", "tests/test_filters.py::test_invalid_warp_vector", "tests/test_filters.py::test_cell_data_to_point_data", "tests/test_filters.py::test_cell_data_to_point_data_composite", "tests/test_filters.py::test_point_data_to_cell_data", "tests/test_filters.py::test_point_data_to_cell_data_composite", "tests/test_filters.py::test_triangulate", "tests/test_filters.py::test_triangulate_composite", "tests/test_filters.py::test_delaunay_3d", "tests/test_filters.py::test_smooth", "tests/test_filters.py::test_smooth_taubin", "tests/test_filters.py::test_resample", "tests/test_filters.py::test_probe[None-True-True]", "tests/test_filters.py::test_probe[None-True-False]", "tests/test_filters.py::test_probe[None-False-True]", "tests/test_filters.py::test_probe[None-False-False]", "tests/test_filters.py::test_probe[locator1-True-True]", "tests/test_filters.py::test_probe[locator1-True-False]", "tests/test_filters.py::test_probe[locator1-False-True]", "tests/test_filters.py::test_probe[locator1-False-False]", "tests/test_filters.py::test_streamlines_dir[forward]", "tests/test_filters.py::test_streamlines_dir[backward]", "tests/test_filters.py::test_streamlines_dir[both]", "tests/test_filters.py::test_streamlines_type[2]", "tests/test_filters.py::test_streamlines_type[4]", "tests/test_filters.py::test_streamlines_type[45]", "tests/test_filters.py::test_streamlines_cell_point[point]", "tests/test_filters.py::test_streamlines_cell_point[cell]", "tests/test_filters.py::test_streamlines_return_source", "tests/test_filters.py::test_streamlines_start_position", "tests/test_filters.py::test_streamlines_errors", "tests/test_filters.py::test_streamlines_from_source", "tests/test_filters.py::test_streamlines_from_source_structured_grids", "tests/test_filters.py::test_streamlines_evenly_spaced_2D", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_sep_dist_ratio", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_start_position", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_vectors", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_integrator_type", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_interpolator_type", "tests/test_filters.py::test_streamlines_evenly_spaced_2D_errors", "tests/test_filters.py::test_sample_over_line", "tests/test_filters.py::test_plot_over_line", "tests/test_filters.py::test_sample_over_multiple_lines", "tests/test_filters.py::test_sample_over_circular_arc", "tests/test_filters.py::test_sample_over_circular_arc_normal", "tests/test_filters.py::test_plot_over_circular_arc", "tests/test_filters.py::test_plot_over_circular_arc_normal", "tests/test_filters.py::test_slice_along_line", "tests/test_filters.py::test_extract_points", "tests/test_filters.py::test_slice_along_line_composite", "tests/test_filters.py::test_interpolate", "tests/test_filters.py::test_select_enclosed_points", "tests/test_filters.py::test_decimate_boundary", "tests/test_filters.py::test_extract_surface", "tests/test_filters.py::test_merge_general", "tests/test_filters.py::test_iadd_general", "tests/test_filters.py::test_compute_cell_quality", "tests/test_filters.py::test_compute_derivatives", "tests/test_filters.py::test_extract_subset", "tests/test_filters.py::test_gaussian_smooth_output_type", "tests/test_filters.py::test_gaussian_smooth_constant_data", "tests/test_filters.py::test_gaussian_smooth_outlier", "tests/test_filters.py::test_gaussian_smooth_cell_data_specified", "tests/test_filters.py::test_gaussian_smooth_cell_data_active", "tests/test_filters.py::test_median_smooth_output_type", "tests/test_filters.py::test_median_smooth_constant_data", "tests/test_filters.py::test_median_smooth_outlier", "tests/test_filters.py::test_image_dilate_erode_output_type", "tests/test_filters.py::test_image_dilate_erode_dilation", "tests/test_filters.py::test_image_dilate_erode_erosion", "tests/test_filters.py::test_image_dilate_erode_cell_data_specified", "tests/test_filters.py::test_image_dilate_erode_cell_data_active", "tests/test_filters.py::test_image_threshold_output_type", "tests/test_filters.py::test_image_threshold_wrong_threshold_length", "tests/test_filters.py::test_image_threshold_wrong_threshold_type", "tests/test_filters.py::test_image_threshold_upper[0-1]", "tests/test_filters.py::test_image_threshold_upper[0-None]", "tests/test_filters.py::test_image_threshold_upper[None-1]", "tests/test_filters.py::test_image_threshold_upper[None-None]", "tests/test_filters.py::test_image_threshold_between[0-1]", "tests/test_filters.py::test_image_threshold_between[0-None]", "tests/test_filters.py::test_image_threshold_between[None-1]", "tests/test_filters.py::test_image_threshold_between[None-None]", "tests/test_filters.py::test_extract_subset_structured", "tests/test_filters.py::test_concatenate_structured", "tests/test_filters.py::test_concatenate_structured_bad_dimensions", "tests/test_filters.py::test_concatenate_structured_bad_inputs", "tests/test_filters.py::test_concatenate_structured_bad_point_data", "tests/test_filters.py::test_concatenate_structured_disconnected", "tests/test_filters.py::test_concatenate_structured_different_arrays", "tests/test_filters.py::test_structured_add_non_grid", "tests/test_filters.py::test_poly_data_strip", "tests/test_filters.py::test_shrink", "tests/test_filters.py::test_tessellate", "tests/test_filters.py::test_transform_mesh[0-0]", "tests/test_filters.py::test_transform_mesh[0-1]", "tests/test_filters.py::test_transform_mesh[0-2]", "tests/test_filters.py::test_transform_mesh[1-0]", "tests/test_filters.py::test_transform_mesh[1-1]", "tests/test_filters.py::test_transform_mesh[1-2]", "tests/test_filters.py::test_transform_mesh[2-0]", "tests/test_filters.py::test_transform_mesh[2-1]", "tests/test_filters.py::test_transform_mesh[2-2]", "tests/test_filters.py::test_transform_mesh_and_vectors[0-0]", "tests/test_filters.py::test_transform_mesh_and_vectors[0-1]", "tests/test_filters.py::test_transform_mesh_and_vectors[0-2]", "tests/test_filters.py::test_transform_mesh_and_vectors[1-0]", "tests/test_filters.py::test_transform_mesh_and_vectors[1-1]", "tests/test_filters.py::test_transform_mesh_and_vectors[1-2]", "tests/test_filters.py::test_transform_mesh_and_vectors[2-0]", "tests/test_filters.py::test_transform_mesh_and_vectors[2-1]", "tests/test_filters.py::test_transform_mesh_and_vectors[2-2]", "tests/test_filters.py::test_transform_int_vectors_warning[0-0]", "tests/test_filters.py::test_transform_int_vectors_warning[0-1]", "tests/test_filters.py::test_transform_int_vectors_warning[0-2]", "tests/test_filters.py::test_transform_int_vectors_warning[1-0]", "tests/test_filters.py::test_transform_int_vectors_warning[1-1]", "tests/test_filters.py::test_transform_int_vectors_warning[1-2]", "tests/test_filters.py::test_transform_int_vectors_warning[2-0]", "tests/test_filters.py::test_transform_int_vectors_warning[2-1]", "tests/test_filters.py::test_transform_int_vectors_warning[2-2]", "tests/test_filters.py::test_transform_inplace_bad_types[dataset0]", "tests/test_filters.py::test_transform_inplace_bad_types[dataset1]", "tests/test_filters.py::test_reflect_mesh_about_point", "tests/test_filters.py::test_reflect_mesh_with_vectors", "tests/test_filters.py::test_reflect_inplace[dataset0]", "tests/test_filters.py::test_reflect_inplace[dataset1]", "tests/test_filters.py::test_reflect_inplace[dataset2]", "tests/test_filters.py::test_transform_inplace_bad_types_2[dataset0]", "tests/test_filters.py::test_transform_inplace_bad_types_2[dataset1]", "tests/test_filters.py::test_extrude_rotate", "tests/test_filters.py::test_extrude_rotate_inplace", "tests/test_filters.py::test_extrude_trim", "tests/test_filters.py::test_extrude_trim_strategy[intersection-boundary_edges]", "tests/test_filters.py::test_extrude_trim_strategy[intersection-all_edges]", "tests/test_filters.py::test_extrude_trim_strategy[minimum_distance-boundary_edges]", "tests/test_filters.py::test_extrude_trim_strategy[minimum_distance-all_edges]", "tests/test_filters.py::test_extrude_trim_strategy[maximum_distance-boundary_edges]", "tests/test_filters.py::test_extrude_trim_strategy[maximum_distance-all_edges]", "tests/test_filters.py::test_extrude_trim_strategy[average_distance-boundary_edges]", "tests/test_filters.py::test_extrude_trim_strategy[average_distance-all_edges]", "tests/test_filters.py::test_extrude_trim_catch", "tests/test_filters.py::test_extrude_trim_inplace", "tests/test_filters.py::test_subdivide_adaptive[True]", "tests/test_filters.py::test_subdivide_adaptive[False]", "tests/test_filters.py::test_invalid_subdivide_adaptive", "tests/test_filters.py::test_collision", "tests/test_filters.py::test_collision_solid_non_triangle", "tests/test_filters.py::test_reconstruct_surface_poly", "tests/test_filters.py::test_is_manifold", "tests/test_filters.py::test_reconstruct_surface_unstructured", "tests/test_filters.py::test_integrate_data_datasets", "tests/test_filters.py::test_integrate_data", "tests/test_filters.py::test_subdivide_tetra"], "environment_setup_commit": "8dd8eeb80248cea4360c753847bd622e8652e059"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-965", "base_commit": "ee775c8a137cd8e0b69b46dc24c23648c31fe34c", "patch": "diff --git a/pydicom/config.py b/pydicom/config.py\n--- a/pydicom/config.py\n+++ b/pydicom/config.py\n@@ -87,9 +87,10 @@ def DS_decimal(use_Decimal_boolean=True):\n \"\"\"\n \n use_none_as_empty_text_VR_value = False\n-\"\"\" If ``True``, the value of decoded empty data element is always ``None``.\n-If ``False`` (the default), the value of an empty data element with\n-a text VR is an empty string, for all other VRs it is also ``None``.\n+\"\"\" If ``True``, the value of a decoded empty data element with\n+a text VR is ``None``, otherwise (the default), it is is an empty string.\n+For all other VRs the behavior does not change - the value is en empty\n+list for VR 'SQ' and ``None`` for all other VRs.\n Note that the default of this value will change to ``True`` in version 2.0.\n \"\"\"\n \ndiff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -48,10 +48,12 @@ def empty_value_for_VR(VR, raw=False):\n \n     The behavior of this property depends on the setting of\n     :attr:`config.use_none_as_empty_value`. If that is set to ``True``,\n-    an empty value is always represented by ``None``, otherwise it depends\n-    on `VR`. For text VRs (this includes 'AE', 'AS', 'CS', 'DA', 'DT', 'LO',\n-    'LT', 'PN', 'SH', 'ST', 'TM', 'UC', 'UI', 'UR' and 'UT') an empty string\n-    is used as empty value representation, for all other VRs, ``None``.\n+    an empty value is represented by ``None`` (except for VR 'SQ'), otherwise\n+    it depends on `VR`. For text VRs (this includes 'AE', 'AS', 'CS', 'DA',\n+    'DT', 'LO', 'LT', 'PN', 'SH', 'ST', 'TM', 'UC', 'UI', 'UR' and 'UT') an\n+    empty string is used as empty value representation, for all other VRs\n+    except 'SQ', ``None``. For empty sequence values (VR 'SQ') an empty list\n+    is used in all cases.\n     Note that this is used only if decoding the element - it is always\n     possible to set the value to another empty value representation,\n     which will be preserved during the element object lifetime.\n@@ -67,10 +69,12 @@ def empty_value_for_VR(VR, raw=False):\n \n     Returns\n     -------\n-    str or bytes or None\n+    str or bytes or None or list\n         The value a data element with `VR` is assigned on decoding\n         if it is empty.\n     \"\"\"\n+    if VR == 'SQ':\n+        return []\n     if config.use_none_as_empty_text_VR_value:\n         return None\n     if VR in ('AE', 'AS', 'CS', 'DA', 'DT', 'LO', 'LT',\n", "test_patch": "diff --git a/pydicom/tests/test_dataelem.py b/pydicom/tests/test_dataelem.py\n--- a/pydicom/tests/test_dataelem.py\n+++ b/pydicom/tests/test_dataelem.py\n@@ -503,6 +503,23 @@ def check_empty_binary_element(value):\n             check_empty_binary_element(MultiValue(int, []))\n             check_empty_binary_element(None)\n \n+    def test_empty_sequence_is_handled_as_array(self):\n+        ds = Dataset()\n+        ds.AcquisitionContextSequence = []\n+        elem = ds['AcquisitionContextSequence']\n+        assert bool(elem.value) is False\n+        assert 0 == elem.VM\n+        assert elem.value == []\n+\n+        fp = DicomBytesIO()\n+        fp.is_little_endian = True\n+        fp.is_implicit_VR = True\n+        filewriter.write_dataset(fp, ds)\n+        ds_read = dcmread(fp, force=True)\n+        elem = ds_read['AcquisitionContextSequence']\n+        assert 0 == elem.VM\n+        assert elem.value == []\n+\n \n class TestRawDataElement(object):\n     \"\"\"Tests for dataelem.RawDataElement.\"\"\"\n", "problem_statement": "Empty data elements with value representation SQ are set to None\n**Describe the bug**\r\nIn the current `master`, empty data elements are not read correctly from files. The attribute value is set to `None` instead of `[]`.\r\n\r\n**Expected behavior**\r\nCreate empty list `[]` for empty sequence, i.e., a sequence with zero items.\r\n\r\n**Steps To Reproduce**\r\n```python\r\nimport pydicom\r\nds = pydicom.Dataset()\r\nds.AcquisitionContextSequence = []\r\nprint(ds)\r\nds.is_little_endian = True\r\nds.is_implicit_VR = True\r\nds.save_as('/tmp/test.dcm')\r\n\r\nreloaded_ds = pydicom.dcmread('/tmp/test.dcm', force=True)\r\nprint(reloaded_ds)\r\n```\r\nThis prints:\r\n```\r\n(0040, 0555)  Acquisition Context Sequence   0 item(s) ----\r\n...\r\nTypeError: With tag (0040, 0555) got exception: object of type 'NoneType' has no len()\r\nTraceback (most recent call last):\r\n  File \"/private/tmp/pydicom/pydicom/tag.py\", line 30, in tag_in_exception\r\n    yield\r\n  File \"/private/tmp/pydicom/pydicom/dataset.py\", line 1599, in _pretty_str\r\n    len(data_element.value)))\r\nTypeError: object of type 'NoneType' has no len()\r\n```\r\n\r\n**Your environment**\r\n```\r\nDarwin-18.6.0-x86_64-i386-64bit\r\nPython  3.7.3 (default, Mar 27 2019, 09:23:15)\r\n[Clang 10.0.1 (clang-1001.0.46.3)]\r\npydicom  1.4.0.dev0\r\n```\n", "hints_text": "", "created_at": "2019-11-01T14:43:06Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_dataelem.py::TestDataElement::test_empty_sequence_is_handled_as_array"], "PASS_TO_PASS": ["pydicom/tests/test_dataelem.py::TestDataElement::test_VM_1", "pydicom/tests/test_dataelem.py::TestDataElement::test_VM_2", "pydicom/tests/test_dataelem.py::TestDataElement::test_DSFloat_conversion", "pydicom/tests/test_dataelem.py::TestDataElement::test_backslash", "pydicom/tests/test_dataelem.py::TestDataElement::test_UID", "pydicom/tests/test_dataelem.py::TestDataElement::test_keyword", "pydicom/tests/test_dataelem.py::TestDataElement::test_retired", "pydicom/tests/test_dataelem.py::TestDataElement::test_description_group_length", "pydicom/tests/test_dataelem.py::TestDataElement::test_description_unknown_private", "pydicom/tests/test_dataelem.py::TestDataElement::test_description_unknown", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_standard_element", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_private_element", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_sequence_element", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_not_rlement", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_inheritance", "pydicom/tests/test_dataelem.py::TestDataElement::test_equality_class_members", "pydicom/tests/test_dataelem.py::TestDataElement::test_inequality_standard", "pydicom/tests/test_dataelem.py::TestDataElement::test_inequality_sequence", "pydicom/tests/test_dataelem.py::TestDataElement::test_hash", "pydicom/tests/test_dataelem.py::TestDataElement::test_repeater_str", "pydicom/tests/test_dataelem.py::TestDataElement::test_str_no_vr", "pydicom/tests/test_dataelem.py::TestDataElement::test_repr_seq", "pydicom/tests/test_dataelem.py::TestDataElement::test_getitem_raises", "pydicom/tests/test_dataelem.py::TestDataElement::test_repval_large_elem", "pydicom/tests/test_dataelem.py::TestDataElement::test_repval_large_vm", "pydicom/tests/test_dataelem.py::TestDataElement::test_repval_strange_type", "pydicom/tests/test_dataelem.py::TestDataElement::test_private_tag_in_repeater_range", "pydicom/tests/test_dataelem.py::TestDataElement::test_private_repeater_tag", "pydicom/tests/test_dataelem.py::TestDataElement::test_known_tags_with_UN_VR", "pydicom/tests/test_dataelem.py::TestDataElement::test_unknown_tags_with_UN_VR", "pydicom/tests/test_dataelem.py::TestDataElement::test_tag_with_long_value_UN_VR", "pydicom/tests/test_dataelem.py::TestDataElement::test_empty_text_values[True-None]", "pydicom/tests/test_dataelem.py::TestDataElement::test_empty_text_values[False-]", "pydicom/tests/test_dataelem.py::TestDataElement::test_empty_binary_values", "pydicom/tests/test_dataelem.py::TestRawDataElement::test_key_error", "pydicom/tests/test_dataelem.py::TestRawDataElement::test_valid_tag", "pydicom/tests/test_dataelem.py::TestRawDataElement::test_data_element_without_encoding", "pydicom/tests/test_dataelem.py::TestRawDataElement::test_unknown_vr"], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pyvista/pyvista", "instance_id": "pyvista__pyvista-4315", "base_commit": "db6ee8dd4a747b8864caae36c5d05883976a3ae5", "patch": "diff --git a/pyvista/core/grid.py b/pyvista/core/grid.py\n--- a/pyvista/core/grid.py\n+++ b/pyvista/core/grid.py\n@@ -135,23 +135,30 @@ def __init__(self, *args, check_duplicates=False, deep=False, **kwargs):\n                     self.shallow_copy(args[0])\n             elif isinstance(args[0], (str, pathlib.Path)):\n                 self._from_file(args[0], **kwargs)\n-            elif isinstance(args[0], np.ndarray):\n-                self._from_arrays(args[0], None, None, check_duplicates)\n+            elif isinstance(args[0], (np.ndarray, Sequence)):\n+                self._from_arrays(np.asanyarray(args[0]), None, None, check_duplicates)\n             else:\n                 raise TypeError(f'Type ({type(args[0])}) not understood by `RectilinearGrid`')\n \n         elif len(args) == 3 or len(args) == 2:\n-            arg0_is_arr = isinstance(args[0], np.ndarray)\n-            arg1_is_arr = isinstance(args[1], np.ndarray)\n+            arg0_is_arr = isinstance(args[0], (np.ndarray, Sequence))\n+            arg1_is_arr = isinstance(args[1], (np.ndarray, Sequence))\n             if len(args) == 3:\n-                arg2_is_arr = isinstance(args[2], np.ndarray)\n+                arg2_is_arr = isinstance(args[2], (np.ndarray, Sequence))\n             else:\n                 arg2_is_arr = False\n \n             if all([arg0_is_arr, arg1_is_arr, arg2_is_arr]):\n-                self._from_arrays(args[0], args[1], args[2], check_duplicates)\n+                self._from_arrays(\n+                    np.asanyarray(args[0]),\n+                    np.asanyarray(args[1]),\n+                    np.asanyarray(args[2]),\n+                    check_duplicates,\n+                )\n             elif all([arg0_is_arr, arg1_is_arr]):\n-                self._from_arrays(args[0], args[1], None, check_duplicates)\n+                self._from_arrays(\n+                    np.asanyarray(args[0]), np.asanyarray(args[1]), None, check_duplicates\n+                )\n             else:\n                 raise TypeError(\"Arguments not understood by `RectilinearGrid`.\")\n \n", "test_patch": "diff --git a/tests/test_grid.py b/tests/test_grid.py\n--- a/tests/test_grid.py\n+++ b/tests/test_grid.py\n@@ -735,6 +735,21 @@ def test_create_rectilinear_grid_from_specs():\n     assert grid.n_cells == 9 * 3 * 19\n     assert grid.n_points == 10 * 4 * 20\n     assert grid.bounds == (-10.0, 8.0, -10.0, 5.0, -10.0, 9.0)\n+\n+    # with Sequence\n+    xrng = [0, 1]\n+    yrng = [0, 1, 2]\n+    zrng = [0, 1, 2, 3]\n+    grid = pyvista.RectilinearGrid(xrng)\n+    assert grid.n_cells == 1\n+    assert grid.n_points == 2\n+    grid = pyvista.RectilinearGrid(xrng, yrng)\n+    assert grid.n_cells == 2\n+    assert grid.n_points == 6\n+    grid = pyvista.RectilinearGrid(xrng, yrng, zrng)\n+    assert grid.n_cells == 6\n+    assert grid.n_points == 24\n+\n     # 2D example\n     cell_spacings = np.array([1.0, 1.0, 2.0, 2.0, 5.0, 10.0])\n     x_coordinates = np.cumsum(cell_spacings)\n", "problem_statement": "Rectilinear grid does not allow Sequences as inputs\n### Describe the bug, what's wrong, and what you expected.\r\n\r\nRectilinear grid gives an error when `Sequence`s are passed in, but `ndarray` are ok.\r\n\r\n### Steps to reproduce the bug.\r\n\r\nThis doesn't work\r\n```python\r\nimport pyvista as pv\r\npv.RectilinearGrid([0, 1], [0, 1], [0, 1])\r\n```\r\n\r\nThis works\r\n```py\r\nimport pyvista as pv\r\nimport numpy as np\r\npv.RectilinearGrid(np.ndarray([0, 1]), np.ndarray([0, 1]), np.ndarray([0, 1]))\r\n```\r\n### System Information\r\n\r\n```shell\r\n--------------------------------------------------------------------------------\r\n  Date: Wed Apr 19 20:15:10 2023 UTC\r\n\r\n                OS : Linux\r\n            CPU(s) : 2\r\n           Machine : x86_64\r\n      Architecture : 64bit\r\n       Environment : IPython\r\n        GPU Vendor : Mesa/X.org\r\n      GPU Renderer : llvmpipe (LLVM 11.0.1, 256 bits)\r\n       GPU Version : 4.5 (Core Profile) Mesa 20.3.5\r\n\r\n  Python 3.11.2 (main, Mar 23 2023, 17:12:29) [GCC 10.2.1 20210110]\r\n\r\n           pyvista : 0.38.5\r\n               vtk : 9.2.6\r\n             numpy : 1.24.2\r\n           imageio : 2.27.0\r\n            scooby : 0.7.1\r\n             pooch : v1.7.0\r\n        matplotlib : 3.7.1\r\n           IPython : 8.12.0\r\n--------------------------------------------------------------------------------\r\n```\r\n\r\n\r\n### Screenshots\r\n\r\n_No response_\n", "hints_text": "", "created_at": "2023-04-21T13:47:31Z", "version": "0.39", "FAIL_TO_PASS": ["tests/test_grid.py::test_create_rectilinear_grid_from_specs"], "PASS_TO_PASS": ["tests/test_grid.py::test_volume", "tests/test_grid.py::test_init_from_polydata", "tests/test_grid.py::test_init_from_structured", "tests/test_grid.py::test_init_from_unstructured", "tests/test_grid.py::test_init_from_numpy_arrays", "tests/test_grid.py::test_init_bad_input", "tests/test_grid.py::test_init_from_arrays[False]", "tests/test_grid.py::test_init_from_arrays[True]", "tests/test_grid.py::test_init_from_dict[False-False]", "tests/test_grid.py::test_init_from_dict[False-True]", "tests/test_grid.py::test_init_from_dict[True-False]", "tests/test_grid.py::test_init_from_dict[True-True]", "tests/test_grid.py::test_init_polyhedron", "tests/test_grid.py::test_cells_dict_hexbeam_file", "tests/test_grid.py::test_cells_dict_variable_length", "tests/test_grid.py::test_cells_dict_empty_grid", "tests/test_grid.py::test_cells_dict_alternating_cells", "tests/test_grid.py::test_destructor", "tests/test_grid.py::test_surface_indices", "tests/test_grid.py::test_extract_feature_edges", "tests/test_grid.py::test_triangulate_inplace", "tests/test_grid.py::test_save[.vtu-True]", "tests/test_grid.py::test_save[.vtu-False]", "tests/test_grid.py::test_save[.vtk-True]", "tests/test_grid.py::test_save[.vtk-False]", "tests/test_grid.py::test_pathlib_read_write", "tests/test_grid.py::test_init_bad_filename", "tests/test_grid.py::test_save_bad_extension", "tests/test_grid.py::test_linear_copy", "tests/test_grid.py::test_linear_copy_surf_elem", "tests/test_grid.py::test_extract_cells[True]", "tests/test_grid.py::test_extract_cells[False]", "tests/test_grid.py::test_merge", "tests/test_grid.py::test_merge_not_main", "tests/test_grid.py::test_merge_list", "tests/test_grid.py::test_merge_invalid", "tests/test_grid.py::test_init_structured_raise", "tests/test_grid.py::test_init_structured", "tests/test_grid.py::test_no_copy_polydata_init", "tests/test_grid.py::test_no_copy_polydata_points_setter", "tests/test_grid.py::test_no_copy_structured_mesh_init", "tests/test_grid.py::test_no_copy_structured_mesh_points_setter", "tests/test_grid.py::test_no_copy_pointset_init", "tests/test_grid.py::test_no_copy_pointset_points_setter", "tests/test_grid.py::test_no_copy_unstructured_grid_points_setter", "tests/test_grid.py::test_no_copy_rectilinear_grid", "tests/test_grid.py::test_grid_repr", "tests/test_grid.py::test_slice_structured", "tests/test_grid.py::test_invalid_init_structured", "tests/test_grid.py::test_save_structured[.vtk-True]", "tests/test_grid.py::test_save_structured[.vtk-False]", "tests/test_grid.py::test_save_structured[.vts-True]", "tests/test_grid.py::test_save_structured[.vts-False]", "tests/test_grid.py::test_load_structured_bad_filename", "tests/test_grid.py::test_instantiate_by_filename", "tests/test_grid.py::test_create_rectilinear_after_init", "tests/test_grid.py::test_create_rectilinear_grid_from_file", "tests/test_grid.py::test_read_rectilinear_grid_from_file", "tests/test_grid.py::test_read_rectilinear_grid_from_pathlib", "tests/test_grid.py::test_raise_rectilinear_grid_non_unique", "tests/test_grid.py::test_cast_rectilinear_grid", "tests/test_grid.py::test_create_uniform_grid_from_specs", "tests/test_grid.py::test_uniform_grid_invald_args", "tests/test_grid.py::test_uniform_setters", "tests/test_grid.py::test_create_uniform_grid_from_file", "tests/test_grid.py::test_read_uniform_grid_from_file", "tests/test_grid.py::test_read_uniform_grid_from_pathlib", "tests/test_grid.py::test_cast_uniform_to_structured", "tests/test_grid.py::test_cast_uniform_to_rectilinear", "tests/test_grid.py::test_uniform_grid_to_tetrahedra", "tests/test_grid.py::test_fft_and_rfft", "tests/test_grid.py::test_fft_low_pass", "tests/test_grid.py::test_fft_high_pass", "tests/test_grid.py::test_save_rectilinear[.vtk-True]", "tests/test_grid.py::test_save_rectilinear[.vtk-False]", "tests/test_grid.py::test_save_rectilinear[.vtr-True]", "tests/test_grid.py::test_save_rectilinear[.vtr-False]", "tests/test_grid.py::test_save_uniform[.vtk-True]", "tests/test_grid.py::test_save_uniform[.vtk-False]", "tests/test_grid.py::test_save_uniform[.vti-True]", "tests/test_grid.py::test_save_uniform[.vti-False]", "tests/test_grid.py::test_grid_points", "tests/test_grid.py::test_grid_extract_selection_points", "tests/test_grid.py::test_gaussian_smooth", "tests/test_grid.py::test_remove_cells[ind0]", "tests/test_grid.py::test_remove_cells[ind1]", "tests/test_grid.py::test_remove_cells[ind2]", "tests/test_grid.py::test_remove_cells_not_inplace[ind0]", "tests/test_grid.py::test_remove_cells_not_inplace[ind1]", "tests/test_grid.py::test_remove_cells_not_inplace[ind2]", "tests/test_grid.py::test_remove_cells_invalid", "tests/test_grid.py::test_hide_cells[ind0]", "tests/test_grid.py::test_hide_cells[ind1]", "tests/test_grid.py::test_hide_cells[ind2]", "tests/test_grid.py::test_hide_points[ind0]", "tests/test_grid.py::test_hide_points[ind1]", "tests/test_grid.py::test_hide_points[ind2]", "tests/test_grid.py::test_set_extent", "tests/test_grid.py::test_UnstructuredGrid_cast_to_explicit_structured_grid", "tests/test_grid.py::test_ExplicitStructuredGrid_init", "tests/test_grid.py::test_ExplicitStructuredGrid_cast_to_unstructured_grid", "tests/test_grid.py::test_ExplicitStructuredGrid_save", "tests/test_grid.py::test_ExplicitStructuredGrid_hide_cells", "tests/test_grid.py::test_ExplicitStructuredGrid_show_cells", "tests/test_grid.py::test_ExplicitStructuredGrid_dimensions", "tests/test_grid.py::test_ExplicitStructuredGrid_visible_bounds", "tests/test_grid.py::test_ExplicitStructuredGrid_cell_id", "tests/test_grid.py::test_ExplicitStructuredGrid_cell_coords", "tests/test_grid.py::test_ExplicitStructuredGrid_neighbors", "tests/test_grid.py::test_ExplicitStructuredGrid_compute_connectivity", "tests/test_grid.py::test_ExplicitStructuredGrid_compute_connections", "tests/test_grid.py::test_ExplicitStructuredGrid_raise_init", "tests/test_grid.py::test_copy_no_copy_wrap_object", "tests/test_grid.py::test_copy_no_copy_wrap_object_vtk9"], "environment_setup_commit": "4c2d1aed10b1600d520271beba8579c71433e808"}, {"repo": "marshmallow-code/marshmallow", "instance_id": "marshmallow-code__marshmallow-1164", "base_commit": "1e26d14facab213df5009300b997481aa43df80a", "patch": "diff --git a/src/marshmallow/fields.py b/src/marshmallow/fields.py\n--- a/src/marshmallow/fields.py\n+++ b/src/marshmallow/fields.py\n@@ -443,6 +443,8 @@ def _serialize(self, nested_obj, attr, obj):\n         schema = self.schema\n         if nested_obj is None:\n             return None\n+        if self.many and utils.is_iterable_but_not_string(nested_obj):\n+            nested_obj = list(nested_obj)\n         if not self.__updated_fields:\n             schema._update_fields(obj=nested_obj, many=self.many)\n             self.__updated_fields = True\ndiff --git a/src/marshmallow/schema.py b/src/marshmallow/schema.py\n--- a/src/marshmallow/schema.py\n+++ b/src/marshmallow/schema.py\n@@ -816,23 +816,14 @@ def __filter_fields(self, field_names, obj, many=False):\n \n         :param set field_names: Field names to include in the final\n             return dictionary.\n+        :param object|Mapping|list obj The object to base filtered fields on.\n         :returns: An dict of field_name:field_obj pairs.\n         \"\"\"\n         if obj and many:\n-            try:  # Homogeneous collection\n-                # Prefer getitem over iter to prevent breaking serialization\n-                # of objects for which iter will modify position in the collection\n-                # e.g. Pymongo cursors\n-                if hasattr(obj, '__getitem__') and callable(getattr(obj, '__getitem__')):\n-                    try:\n-                        obj_prototype = obj[0]\n-                    except KeyError:\n-                        obj_prototype = next(iter(obj))\n-                else:\n-                    obj_prototype = next(iter(obj))\n-            except (StopIteration, IndexError):  # Nothing to serialize\n+            try:  # list\n+                obj = obj[0]\n+            except IndexError:  # Nothing to serialize\n                 return dict((k, v) for k, v in self.declared_fields.items() if k in field_names)\n-            obj = obj_prototype\n         ret = self.dict_class()\n         for key in field_names:\n             if key in self.declared_fields:\n", "test_patch": "diff --git a/tests/test_serialization.py b/tests/test_serialization.py\n--- a/tests/test_serialization.py\n+++ b/tests/test_serialization.py\n@@ -782,3 +782,21 @@ class ValueSchema(Schema):\n \n     serialized = ValueSchema(many=True).dump(slice).data\n     assert serialized == values\n+\n+\n+# https://github.com/marshmallow-code/marshmallow/issues/1163\n+def test_nested_field_many_serializing_generator():\n+    class MySchema(Schema):\n+        name = fields.Str()\n+\n+    class OtherSchema(Schema):\n+        objects = fields.Nested(MySchema, many=True)\n+\n+    def gen():\n+        yield {'name': 'foo'}\n+        yield {'name': 'bar'}\n+\n+    obj = {'objects': gen()}\n+    data, _ = OtherSchema().dump(obj)\n+\n+    assert data.get('objects') == [{'name': 'foo'}, {'name': 'bar'}]\n", "problem_statement": "2.x: Nested(many=True) eats first element from generator value when dumping\nAs reproduced in Python 3.6.8:\r\n\r\n```py\r\nfrom marshmallow import Schema, fields\r\n\r\nclass O(Schema):\r\n    i = fields.Int()\r\n\r\nclass P(Schema):\r\n    os = fields.Nested(O, many=True)\r\n\r\ndef gen():\r\n    yield {'i': 1}\r\n    yield {'i': 0}\r\n\r\np = P()\r\np.dump({'os': gen()})\r\n# MarshalResult(data={'os': [{'i': 0}]}, errors={})\r\n```\r\n\r\nProblematic code is here:\r\n\r\nhttps://github.com/marshmallow-code/marshmallow/blob/2.x-line/src/marshmallow/fields.py#L447\r\n\r\nAnd here:\r\n\r\nhttps://github.com/marshmallow-code/marshmallow/blob/2.x-line/src/marshmallow/schema.py#L832\r\n\r\nThe easiest solution would be to cast `nested_obj` to list before calling `schema._update_fields`, just like a normal Schema with `many=True` does.\n", "hints_text": "I confirmed that this is no longer an issue in marshmallow 3. I was able to reproduce this with python 2 and 3 using the latest version of marshmallow 2.\n`next(iter(...))` is not a safe operation for generators.\r\n\r\n```py\r\ndef gen():\r\n    yield 1\r\n    yield 2\r\n\r\nx = gen()\r\nnext(iter(x))\r\n# 1\r\nlist(x)\r\n# [2]\r\n```\r\n\r\nI suspect `list` would be an acceptable solution. If it was a performance concern we could use `itertools.tee` to copy the generator before peeking at the first item.\n`next(iter(...))` is apparently fine because `obj` is guaranteed to be a list here:\r\n\r\nhttps://github.com/marshmallow-code/marshmallow/blob/2.x-line/src/marshmallow/schema.py#L489\r\n\r\nIt's just that usage of `Schema._update_fileds` in `Nested` ignores the requirement.\r\n", "created_at": "2019-03-01T17:03:05Z", "version": "2.18", "FAIL_TO_PASS": ["tests/test_serialization.py::test_nested_field_many_serializing_generator"], "PASS_TO_PASS": ["tests/test_serialization.py::TestFieldSerialization::test_default", "tests/test_serialization.py::TestFieldSerialization::test_number[42-42.0]", "tests/test_serialization.py::TestFieldSerialization::test_number[0-0.0]", "tests/test_serialization.py::TestFieldSerialization::test_number[None-None]", "tests/test_serialization.py::TestFieldSerialization::test_number_as_string", "tests/test_serialization.py::TestFieldSerialization::test_number_as_string_passed_none", "tests/test_serialization.py::TestFieldSerialization::test_callable_default", "tests/test_serialization.py::TestFieldSerialization::test_function_field_passed_func", "tests/test_serialization.py::TestFieldSerialization::test_function_field_passed_serialize", "tests/test_serialization.py::TestFieldSerialization::test_function_field_passed_func_is_deprecated", "tests/test_serialization.py::TestFieldSerialization::test_function_field_passed_serialize_with_context", "tests/test_serialization.py::TestFieldSerialization::test_function_field_passed_uncallable_object", "tests/test_serialization.py::TestFieldSerialization::test_integer_field", "tests/test_serialization.py::TestFieldSerialization::test_integer_as_string_field", "tests/test_serialization.py::TestFieldSerialization::test_integer_field_default", "tests/test_serialization.py::TestFieldSerialization::test_integer_field_default_set_to_none", "tests/test_serialization.py::TestFieldSerialization::test_callable_field", "tests/test_serialization.py::TestFieldSerialization::test_uuid_field", "tests/test_serialization.py::TestFieldSerialization::test_decimal_field", "tests/test_serialization.py::TestFieldSerialization::test_decimal_field_string", "tests/test_serialization.py::TestFieldSerialization::test_decimal_field_special_values", "tests/test_serialization.py::TestFieldSerialization::test_decimal_field_special_values_not_permitted", "tests/test_serialization.py::TestFieldSerialization::test_decimal_field_fixed_point_representation", "tests/test_serialization.py::TestFieldSerialization::test_boolean_field_serialization", "tests/test_serialization.py::TestFieldSerialization::test_function_with_uncallable_param", "tests/test_serialization.py::TestFieldSerialization::test_email_field_validates", "tests/test_serialization.py::TestFieldSerialization::test_email_field_serialize_none", "tests/test_serialization.py::TestFieldSerialization::test_dict_field_serialize_none", "tests/test_serialization.py::TestFieldSerialization::test_dict_field_invalid_dict_but_okay", "tests/test_serialization.py::TestFieldSerialization::test_dict_field_serialize", "tests/test_serialization.py::TestFieldSerialization::test_dict_field_serialize_ordereddict", "tests/test_serialization.py::TestFieldSerialization::test_url_field_serialize_none", "tests/test_serialization.py::TestFieldSerialization::test_url_field_validates", "tests/test_serialization.py::TestFieldSerialization::test_method_field_with_method_missing", "tests/test_serialization.py::TestFieldSerialization::test_method_field_with_uncallable_attribute", "tests/test_serialization.py::TestFieldSerialization::test_method_prefers_serialize_over_method_name", "tests/test_serialization.py::TestFieldSerialization::test_method_with_no_serialize_is_missing", "tests/test_serialization.py::TestFieldSerialization::test_serialize_with_dump_to_param", "tests/test_serialization.py::TestFieldSerialization::test_serialize_with_attribute_and_dump_to_uses_dump_to", "tests/test_serialization.py::TestFieldSerialization::test_datetime_serializes_to_iso_by_default", "tests/test_serialization.py::TestFieldSerialization::test_datetime_invalid_serialization[invalid]", "tests/test_serialization.py::TestFieldSerialization::test_datetime_invalid_serialization[value1]", "tests/test_serialization.py::TestFieldSerialization::test_datetime_invalid_serialization[24]", "tests/test_serialization.py::TestFieldSerialization::test_datetime_field_rfc822[rfc]", "tests/test_serialization.py::TestFieldSerialization::test_datetime_field_rfc822[rfc822]", "tests/test_serialization.py::TestFieldSerialization::test_localdatetime_rfc_field", "tests/test_serialization.py::TestFieldSerialization::test_datetime_iso8601[iso]", "tests/test_serialization.py::TestFieldSerialization::test_datetime_iso8601[iso8601]", "tests/test_serialization.py::TestFieldSerialization::test_localdatetime_iso", "tests/test_serialization.py::TestFieldSerialization::test_datetime_format", "tests/test_serialization.py::TestFieldSerialization::test_string_field", "tests/test_serialization.py::TestFieldSerialization::test_formattedstring_field", "tests/test_serialization.py::TestFieldSerialization::test_formattedstring_field_on_schema", "tests/test_serialization.py::TestFieldSerialization::test_string_field_default_to_empty_string", "tests/test_serialization.py::TestFieldSerialization::test_time_field", "tests/test_serialization.py::TestFieldSerialization::test_invalid_time_field_serialization[badvalue]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_time_field_serialization[]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_time_field_serialization[in_data2]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_time_field_serialization[42]", "tests/test_serialization.py::TestFieldSerialization::test_date_field", "tests/test_serialization.py::TestFieldSerialization::test_invalid_date_field_serialization[badvalue]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_date_field_serialization[]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_date_field_serialization[in_data2]", "tests/test_serialization.py::TestFieldSerialization::test_invalid_date_field_serialization[42]", "tests/test_serialization.py::TestFieldSerialization::test_timedelta_field", "tests/test_serialization.py::TestFieldSerialization::test_datetime_list_field", "tests/test_serialization.py::TestFieldSerialization::test_list_field_with_error", "tests/test_serialization.py::TestFieldSerialization::test_datetime_list_serialize_single_value", "tests/test_serialization.py::TestFieldSerialization::test_list_field_serialize_none_returns_none", "tests/test_serialization.py::TestFieldSerialization::test_list_field_respect_inner_attribute", "tests/test_serialization.py::TestFieldSerialization::test_list_field_respect_inner_attribute_single_value", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_generator_single_value", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_generators_multiple_values", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_generators_error", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_generators_empty_generator_returns_none_for_every_non_returning_yield_statement", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_set", "tests/test_serialization.py::TestFieldSerialization::test_list_field_work_with_custom_class_with_iterator_protocol", "tests/test_serialization.py::TestFieldSerialization::test_bad_list_field", "tests/test_serialization.py::TestFieldSerialization::test_serialize_does_not_apply_validators", "tests/test_serialization.py::TestFieldSerialization::test_constant_field_serialization", "tests/test_serialization.py::TestFieldSerialization::test_constant_is_always_included_in_serialized_data", "tests/test_serialization.py::TestFieldSerialization::test_constant_field_serialize_when_omitted", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[String]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Integer]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Boolean]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Float]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Number]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[DateTime]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[LocalDateTime]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Time]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Date]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[TimeDelta]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Dict]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Url]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Email]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[FormattedString]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[UUID]", "tests/test_serialization.py::TestFieldSerialization::test_all_fields_serialize_none_to_none[Decimal]", "tests/test_serialization.py::test_serializing_named_tuple", "tests/test_serialization.py::test_serializing_named_tuple_with_meta", "tests/test_serialization.py::test_serializing_slice"], "environment_setup_commit": "1e26d14facab213df5009300b997481aa43df80a"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1334", "base_commit": "24a86b316441ac3a46e569779627e24482786a8a", "patch": "diff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -3,6 +3,7 @@\n \n import datetime\n from decimal import Decimal\n+from math import floor, isfinite, log10\n import platform\n import re\n import sys\n@@ -329,15 +330,139 @@ def __reduce_ex__(self, protocol: int) -> Union[str, Tuple[Any, ...]]:\n             )\n \n \n+# Regex to match strings that represent valid DICOM decimal strings (DS)\n+_DS_REGEX = re.compile(r'\\s*[\\+\\-]?\\d+(\\.\\d+)?([eE][\\+\\-]?\\d+)?\\s*$')\n+\n+\n+def is_valid_ds(s: str) -> bool:\n+    \"\"\"Check whether this string is a valid decimal string.\n+\n+    Valid decimal strings must be 16 characters or fewer, and contain only\n+    characters from a limited set.\n+\n+    Parameters\n+    ----------\n+    s: str\n+        String to test.\n+\n+    Returns\n+    -------\n+    bool\n+        True if the string is a valid decimal string. Otherwise False.\n+    \"\"\"\n+    # Check that the length is within the limits\n+    if len(s) > 16:\n+        return False\n+\n+    return _DS_REGEX.match(s) is not None\n+\n+\n+def format_number_as_ds(val: Union[float, Decimal]) -> str:\n+    \"\"\"Truncate a float's representation to give a valid Decimal String (DS).\n+\n+    DICOM's decimal string (DS) representation is limited to strings with 16\n+    characters and a limited set of characters. This function represents a\n+    float that satisfies these constraints while retaining as much\n+    precision as possible. Some floats are represented using scientific\n+    notation to make more efficient use of the limited number of characters.\n+\n+    Note that this will incur a loss of precision if the number cannot be\n+    represented with 16 characters. Furthermore, non-finite floats (infs and\n+    nans) cannot be represented as decimal strings and will cause an error to\n+    be raised.\n+\n+    Parameters\n+    ----------\n+    val: Union[float, Decimal]\n+        The floating point value whose representation is required.\n+\n+    Returns\n+    -------\n+    str\n+        String representation of the float satisfying the constraints of the\n+        decimal string representation.\n+\n+    Raises\n+    ------\n+    ValueError\n+        If val does not represent a finite value\n+\n+    \"\"\"\n+    if not isinstance(val, (float, Decimal)):\n+        raise TypeError(\"'val' must be of type float or decimal.Decimal\")\n+    if not isfinite(val):\n+        raise ValueError(\n+            \"Cannot encode non-finite floats as DICOM decimal strings. \"\n+            f\"Got '{val}'\"\n+        )\n+\n+    valstr = str(val)\n+\n+    # In the simple case, the default python string representation\n+    # will do\n+    if len(valstr) <= 16:\n+        return valstr\n+\n+    # Decide whether to use scientific notation\n+    logval = log10(abs(val))\n+\n+    # Characters needed for '-' at start\n+    sign_chars = 1 if val < 0.0 else 0\n+\n+    # Numbers larger than 1e14 cannot be correctly represented by truncating\n+    # their string representations to 16 chars, e.g pi * 10^13 would become\n+    # '314159265358979.', which may not be universally understood. This limit\n+    # is 1e13 for negative numbers because of the minus sign.\n+    # For negative exponents, the point of equal precision between scientific\n+    # and standard notation is 1e-4 e.g. '0.00031415926535' and\n+    # '3.1415926535e-04' are both 16 chars\n+    use_scientific = logval < -4 or logval >= (14 - sign_chars)\n+\n+    if use_scientific:\n+        # In principle, we could have a number where the exponent\n+        # needs three digits to be represented (bigger than this cannot be\n+        # represented by floats). Due to floating point limitations\n+        # this is best checked for by doing the string conversion\n+        remaining_chars = 10 - sign_chars\n+        trunc_str = f'%.{remaining_chars}e' % val\n+        if len(trunc_str) > 16:\n+            trunc_str = f'%.{remaining_chars - 1}e' % val\n+        return trunc_str\n+    else:\n+        if logval >= 1.0:\n+            # chars remaining for digits after sign, digits left of '.' and '.'\n+            remaining_chars = 14 - sign_chars - int(floor(logval))\n+        else:\n+            remaining_chars = 14 - sign_chars\n+        return f'%.{remaining_chars}f' % val\n+\n+\n class DSfloat(float):\n     \"\"\"Store value for an element with VR **DS** as :class:`float`.\n \n     If constructed from an empty string, return the empty string,\n     not an instance of this class.\n \n+    Parameters\n+    ----------\n+    val: Union[str, int, float, Decimal]\n+        Value to store as a DS.\n+    auto_format: bool\n+        If True, automatically format the string representation of this\n+        number to ensure it satisfies the constraints in the DICOM standard.\n+        Note that this will lead to loss of precision for some numbers.\n+\n     \"\"\"\n+    def __new__(\n+            cls,\n+            val: Union[str, int, float, Decimal],\n+            auto_format: bool = False\n+    ) -> [_DSfloat]:\n+        return super().__new__(cls, val)\n+\n     def __init__(\n-        self, val: Union[str, int, float, Decimal]\n+        self, val: Union[str, int, float, Decimal],\n+        auto_format: bool = False\n     ) -> None:\n         \"\"\"Store the original string if one given, for exact write-out of same\n         value later.\n@@ -350,28 +475,70 @@ def __init__(\n         elif isinstance(val, (DSfloat, DSdecimal)) and has_attribute:\n             self.original_string = val.original_string\n \n+        self.auto_format = auto_format\n+        if self.auto_format:\n+            # If auto_format is True, keep the float value the same, but change\n+            # the string representation stored in original_string if necessary\n+            if hasattr(self, 'original_string'):\n+                if not is_valid_ds(self.original_string):\n+                    self.original_string = format_number_as_ds(\n+                        float(self.original_string)\n+                    )\n+            else:\n+                self.original_string = format_number_as_ds(self)\n+\n+        if config.enforce_valid_values and not self.auto_format:\n+            if len(repr(self).strip('\"')) > 16:\n+                raise OverflowError(\n+                    \"Values for elements with a VR of 'DS' must be <= 16 \"\n+                    \"characters long, but the float provided requires > 16 \"\n+                    \"characters to be accurately represented. Use a smaller \"\n+                    \"string, set 'config.enforce_valid_values' to False to \"\n+                    \"override the length check, or explicitly construct a DS \"\n+                    \"object with 'auto_format' set to True\"\n+                )\n+            if not is_valid_ds(repr(self).strip('\"')):\n+                # This will catch nan and inf\n+                raise ValueError(\n+                    f'Value \"{str(self)}\" is not valid for elements with a VR '\n+                    'of DS'\n+                )\n+\n     def __str__(self) -> str:\n-        if hasattr(self, 'original_string'):\n+        if hasattr(self, 'original_string') and not self.auto_format:\n             return self.original_string\n \n         # Issue #937 (Python 3.8 compatibility)\n         return repr(self)[1:-1]\n \n     def __repr__(self) -> str:\n+        if self.auto_format and hasattr(self, 'original_string'):\n+            return f'\"{self.original_string}\"'\n         return f'\"{super().__repr__()}\"'\n \n \n class DSdecimal(Decimal):\n     \"\"\"Store value for an element with VR **DS** as :class:`decimal.Decimal`.\n \n+    Parameters\n+    ----------\n+    val: Union[str, int, float, Decimal]\n+        Value to store as a DS.\n+    auto_format: bool\n+        If True, automatically format the string representation of this\n+        number to ensure it satisfies the constraints in the DICOM standard.\n+        Note that this will lead to loss of precision for some numbers.\n+\n     Notes\n     -----\n     If constructed from an empty string, returns the empty string, not an\n     instance of this class.\n+\n     \"\"\"\n     def __new__(\n         cls: Type[_DSdecimal],\n-        val: Union[str, int, float, Decimal]\n+        val: Union[str, int, float, Decimal],\n+        auto_format: bool = False\n     ) -> Optional[_DSdecimal]:\n         \"\"\"Create an instance of DS object, or return a blank string if one is\n         passed in, e.g. from a type 2 DICOM blank value.\n@@ -395,19 +562,13 @@ def __new__(\n                 return None\n \n         val = super().__new__(cls, val)\n-        if len(str(val)) > 16 and config.enforce_valid_values:\n-            raise OverflowError(\n-                \"Values for elements with a VR of 'DS' values must be <= 16 \"\n-                \"characters long. Use a smaller string, set \"\n-                \"'config.enforce_valid_values' to False to override the \"\n-                \"length check, or use 'Decimal.quantize()' and initialize \"\n-                \"with a 'Decimal' instance.\"\n-            )\n \n         return val\n \n     def __init__(\n-        self, val: Union[str, int, float, Decimal]\n+        self,\n+        val: Union[str, int, float, Decimal],\n+        auto_format: bool = False\n     ) -> None:\n         \"\"\"Store the original string if one given, for exact write-out of same\n         value later. E.g. if set ``'1.23e2'``, :class:`~decimal.Decimal` would\n@@ -421,6 +582,35 @@ def __init__(\n         elif isinstance(val, (DSfloat, DSdecimal)) and has_str:\n             self.original_string = val.original_string\n \n+        self.auto_format = auto_format\n+        if self.auto_format:\n+            # If auto_format is True, keep the float value the same, but change\n+            # the string representation stored in original_string if necessary\n+            if hasattr(self, 'original_string'):\n+                if not is_valid_ds(self.original_string):\n+                    self.original_string = format_number_as_ds(\n+                        float(self.original_string)\n+                    )\n+            else:\n+                self.original_string = format_number_as_ds(self)\n+\n+        if config.enforce_valid_values:\n+            if len(repr(self).strip('\"')) > 16:\n+                raise OverflowError(\n+                    \"Values for elements with a VR of 'DS' values must be \"\n+                    \"<= 16 characters long. Use a smaller string, set \"\n+                    \"'config.enforce_valid_values' to False to override the \"\n+                    \"length check, use 'Decimal.quantize()' and initialize \"\n+                    \"with a 'Decimal' instance, or explicitly construct a DS \"\n+                    \"instance with 'auto_format' set to True\"\n+                )\n+            if not is_valid_ds(repr(self).strip('\"')):\n+                # This will catch nan and inf\n+                raise ValueError(\n+                    f'Value \"{str(self)}\" is not valid for elements with a VR '\n+                    'of DS'\n+                )\n+\n     def __str__(self) -> str:\n         has_str = hasattr(self, 'original_string')\n         if has_str and len(self.original_string) <= 16:\n@@ -429,6 +619,8 @@ def __str__(self) -> str:\n         return super().__str__()\n \n     def __repr__(self) -> str:\n+        if self.auto_format and hasattr(self, 'original_string'):\n+            return f'\"{self.original_string}\"'\n         return f'\"{str(self)}\"'\n \n \n@@ -440,7 +632,8 @@ def __repr__(self) -> str:\n \n \n def DS(\n-    val: Union[None, str, int, float, Decimal]\n+    val: Union[None, str, int, float, Decimal],\n+    auto_format: bool = False\n ) -> Union[None, str, DSfloat, DSdecimal]:\n     \"\"\"Factory function for creating DS class instances.\n \n@@ -458,7 +651,7 @@ def DS(\n     if val == '' or val is None:\n         return val\n \n-    return DSclass(val)\n+    return DSclass(val, auto_format=auto_format)\n \n \n class IS(int):\n", "test_patch": "diff --git a/pydicom/tests/test_valuerep.py b/pydicom/tests/test_valuerep.py\n--- a/pydicom/tests/test_valuerep.py\n+++ b/pydicom/tests/test_valuerep.py\n@@ -9,7 +9,9 @@\n     import cPickle as pickle\n except ImportError:\n     import pickle\n+import math\n import sys\n+from typing import Union\n \n from pydicom.tag import Tag\n from pydicom.values import convert_value\n@@ -29,6 +31,27 @@\n default_encoding = \"iso8859\"\n \n \n+@pytest.fixture()\n+def enforce_valid_true_fixture():\n+    \"\"\"Fixture to run tests with enforce_valid_values True and ensure it is\n+       reset afterwards regardless of whether test succeeds.\"\"\"\n+    enforce_flag_original = config.enforce_valid_values\n+    config.enforce_valid_values = True\n+    yield\n+    config.enforce_valid_values = enforce_flag_original\n+\n+\n+@pytest.fixture(params=(True, False))\n+def enforce_valid_both_fixture(request):\n+    \"\"\"Fixture to run tests with enforce_valid_values with both True and False\n+       and ensure it is reset afterwards regardless of whether test succeeds.\n+    \"\"\"\n+    enforce_flag_original = config.enforce_valid_values\n+    config.enforce_valid_values = request.param\n+    yield\n+    config.enforce_valid_values = enforce_flag_original\n+\n+\n class TestTM:\n     \"\"\"Unit tests for pickling TM\"\"\"\n     def test_pickling(self):\n@@ -191,6 +214,115 @@ def test_new_obj_conversion(self):\n             pydicom.valuerep.DA(123456)\n \n \n+class TestIsValidDS:\n+    \"\"\"Unit tests for the is_valid_ds function.\"\"\"\n+    @pytest.mark.parametrize(\n+        's',\n+        [\n+            '1',\n+            '3.14159265358979',\n+            '-1234.456e78',\n+            '1.234E-5',\n+            '1.234E+5',\n+            '+1',\n+            '    42',  # leading spaces allowed\n+            '42    ',  # trailing spaces allowed\n+        ]\n+    )\n+    def test_valid(self, s: str):\n+        \"\"\"Various valid decimal strings.\"\"\"\n+        assert pydicom.valuerep.is_valid_ds(s)\n+\n+    @pytest.mark.parametrize(\n+        's',\n+        [\n+            'nan',\n+            '-inf',\n+            '3.141592653589793',  # too long\n+            '1,000',              # no commas\n+            '1 000',              # no embedded spaces\n+            '127.0.0.1',          # not a number\n+            '1.e',                # not a number\n+            '',\n+        ]\n+    )\n+    def test_invalid(self, s: str):\n+        \"\"\"Various invalid decimal strings.\"\"\"\n+        assert not pydicom.valuerep.is_valid_ds(s)\n+\n+\n+class TestTruncateFloatForDS:\n+    \"\"\"Unit tests for float truncation function\"\"\"\n+    def check_valid(self, s: str) -> bool:\n+        # Use the pydicom test function\n+        if not pydicom.valuerep.is_valid_ds(s):\n+            return False\n+\n+        # Disallow floats ending in '.' since this may not be correctly\n+        # interpreted\n+        if s.endswith('.'):\n+            return False\n+\n+        # Otherwise return True\n+        return True\n+\n+    @pytest.mark.parametrize(\n+        'val,expected_str',\n+        [\n+            [1.0, \"1.0\"],\n+            [0.0, \"0.0\"],\n+            [-0.0, \"-0.0\"],\n+            [0.123, \"0.123\"],\n+            [-0.321, \"-0.321\"],\n+            [0.00001, \"1e-05\"],\n+            [3.14159265358979323846, '3.14159265358979'],\n+            [-3.14159265358979323846, '-3.1415926535898'],\n+            [5.3859401928763739403e-7, '5.3859401929e-07'],\n+            [-5.3859401928763739403e-7, '-5.385940193e-07'],\n+            [1.2342534378125532912998323e10, '12342534378.1255'],\n+            [6.40708699858767842501238e13, '64070869985876.8'],\n+            [1.7976931348623157e+308, '1.797693135e+308'],\n+        ]\n+    )\n+    def test_auto_format(self, val: float, expected_str: str):\n+        \"\"\"Test truncation of some basic values.\"\"\"\n+        assert pydicom.valuerep.format_number_as_ds(val) == expected_str\n+\n+    @pytest.mark.parametrize(\n+        'exp', [-101, -100, 100, 101] + list(range(-16, 17))\n+    )\n+    def test_powers_of_pi(self, exp: int):\n+        \"\"\"Raise pi to various powers to test truncation.\"\"\"\n+        val = math.pi * 10 ** exp\n+        s = pydicom.valuerep.format_number_as_ds(val)\n+        assert self.check_valid(s)\n+\n+    @pytest.mark.parametrize(\n+        'exp', [-101, -100, 100, 101] + list(range(-16, 17))\n+    )\n+    def test_powers_of_negative_pi(self, exp: int):\n+        \"\"\"Raise negative pi to various powers to test truncation.\"\"\"\n+        val = -math.pi * 10 ** exp\n+        s = pydicom.valuerep.format_number_as_ds(val)\n+        assert self.check_valid(s)\n+\n+    @pytest.mark.parametrize(\n+        'val', [float('-nan'), float('nan'), float('-inf'), float('inf')]\n+    )\n+    def test_invalid(self, val: float):\n+        \"\"\"Test non-finite floating point numbers raise an error\"\"\"\n+        with pytest.raises(ValueError):\n+            pydicom.valuerep.format_number_as_ds(val)\n+\n+    def test_wrong_type(self):\n+        \"\"\"Test calling with a string raises an error\"\"\"\n+        with pytest.raises(\n+            TypeError,\n+            match=\"'val' must be of type float or decimal.Decimal\"\n+        ):\n+            pydicom.valuerep.format_number_as_ds('1.0')\n+\n+\n class TestDS:\n     \"\"\"Unit tests for DS values\"\"\"\n     def test_empty_value(self):\n@@ -249,6 +381,57 @@ def test_DSdecimal(self):\n         assert 1.2345 == y\n         assert \"1.2345\" == y.original_string\n \n+    def test_auto_format(self, enforce_valid_both_fixture):\n+        \"\"\"Test truncating floats\"\"\"\n+        x = pydicom.valuerep.DSfloat(math.pi, auto_format=True)\n+\n+        # Float representation should be unaltered by truncation\n+        assert x == math.pi\n+        # String representations should be correctly formatted\n+        assert str(x) == '3.14159265358979'\n+        assert repr(x) == '\"3.14159265358979\"'\n+\n+    def test_auto_format_invalid_string(self, enforce_valid_both_fixture):\n+        \"\"\"If the user supplies an invalid string, this should be formatted.\"\"\"\n+        x = pydicom.valuerep.DSfloat('3.141592653589793', auto_format=True)\n+\n+        # Float representation should be unaltered by truncation\n+        assert x == float('3.141592653589793')\n+        # String representations should be correctly formatted\n+        assert str(x) == '3.14159265358979'\n+        assert repr(x) == '\"3.14159265358979\"'\n+\n+    def test_auto_format_valid_string(self, enforce_valid_both_fixture):\n+        \"\"\"If the user supplies a valid string, this should not be altered.\"\"\"\n+        x = pydicom.valuerep.DSfloat('1.234e-1', auto_format=True)\n+\n+        # Float representation should be correct\n+        assert x == 0.1234\n+        # String representations should be unaltered\n+        assert str(x) == '1.234e-1'\n+        assert repr(x) == '\"1.234e-1\"'\n+\n+    def test_enforce_valid_values_length(self, enforce_valid_true_fixture):\n+        \"\"\"Test that errors are raised when length is too long.\"\"\"\n+        with pytest.raises(OverflowError):\n+            valuerep.DSfloat('3.141592653589793')\n+\n+    @pytest.mark.parametrize(\n+        'val',\n+        [\n+            'nan', '-nan', 'inf', '-inf', float('nan'), float('-nan'),\n+            float('-inf'), float('inf')\n+        ]\n+    )\n+    def test_enforce_valid_values_value(\n+        self,\n+        val: Union[float, str],\n+        enforce_valid_true_fixture\n+    ):\n+        \"\"\"Test that errors are raised when value is invalid.\"\"\"\n+        with pytest.raises(ValueError):\n+            valuerep.DSfloat(val)\n+\n \n class TestDSdecimal:\n     \"\"\"Unit tests for pickling DSdecimal\"\"\"\n@@ -297,6 +480,52 @@ def test_repr(self):\n         x = pydicom.valuerep.DSdecimal('1.2345')\n         assert '\"1.2345\"' == repr(x)\n \n+    def test_auto_format(self, enforce_valid_both_fixture):\n+        \"\"\"Test truncating decimal\"\"\"\n+        x = pydicom.valuerep.DSdecimal(Decimal(math.pi), auto_format=True)\n+\n+        # Decimal representation should be unaltered by truncation\n+        assert x == Decimal(math.pi)\n+        # String representations should be correctly formatted\n+        assert str(x) == '3.14159265358979'\n+        assert repr(x) == '\"3.14159265358979\"'\n+\n+    def test_auto_format_invalid_string(self, enforce_valid_both_fixture):\n+        \"\"\"If the user supplies an invalid string, this should be formatted.\"\"\"\n+        x = pydicom.valuerep.DSdecimal('3.141592653589793', auto_format=True)\n+\n+        # Decimal representation should be unaltered by truncation\n+        assert x == Decimal('3.141592653589793')\n+        # String representations should be correctly formatted\n+        assert str(x) == '3.14159265358979'\n+        assert repr(x) == '\"3.14159265358979\"'\n+\n+    @pytest.mark.parametrize(\n+        'val',\n+        [\n+            'NaN', '-NaN', 'Infinity', '-Infinity', Decimal('NaN'),\n+            Decimal('-NaN'), Decimal('-Infinity'), Decimal('Infinity')\n+        ]\n+    )\n+    def test_enforce_valid_values_value(\n+        self,\n+        val: Union[Decimal, str],\n+        enforce_valid_true_fixture\n+    ):\n+        \"\"\"Test that errors are raised when value is invalid.\"\"\"\n+        with pytest.raises(ValueError):\n+            valuerep.DSdecimal(val)\n+\n+    def test_auto_format_valid_string(self, enforce_valid_both_fixture):\n+        \"\"\"If the user supplies a valid string, this should not be altered.\"\"\"\n+        x = pydicom.valuerep.DSdecimal('1.234e-1', auto_format=True)\n+\n+        # Decimal representation should be correct\n+        assert x == Decimal('1.234e-1')\n+        # String representations should be unaltered\n+        assert str(x) == '1.234e-1'\n+        assert repr(x) == '\"1.234e-1\"'\n+\n \n class TestIS:\n     \"\"\"Unit tests for IS\"\"\"\n@@ -355,7 +584,6 @@ def test_str(self):\n         val = pydicom.valuerep.IS(\"1.0\")\n         assert \"1.0\" == str(val)\n \n-\n     def test_repr(self):\n         \"\"\"Test IS.__repr__().\"\"\"\n         val = pydicom.valuerep.IS(1)\n@@ -428,13 +656,6 @@ def test_valid_decimal_strings(self):\n         assert isinstance(ds, valuerep.DSdecimal)\n         assert len(str(ds)) <= 16\n \n-        # Now the input string is too long but decimal.Decimal can convert it\n-        # to a valid 16-character string\n-        long_str = \"-0.000000981338674\"\n-        ds = valuerep.DS(long_str)\n-        assert isinstance(ds, valuerep.DSdecimal)\n-        assert len(str(ds)) <= 16\n-\n     def test_invalid_decimal_strings(self, enforce_valid_values):\n         # Now the input string truly is invalid\n         invalid_string = \"-9.813386743e-006\"\n", "problem_statement": "Strings with Value Representation DS are too long\n**Describe the bug**\r\nStrings of Value Representation DS are restricted to a maximum length of 16 bytes according to [Part 5 Section 6.2](http://dicom.nema.org/medical/dicom/current/output/chtml/part05/sect_6.2.html#para_15754884-9ca2-4b12-9368-d66f32bc8ce1), but `pydicom.valuerep.DS` may represent numbers with more than 16 bytes.\r\n\r\n**Expected behavior**\r\n`pydicom.valuerep.DS` should create a string of maximum length 16, when passed a fixed point number with many decimals.\r\n\r\n**Steps To Reproduce**\r\n```python\r\nlen(str(pydicom.valuerep.DS(3.14159265358979323846264338327950288419716939937510582097)).encode('utf-8'))\r\nlen(str(pydicom.valuerep.DS(\"3.14159265358979323846264338327950288419716939937510582097\")).encode('utf-8'))\r\n```\r\nreturns `17` and `58`, respectively, instead of `16`.\r\n\r\n**Your environment**\r\n```\r\nmodule       | version\r\n------       | -------\r\nplatform     | macOS-10.15.6-x86_64-i386-64bit\r\nPython       | 3.8.6 (default, Oct  8 2020, 14:06:32)  [Clang 12.0.0 (clang-1200.0.32.2)]\r\npydicom      | 2.0.0\r\ngdcm         | _module not found_\r\njpeg_ls      | _module not found_\r\nnumpy        | 1.19.4\r\nPIL          | 8.0.1\r\n```\n", "hints_text": "Thanks for this, @hackermd.  Pydicom has traditionally been permissive about values as they are set, because sometimes people want to replicate existing invalid DICOM.  But for sure this should be an error when `config.enforce_valid_values` is `True` (perhaps a warning otherwise), and pydicom should leave it to the calling code to figure out how to truncate the value.\nI've just checked, because I thought I remembered this coming up before.  There is a check - but only for `DS` derived from `Decimal` (and when `enforce_valid_values is True`).  Wouldn't be hard to replicate the checks for the `DSfloat` class.", "created_at": "2021-04-04T22:06:46Z", "version": "2.1", "FAIL_TO_PASS": ["pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[3.14159265358979]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[-1234.456e78]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E-5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[1.234E+5]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[+1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_valid[42", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[nan]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[3.141592653589793]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1,000]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[127.0.0.1]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[1.e]", "pydicom/tests/test_valuerep.py::TestIsValidDS::test_invalid[]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.0-1.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.0-0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.0--0.0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[0.123-0.123]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-0.321--0.321]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1e-05-1e-05]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[3.141592653589793-3.14159265358979]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-3.141592653589793--3.1415926535898]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[5.385940192876374e-07-5.3859401929e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[-5.385940192876374e-07--5.385940193e-07]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[12342534378.125532-12342534378.1255]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[64070869985876.78-64070869985876.8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_auto_format[1.7976931348623157e+308-1.797693135e+308]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[100]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[101]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[-1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[2]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[3]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[4]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[5]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[6]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[7]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[8]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[9]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[10]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[11]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[12]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[13]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[14]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[15]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_powers_of_negative_pi[16]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan0]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[nan1]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[-inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_invalid[inf]", "pydicom/tests/test_valuerep.py::TestTruncateFloatForDS::test_wrong_type", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_auto_format_valid_string[False]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_length", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-nan]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf0]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[nan2]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[-inf1]", "pydicom/tests/test_valuerep.py::TestDSfloat::test_enforce_valid_values_value[inf1]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_invalid_string[False]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-NaN]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[-Infinity]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val4]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val5]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val6]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_enforce_valid_values_value[val7]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[True]", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_auto_format_valid_string[False]"], "PASS_TO_PASS": ["pydicom/tests/test_valuerep.py::TestTM::test_pickling", "pydicom/tests/test_valuerep.py::TestTM::test_str", "pydicom/tests/test_valuerep.py::TestTM::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestTM::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestTM::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_pickling", "pydicom/tests/test_valuerep.py::TestDT::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestDT::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDT::test_new_str_conversion", "pydicom/tests/test_valuerep.py::TestDA::test_pickling", "pydicom/tests/test_valuerep.py::TestDA::test_new_obj_conversion", "pydicom/tests/test_valuerep.py::TestDS::test_empty_value", "pydicom/tests/test_valuerep.py::TestDS::test_float_values", "pydicom/tests/test_valuerep.py::TestDSfloat::test_pickling", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str", "pydicom/tests/test_valuerep.py::TestDSfloat::test_repr", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSfloat", "pydicom/tests/test_valuerep.py::TestDSfloat::test_DSdecimal", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_pickling", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_float_value", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_new_empty_str", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSfloat", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_DSdecimal", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_repr", "pydicom/tests/test_valuerep.py::TestIS::test_empty_value", "pydicom/tests/test_valuerep.py::TestIS::test_valid_value", "pydicom/tests/test_valuerep.py::TestIS::test_invalid_value", "pydicom/tests/test_valuerep.py::TestIS::test_pickling", "pydicom/tests/test_valuerep.py::TestIS::test_longint", "pydicom/tests/test_valuerep.py::TestIS::test_overflow", "pydicom/tests/test_valuerep.py::TestIS::test_str", "pydicom/tests/test_valuerep.py::TestIS::test_repr", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_default", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_enforce_valid_value", "pydicom/tests/test_valuerep.py::TestDecimalString::test_DS_decimal_set", "pydicom/tests/test_valuerep.py::TestDecimalString::test_valid_decimal_strings", "pydicom/tests/test_valuerep.py::TestDecimalString::test_invalid_decimal_strings", "pydicom/tests/test_valuerep.py::TestPersonName::test_last_first", "pydicom/tests/test_valuerep.py::TestPersonName::test_copy", "pydicom/tests/test_valuerep.py::TestPersonName::test_three_component", "pydicom/tests/test_valuerep.py::TestPersonName::test_formatting", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_kr", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_comp_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_caret_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_not_equal", "pydicom/tests/test_valuerep.py::TestPersonName::test_encoding_carried", "pydicom/tests/test_valuerep.py::TestPersonName::test_hash", "pydicom/tests/test_valuerep.py::TestPersonName::test_next", "pydicom/tests/test_valuerep.py::TestPersonName::test_iterator", "pydicom/tests/test_valuerep.py::TestPersonName::test_contains", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_kr_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_veterinary", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator", "pydicom/tests/test_valuerep.py::TestPersonName::test_from_named_components_with_separator_from_bytes", "pydicom/tests/test_valuerep.py::TestDateTime::test_date", "pydicom/tests/test_valuerep.py::TestDateTime::test_date_time", "pydicom/tests/test_valuerep.py::TestDateTime::test_time", "pydicom/tests/test_valuerep.py::test_person_name_unicode_warns"], "environment_setup_commit": "506ecea8f378dc687d5c504788fc78810a190b7a"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1139", "base_commit": "b9fb05c177b685bf683f7f57b2d57374eb7d882d", "patch": "diff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -1,6 +1,5 @@\n # Copyright 2008-2018 pydicom authors. See LICENSE file for details.\n \"\"\"Special classes for DICOM value representations (VR)\"\"\"\n-from copy import deepcopy\n from decimal import Decimal\n import re\n \n@@ -750,6 +749,25 @@ def __ne__(self, other):\n     def __str__(self):\n         return '='.join(self.components).__str__()\n \n+    def __next__(self):\n+        # Get next character or stop iteration\n+        if self._i < self._rep_len:\n+            c = self._str_rep[self._i]\n+            self._i += 1\n+            return c\n+        else:\n+            raise StopIteration\n+\n+    def __iter__(self):\n+        # Get string rep. and length, initialize index counter\n+        self._str_rep = self.__str__()\n+        self._rep_len = len(self._str_rep)\n+        self._i = 0\n+        return self\n+\n+    def __contains__(self, x):\n+        return x in self.__str__()\n+\n     def __repr__(self):\n         return '='.join(self.components).__repr__()\n \n", "test_patch": "diff --git a/pydicom/tests/test_valuerep.py b/pydicom/tests/test_valuerep.py\n--- a/pydicom/tests/test_valuerep.py\n+++ b/pydicom/tests/test_valuerep.py\n@@ -427,6 +427,62 @@ def test_hash(self):\n         )\n         assert hash(pn1) == hash(pn2)\n \n+    def test_next(self):\n+        \"\"\"Test that the next function works on it's own\"\"\"\n+        # Test getting the first character\n+        pn1 = PersonName(\"John^Doe^^Dr\", encodings=default_encoding)\n+        pn1_itr = iter(pn1)\n+        assert next(pn1_itr) == \"J\"\n+\n+        # Test getting multiple characters\n+        pn2 = PersonName(\n+            \"Yamada^Tarou=\u5c71\u7530^\u592a\u90ce=\u3084\u307e\u3060^\u305f\u308d\u3046\", [default_encoding, \"iso2022_jp\"]\n+        )\n+        pn2_itr = iter(pn2)\n+        assert next(pn2_itr) == \"Y\"\n+        assert next(pn2_itr) == \"a\"\n+\n+        # Test getting all characters\n+        pn3 = PersonName(\"SomeName\")\n+        pn3_itr = iter(pn3)\n+        assert next(pn3_itr) == \"S\"\n+        assert next(pn3_itr) == \"o\"\n+        assert next(pn3_itr) == \"m\"\n+        assert next(pn3_itr) == \"e\"\n+        assert next(pn3_itr) == \"N\"\n+        assert next(pn3_itr) == \"a\"\n+        assert next(pn3_itr) == \"m\"\n+        assert next(pn3_itr) == \"e\"\n+\n+        # Attempting to get next characeter should stop the iteration\n+        # I.e. next can only start once\n+        with pytest.raises(StopIteration):\n+            next(pn3_itr)\n+\n+        # Test that next() doesn't work without instantiating an iterator\n+        pn4 = PersonName(\"SomeName\")\n+        with pytest.raises(AttributeError):\n+            next(pn4)\n+\n+    def test_iterator(self):\n+        \"\"\"Test that iterators can be corretly constructed\"\"\"\n+        name_str = \"John^Doe^^Dr\"\n+        pn1 = PersonName(name_str)\n+        \n+        for i, c in enumerate(pn1):\n+            assert name_str[i] == c\n+\n+        # Ensure that multiple iterators can be created on the same variable\n+        for i, c in enumerate(pn1):\n+            assert name_str[i] == c\n+\n+    def test_contains(self):\n+        \"\"\"Test that characters can be check if they are within the name\"\"\"\n+        pn1 = PersonName(\"John^Doe\")\n+        assert (\"J\" in pn1) == True\n+        assert (\"o\" in pn1) == True\n+        assert (\"x\" in pn1) == False\n+\n \n class TestDateTime:\n     \"\"\"Unit tests for DA, DT, TM conversion to datetime objects\"\"\"\n", "problem_statement": "Make PersonName3 iterable\n```python\r\nfrom pydicom import Dataset\r\n\r\nds = Dataset()\r\nds.PatientName = 'SomeName'\r\n\r\n'S' in ds.PatientName\r\n```\r\n```\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\nTypeError: argument of type 'PersonName3' is not iterable\r\n```\r\n\r\nI'm not really sure if this is intentional or if PN elements should support `str` methods. And yes I know I can `str(ds.PatientName)` but it's a bit silly, especially when I keep having to write exceptions to my element iterators just for PN elements.\n", "hints_text": "I think it is reasonable to support at least some `str` methods (definitely `__contains__` for the example above), but there are many that don't make a lot of sense in this context though - e.g. `join`, `ljust`, `maketrans`, `splitlines` just to name a few, but I suppose each would either never be actually used or would have no effect.\r\n\r\nI have a vague memory that one or more of the `PersonName` classes was at one time subclassed from `str`, or at least that it was discussed... does anyone remember?  Maybe it would be easier now with only Python 3 supported.\n`PersonName` was derived from `str` or `unicode` in Python 2, but that caused a number of problems, which is why you switched to `PersonName3` in Python 3, I think. I agree though that it makes sense to implement `str` methods, either by implementing some of them, or generically by adding `__getattr__` that converts it to `str` and applies the attribute to that string. ", "created_at": "2020-06-26T11:47:17Z", "version": "2.0", "FAIL_TO_PASS": ["pydicom/tests/test_valuerep.py::TestPersonName::test_next", "pydicom/tests/test_valuerep.py::TestPersonName::test_iterator", "pydicom/tests/test_valuerep.py::TestPersonName::test_contains"], "PASS_TO_PASS": ["pydicom/tests/test_valuerep.py::TestTM::test_pickling", "pydicom/tests/test_valuerep.py::TestDT::test_pickling", "pydicom/tests/test_valuerep.py::TestDA::test_pickling", "pydicom/tests/test_valuerep.py::TestDS::test_empty_value", "pydicom/tests/test_valuerep.py::TestDS::test_float_values", "pydicom/tests/test_valuerep.py::TestDSfloat::test_pickling", "pydicom/tests/test_valuerep.py::TestDSfloat::test_str", "pydicom/tests/test_valuerep.py::TestDSfloat::test_repr", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_pickling", "pydicom/tests/test_valuerep.py::TestDSdecimal::test_float_value", "pydicom/tests/test_valuerep.py::TestIS::test_empty_value", "pydicom/tests/test_valuerep.py::TestIS::test_valid_value", "pydicom/tests/test_valuerep.py::TestIS::test_invalid_value", "pydicom/tests/test_valuerep.py::TestIS::test_pickling", "pydicom/tests/test_valuerep.py::TestIS::test_longint", "pydicom/tests/test_valuerep.py::TestIS::test_overflow", "pydicom/tests/test_valuerep.py::TestIS::test_str", "pydicom/tests/test_valuerep.py::TestIS::test_repr", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_default", "pydicom/tests/test_valuerep.py::TestBadValueRead::test_read_bad_value_in_VR_enforce_valid_value", "pydicom/tests/test_valuerep.py::TestDecimalString::test_DS_decimal_set", "pydicom/tests/test_valuerep.py::TestDecimalString::test_valid_decimal_strings", "pydicom/tests/test_valuerep.py::TestDecimalString::test_invalid_decimal_strings", "pydicom/tests/test_valuerep.py::TestPersonName::test_last_first", "pydicom/tests/test_valuerep.py::TestPersonName::test_copy", "pydicom/tests/test_valuerep.py::TestPersonName::test_three_component", "pydicom/tests/test_valuerep.py::TestPersonName::test_formatting", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_kr", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_comp_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_bytes_caret_delimiter", "pydicom/tests/test_valuerep.py::TestPersonName::test_unicode_jp_from_unicode", "pydicom/tests/test_valuerep.py::TestPersonName::test_not_equal", "pydicom/tests/test_valuerep.py::TestPersonName::test_encoding_carried", "pydicom/tests/test_valuerep.py::TestPersonName::test_hash", "pydicom/tests/test_valuerep.py::TestDateTime::test_date", "pydicom/tests/test_valuerep.py::TestDateTime::test_date_time", "pydicom/tests/test_valuerep.py::TestDateTime::test_time"], "environment_setup_commit": "9d69811e539774f296c2f289839147e741251716"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-763", "base_commit": "0a14b273517a082603cc157faa88a5ab0ac4cac9", "patch": "diff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\n--- a/pvlib/pvsystem.py\n+++ b/pvlib/pvsystem.py\n@@ -2473,7 +2473,8 @@ def singlediode(photocurrent, saturation_current, resistance_series,\n \n \n def max_power_point(photocurrent, saturation_current, resistance_series,\n-                    resistance_shunt, nNsVth, method='brentq'):\n+                    resistance_shunt, nNsVth, d2mutau=0, NsVbi=np.Inf,\n+                    method='brentq'):\n     \"\"\"\n     Given the single diode equation coefficients, calculates the maximum power\n     point (MPP).\n@@ -2491,6 +2492,17 @@ def max_power_point(photocurrent, saturation_current, resistance_series,\n     nNsVth : numeric\n         product of thermal voltage ``Vth`` [V], diode ideality factor ``n``,\n         and number of serices cells ``Ns``\n+    d2mutau : numeric, default 0\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that accounts for recombination current in the\n+        intrinsic layer. The value is the ratio of intrinsic layer thickness\n+        squared :math:`d^2` to the diffusion length of charge carriers\n+        :math:`\\\\mu \\\\tau`. [V]\n+    NsVbi : numeric, default np.inf\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that is the product of the PV module number of series\n+        cells ``Ns`` and the builtin voltage ``Vbi`` of the intrinsic layer.\n+        [V].\n     method : str\n         either ``'newton'`` or ``'brentq'``\n \n@@ -2508,7 +2520,8 @@ def max_power_point(photocurrent, saturation_current, resistance_series,\n     \"\"\"\n     i_mp, v_mp, p_mp = _singlediode.bishop88_mpp(\n         photocurrent, saturation_current, resistance_series,\n-        resistance_shunt, nNsVth, method=method.lower()\n+        resistance_shunt, nNsVth, d2mutau=0, NsVbi=np.Inf,\n+        method=method.lower()\n     )\n     if isinstance(photocurrent, pd.Series):\n         ivp = {'i_mp': i_mp, 'v_mp': v_mp, 'p_mp': p_mp}\ndiff --git a/pvlib/singlediode.py b/pvlib/singlediode.py\n--- a/pvlib/singlediode.py\n+++ b/pvlib/singlediode.py\n@@ -94,14 +94,17 @@ def bishop88(diode_voltage, photocurrent, saturation_current,\n     nNsVth : numeric\n         product of thermal voltage ``Vth`` [V], diode ideality factor ``n``,\n         and number of series cells ``Ns``\n-    d2mutau : numeric\n-        PVSyst thin-film recombination parameter that is the ratio of thickness\n-        of the intrinsic layer squared :math:`d^2` and the diffusion length of\n-        charge carriers :math:`\\\\mu \\\\tau`, in volts [V], defaults to 0[V]\n-    NsVbi : numeric\n-        PVSyst thin-film recombination parameter that is the product of the PV\n-        module number of series cells ``Ns`` and the builtin voltage ``Vbi`` of\n-        the intrinsic layer, in volts [V], defaults to ``np.inf``\n+    d2mutau : numeric, default 0\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that accounts for recombination current in the\n+        intrinsic layer. The value is the ratio of intrinsic layer thickness\n+        squared :math:`d^2` to the diffusion length of charge carriers\n+        :math:`\\\\mu \\\\tau`. [V]\n+    NsVbi : numeric, default np.inf\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that is the product of the PV module number of series\n+        cells ``Ns`` and the builtin voltage ``Vbi`` of the intrinsic layer.\n+        [V].\n     gradients : bool\n         False returns only I, V, and P. True also returns gradients\n \n@@ -116,8 +119,8 @@ def bishop88(diode_voltage, photocurrent, saturation_current,\n     Notes\n     -----\n     The PVSyst thin-film recombination losses parameters ``d2mutau`` and\n-    ``NsVbi`` are only applied to cadmium-telluride (CdTe) and amorphous-\n-    silicon (a:Si) PV modules, [2]_, [3]_. The builtin voltage :math:`V_{bi}`\n+    ``NsVbi`` should only be applied to cadmium-telluride (CdTe) and amorphous-\n+    silicon (a-Si) PV modules, [2]_, [3]_. The builtin voltage :math:`V_{bi}`\n     should account for all junctions. For example: tandem and triple junction\n     cells would have builtin voltages of 1.8[V] and 2.7[V] respectively, based\n     on the default of 0.9[V] for a single junction. The parameter ``NsVbi``\n@@ -173,7 +176,7 @@ def bishop88(diode_voltage, photocurrent, saturation_current,\n \n def bishop88_i_from_v(voltage, photocurrent, saturation_current,\n                       resistance_series, resistance_shunt, nNsVth,\n-                      method='newton'):\n+                      d2mutau=0, NsVbi=np.Inf, method='newton'):\n     \"\"\"\n     Find current given any voltage.\n \n@@ -192,6 +195,17 @@ def bishop88_i_from_v(voltage, photocurrent, saturation_current,\n     nNsVth : numeric\n         product of diode ideality factor (n), number of series cells (Ns), and\n         thermal voltage (Vth = k_b * T / q_e) in volts [V]\n+    d2mutau : numeric, default 0\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that accounts for recombination current in the\n+        intrinsic layer. The value is the ratio of intrinsic layer thickness\n+        squared :math:`d^2` to the diffusion length of charge carriers\n+        :math:`\\\\mu \\\\tau`. [V]\n+    NsVbi : numeric, default np.inf\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that is the product of the PV module number of series\n+        cells ``Ns`` and the builtin voltage ``Vbi`` of the intrinsic layer.\n+        [V].\n     method : str\n         one of two optional search methods: either ``'brentq'``, a reliable and\n         bounded method or ``'newton'`` which is the default.\n@@ -203,7 +217,7 @@ def bishop88_i_from_v(voltage, photocurrent, saturation_current,\n     \"\"\"\n     # collect args\n     args = (photocurrent, saturation_current, resistance_series,\n-            resistance_shunt, nNsVth)\n+            resistance_shunt, nNsVth, d2mutau, NsVbi)\n \n     def fv(x, v, *a):\n         # calculate voltage residual given diode voltage \"x\"\n@@ -216,8 +230,9 @@ def fv(x, v, *a):\n         # brentq only works with scalar inputs, so we need a set up function\n         # and np.vectorize to repeatedly call the optimizer with the right\n         # arguments for possible array input\n-        def vd_from_brent(voc, v, iph, isat, rs, rsh, gamma):\n-            return brentq(fv, 0.0, voc, args=(v, iph, isat, rs, rsh, gamma))\n+        def vd_from_brent(voc, v, iph, isat, rs, rsh, gamma, d2mutau, NsVbi):\n+            return brentq(fv, 0.0, voc,\n+                          args=(v, iph, isat, rs, rsh, gamma, d2mutau, NsVbi))\n \n         vd_from_brent_vectorized = np.vectorize(vd_from_brent)\n         vd = vd_from_brent_vectorized(voc_est, voltage, *args)\n@@ -235,7 +250,7 @@ def vd_from_brent(voc, v, iph, isat, rs, rsh, gamma):\n \n def bishop88_v_from_i(current, photocurrent, saturation_current,\n                       resistance_series, resistance_shunt, nNsVth,\n-                      method='newton'):\n+                      d2mutau=0, NsVbi=np.Inf, method='newton'):\n     \"\"\"\n     Find voltage given any current.\n \n@@ -254,6 +269,17 @@ def bishop88_v_from_i(current, photocurrent, saturation_current,\n     nNsVth : numeric\n         product of diode ideality factor (n), number of series cells (Ns), and\n         thermal voltage (Vth = k_b * T / q_e) in volts [V]\n+    d2mutau : numeric, default 0\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that accounts for recombination current in the\n+        intrinsic layer. The value is the ratio of intrinsic layer thickness\n+        squared :math:`d^2` to the diffusion length of charge carriers\n+        :math:`\\\\mu \\\\tau`. [V]\n+    NsVbi : numeric, default np.inf\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that is the product of the PV module number of series\n+        cells ``Ns`` and the builtin voltage ``Vbi`` of the intrinsic layer.\n+        [V].\n     method : str\n         one of two optional search methods: either ``'brentq'``, a reliable and\n         bounded method or ``'newton'`` which is the default.\n@@ -265,7 +291,7 @@ def bishop88_v_from_i(current, photocurrent, saturation_current,\n     \"\"\"\n     # collect args\n     args = (photocurrent, saturation_current, resistance_series,\n-            resistance_shunt, nNsVth)\n+            resistance_shunt, nNsVth, d2mutau, NsVbi)\n     # first bound the search using voc\n     voc_est = estimate_voc(photocurrent, saturation_current, nNsVth)\n \n@@ -277,8 +303,9 @@ def fi(x, i, *a):\n         # brentq only works with scalar inputs, so we need a set up function\n         # and np.vectorize to repeatedly call the optimizer with the right\n         # arguments for possible array input\n-        def vd_from_brent(voc, i, iph, isat, rs, rsh, gamma):\n-            return brentq(fi, 0.0, voc, args=(i, iph, isat, rs, rsh, gamma))\n+        def vd_from_brent(voc, i, iph, isat, rs, rsh, gamma, d2mutau, NsVbi):\n+            return brentq(fi, 0.0, voc,\n+                          args=(i, iph, isat, rs, rsh, gamma, d2mutau, NsVbi))\n \n         vd_from_brent_vectorized = np.vectorize(vd_from_brent)\n         vd = vd_from_brent_vectorized(voc_est, current, *args)\n@@ -295,7 +322,8 @@ def vd_from_brent(voc, i, iph, isat, rs, rsh, gamma):\n \n \n def bishop88_mpp(photocurrent, saturation_current, resistance_series,\n-                 resistance_shunt, nNsVth, method='newton'):\n+                 resistance_shunt, nNsVth, d2mutau=0, NsVbi=np.Inf,\n+                 method='newton'):\n     \"\"\"\n     Find max power point.\n \n@@ -312,6 +340,17 @@ def bishop88_mpp(photocurrent, saturation_current, resistance_series,\n     nNsVth : numeric\n         product of diode ideality factor (n), number of series cells (Ns), and\n         thermal voltage (Vth = k_b * T / q_e) in volts [V]\n+    d2mutau : numeric, default 0\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that accounts for recombination current in the\n+        intrinsic layer. The value is the ratio of intrinsic layer thickness\n+        squared :math:`d^2` to the diffusion length of charge carriers\n+        :math:`\\\\mu \\\\tau`. [V]\n+    NsVbi : numeric, default np.inf\n+        PVsyst parameter for cadmium-telluride (CdTe) and amorphous-silicon\n+        (a-Si) modules that is the product of the PV module number of series\n+        cells ``Ns`` and the builtin voltage ``Vbi`` of the intrinsic layer.\n+        [V].\n     method : str\n         one of two optional search methods: either ``'brentq'``, a reliable and\n         bounded method or ``'newton'`` which is the default.\n@@ -324,7 +363,7 @@ def bishop88_mpp(photocurrent, saturation_current, resistance_series,\n     \"\"\"\n     # collect args\n     args = (photocurrent, saturation_current, resistance_series,\n-            resistance_shunt, nNsVth)\n+            resistance_shunt, nNsVth, d2mutau, NsVbi)\n     # first bound the search using voc\n     voc_est = estimate_voc(photocurrent, saturation_current, nNsVth)\n \n@@ -334,8 +373,9 @@ def fmpp(x, *a):\n     if method.lower() == 'brentq':\n         # break out arguments for numpy.vectorize to handle broadcasting\n         vec_fun = np.vectorize(\n-            lambda voc, iph, isat, rs, rsh, gamma:\n-                brentq(fmpp, 0.0, voc, args=(iph, isat, rs, rsh, gamma))\n+            lambda voc, iph, isat, rs, rsh, gamma, d2mutau, NsVbi:\n+                brentq(fmpp, 0.0, voc,\n+                       args=(iph, isat, rs, rsh, gamma, d2mutau, NsVbi))\n         )\n         vd = vec_fun(voc_est, *args)\n     elif method.lower() == 'newton':\n", "test_patch": "diff --git a/pvlib/test/test_singlediode.py b/pvlib/test/test_singlediode.py\n--- a/pvlib/test/test_singlediode.py\n+++ b/pvlib/test/test_singlediode.py\n@@ -4,7 +4,8 @@\n \n import numpy as np\n from pvlib import pvsystem\n-from pvlib.singlediode import bishop88, estimate_voc, VOLTAGE_BUILTIN\n+from pvlib.singlediode import (bishop88_mpp, estimate_voc, VOLTAGE_BUILTIN,\n+                               bishop88, bishop88_i_from_v, bishop88_v_from_i)\n import pytest\n from conftest import requires_scipy\n \n@@ -153,9 +154,13 @@ def get_pvsyst_fs_495():\n         'temp_ref': 25, 'irrad_ref': 1000, 'I_L_ref': 1.5743233463848496\n     }\n \n+# DeSoto @(888[W/m**2], 55[degC]) = {Pmp: 72.71, Isc: 1.402, Voc: 75.42)\n \n+\n+@requires_scipy\n @pytest.mark.parametrize(\n     'poa, temp_cell, expected, tol', [\n+        # reference conditions\n         (\n             get_pvsyst_fs_495()['irrad_ref'],\n             get_pvsyst_fs_495()['temp_ref'],\n@@ -167,9 +172,21 @@ def get_pvsyst_fs_495():\n             },\n             (5e-4, 0.04)\n         ),\n-        (POA, TCELL, {'pmp': 76.26, 'isc': 1.387, 'voc': 79.29}, (1e-3, 1e-3))]\n-)  # DeSoto @(888[W/m**2], 55[degC]) = {Pmp: 72.71, Isc: 1.402, Voc: 75.42)\n-def test_pvsyst_recombination_loss(poa, temp_cell, expected, tol):\n+        # other conditions\n+        (\n+            POA,\n+            TCELL,\n+            {\n+                'pmp': 76.262,\n+                'isc': 1.3868,\n+                'voc': 79.292\n+            },\n+            (1e-4, 1e-4)\n+        )\n+    ]\n+)\n+@pytest.mark.parametrize('method', ['newton', 'brentq'])\n+def test_pvsyst_recombination_loss(method, poa, temp_cell, expected, tol):\n     \"\"\"test PVSst recombination loss\"\"\"\n     pvsyst_fs_495 = get_pvsyst_fs_495()\n     # first evaluate PVSyst model with thin-film recombination loss current\n@@ -199,9 +216,30 @@ def test_pvsyst_recombination_loss(poa, temp_cell, expected, tol):\n     )\n     # test max power\n     assert np.isclose(max(pvsyst[2]), expected['pmp'], *tol)\n+\n     # test short circuit current\n     isc_pvsyst = np.interp(0, pvsyst[1], pvsyst[0])\n     assert np.isclose(isc_pvsyst, expected['isc'], *tol)\n-    # test open circuit current\n+\n+    # test open circuit voltage\n     voc_pvsyst = np.interp(0, pvsyst[0][::-1], pvsyst[1][::-1])\n     assert np.isclose(voc_pvsyst, expected['voc'], *tol)\n+\n+    # repeat tests as above with specialized bishop88 functions\n+    y = dict(d2mutau=pvsyst_fs_495['d2mutau'],\n+             NsVbi=VOLTAGE_BUILTIN*pvsyst_fs_495['cells_in_series'])\n+\n+    mpp_88 = bishop88_mpp(*x, **y, method=method)\n+    assert np.isclose(mpp_88[2], expected['pmp'], *tol)\n+\n+    isc_88 = bishop88_i_from_v(0, *x, **y, method=method)\n+    assert np.isclose(isc_88, expected['isc'], *tol)\n+\n+    voc_88 = bishop88_v_from_i(0, *x, **y, method=method)\n+    assert np.isclose(voc_88, expected['voc'], *tol)\n+\n+    ioc_88 = bishop88_i_from_v(voc_88, *x, **y, method=method)\n+    assert np.isclose(ioc_88, 0.0, *tol)\n+\n+    vsc_88 = bishop88_v_from_i(isc_88, *x, **y, method=method)\n+    assert np.isclose(vsc_88, 0.0, *tol)\n", "problem_statement": "Add recombination current params to all bishop88 functions\nThe changes made in #163 incorporate recombination current into the `bishop88()` function.  Functions that build on the `bishop88()` function should likewise accept these parameters.\r\n\n", "hints_text": "", "created_at": "2019-08-13T20:05:11Z", "version": "0.6", "FAIL_TO_PASS": ["pvlib/test/test_singlediode.py::test_pvsyst_recombination_loss[newton-1000-25-expected0-tol0]", "pvlib/test/test_singlediode.py::test_pvsyst_recombination_loss[newton-888-55-expected1-tol1]", "pvlib/test/test_singlediode.py::test_pvsyst_recombination_loss[brentq-1000-25-expected0-tol0]", "pvlib/test/test_singlediode.py::test_pvsyst_recombination_loss[brentq-888-55-expected1-tol1]"], "PASS_TO_PASS": ["pvlib/test/test_singlediode.py::test_newton_spr_e20_327", "pvlib/test/test_singlediode.py::test_newton_fs_495", "pvlib/test/test_singlediode.py::test_brentq_spr_e20_327", "pvlib/test/test_singlediode.py::test_brentq_fs_495"], "environment_setup_commit": "b91d178868d193afd56f8e3b013661a473d699c3"}, {"repo": "pyvista/pyvista", "instance_id": "pyvista__pyvista-432", "base_commit": "e1bac62d2f3df7bcba25d58ed5e83d89f9a6be98", "patch": "diff --git a/examples/01-filter/decimate.py b/examples/01-filter/decimate.py\n--- a/examples/01-filter/decimate.py\n+++ b/examples/01-filter/decimate.py\n@@ -13,10 +13,10 @@\n \n # Define a camera potion the shows this mesh properly\n cpos = [(0.4, -0.07, -0.31), (0.05, -0.13, -0.06), (-0.1, 1, 0.08)]\n-dargs = dict(cpos=cpos, show_edges=True, color=True)\n+dargs = dict(show_edges=True, color=True)\n \n # Preview the mesh\n-mesh.plot(**dargs)\n+mesh.plot(cpos=cpos, **dargs)\n \n ###############################################################################\n #  Now let's define a target reduction and compare the\n@@ -28,13 +28,13 @@\n ###############################################################################\n decimated = mesh.decimate(target_reduction)\n \n-decimated.plot(**dargs)\n+decimated.plot(cpos=cpos, **dargs)\n \n \n ###############################################################################\n pro_decimated = mesh.decimate_pro(target_reduction, preserve_topology=True)\n \n-pro_decimated.plot(**dargs)\n+pro_decimated.plot(cpos=cpos, **dargs)\n \n \n ###############################################################################\ndiff --git a/examples/01-filter/streamlines.py b/examples/01-filter/streamlines.py\n--- a/examples/01-filter/streamlines.py\n+++ b/examples/01-filter/streamlines.py\n@@ -62,7 +62,7 @@\n boundary = mesh.decimate_boundary().wireframe()\n \n p = pv.Plotter()\n-p.add_mesh(streamlines.tube(radius=0.2), ligthing=False)\n+p.add_mesh(streamlines.tube(radius=0.2), lighting=False)\n p.add_mesh(src)\n p.add_mesh(boundary, color=\"grey\", opacity=0.25)\n p.camera_position = [(10, 9.5, -43), (87.0, 73.5, 123.0), (-0.5, -0.7, 0.5)]\ndiff --git a/pyvista/core/filters.py b/pyvista/core/filters.py\n--- a/pyvista/core/filters.py\n+++ b/pyvista/core/filters.py\n@@ -32,7 +32,8 @@\n \n import pyvista\n from pyvista.utilities import (CELL_DATA_FIELD, POINT_DATA_FIELD, NORMALS,\n-                               generate_plane, get_array, wrap)\n+                               assert_empty_kwargs, generate_plane, get_array,\n+                               wrap)\n \n \n def _get_output(algorithm, iport=0, iconnection=0, oport=0, active_scalar=None,\n@@ -1031,14 +1032,13 @@ def warp_by_scalar(dataset, scalars=None, factor=1.0, normal=None,\n         inplace : bool\n             If True, the points of the give dataset will be updated.\n         \"\"\"\n+        factor = kwargs.pop('scale_factor', factor)\n+        assert_empty_kwargs(**kwargs)\n         if scalars is None:\n             field, scalars = dataset.active_scalar_info\n         arr, field = get_array(dataset, scalars, preference='point', info=True)\n         if field != pyvista.POINT_DATA_FIELD:\n             raise AssertionError('Dataset can only by warped by a point data array.')\n-        scale_factor = kwargs.get('scale_factor', None)\n-        if scale_factor is not None:\n-            factor = scale_factor\n         # Run the algorithm\n         alg = vtk.vtkWarpScalar()\n         alg.SetInputDataObject(dataset)\n@@ -2374,7 +2374,7 @@ def plot_curvature(poly_data, curv_type='mean', **kwargs):\n             - Minimum\n \n         **kwargs : optional\n-            See help(pyvista.plot)\n+            See :func:`pyvista.plot`\n \n         Returns\n         -------\n@@ -2973,6 +2973,7 @@ def clean(poly_data, point_merging=True, tolerance=None, lines_to_points=True,\n         \"\"\"\n         if tolerance is None:\n             tolerance = kwargs.pop('merge_tol', None)\n+        assert_empty_kwargs(**kwargs)\n         clean = vtk.vtkCleanPolyData()\n         clean.SetPointMerging(point_merging)\n         clean.SetConvertLinesToPoints(lines_to_points)\n@@ -3133,14 +3134,26 @@ def ray_trace(poly_data, origin, end_point, first_point=False, plot=False,\n         return intersection_points, intersection_cells\n \n \n-    def plot_boundaries(poly_data, **kwargs):\n-        \"\"\" Plots boundaries of a mesh \"\"\"\n+    def plot_boundaries(poly_data, edge_color=\"red\", **kwargs):\n+        \"\"\" Plots boundaries of a mesh\n+\n+        Parameters\n+        ----------\n+        edge_color : str, etc.\n+            The color of the edges when they are added to the plotter.\n+\n+        kwargs : optional\n+            All additional keyword arguments will be passed to\n+            :func:`pyvista.BasePlotter.add_mesh`\n+\n+        \"\"\"\n         edges = DataSetFilters.extract_edges(poly_data)\n \n         plotter = pyvista.Plotter(off_screen=kwargs.pop('off_screen', False),\n                                   notebook=kwargs.pop('notebook', None))\n-        plotter.add_mesh(edges, 'r', style='wireframe', legend='Edges')\n-        plotter.add_mesh(poly_data, legend='Mesh', **kwargs)\n+        plotter.add_mesh(edges, color=edge_color, style='wireframe', label='Edges')\n+        plotter.add_mesh(poly_data, label='Mesh', **kwargs)\n+        plotter.add_legend()\n         return plotter.show()\n \n \ndiff --git a/pyvista/core/objects.py b/pyvista/core/objects.py\n--- a/pyvista/core/objects.py\n+++ b/pyvista/core/objects.py\n@@ -5,8 +5,9 @@\n import vtk\n \n import pyvista\n-from pyvista.utilities import (ROW_DATA_FIELD, convert_array, get_array,\n-                               parse_field_choice, row_array, vtk_bit_array_to_char)\n+from pyvista.utilities import (ROW_DATA_FIELD, assert_empty_kwargs,\n+                               convert_array, get_array, parse_field_choice,\n+                               row_array, vtk_bit_array_to_char)\n \n from .common import DataObject, _ScalarsDict\n \n@@ -409,6 +410,8 @@ def adder(self, scalars, name, set_active=False, deep=True):\n class Texture(vtk.vtkTexture):\n     \"\"\"A helper class for vtkTextures\"\"\"\n     def __init__(self, *args, **kwargs):\n+        assert_empty_kwargs(**kwargs)\n+\n         if len(args) == 1:\n             if isinstance(args[0], vtk.vtkTexture):\n                 self._from_texture(args[0])\ndiff --git a/pyvista/plotting/helpers.py b/pyvista/plotting/helpers.py\n--- a/pyvista/plotting/helpers.py\n+++ b/pyvista/plotting/helpers.py\n@@ -74,6 +74,11 @@ def plot(var_item, off_screen=None, full_screen=False, screenshot=None,\n     if notebook is None:\n         notebook = scooby.in_ipykernel()\n \n+    eye_dome_lighting = kwargs.pop(\"edl\", eye_dome_lighting)\n+    show_grid = kwargs.pop('show_grid', False)\n+    height = kwargs.get('height', 400)\n+    auto_close = kwargs.get('auto_close', rcParams['auto_close'])\n+\n     if notebook:\n         off_screen = notebook\n     plotter = Plotter(off_screen=off_screen, notebook=notebook)\n@@ -109,11 +114,10 @@ def plot(var_item, off_screen=None, full_screen=False, screenshot=None,\n     if text:\n         plotter.add_text(text)\n \n-    if show_bounds or kwargs.get('show_grid', False):\n-        if kwargs.get('show_grid', False):\n-            plotter.show_grid()\n-        else:\n-            plotter.show_bounds()\n+    if show_grid:\n+        plotter.show_grid()\n+    elif show_bounds:\n+        plotter.show_bounds()\n \n     if cpos is None:\n         cpos = plotter.get_default_cam_pos()\n@@ -122,7 +126,6 @@ def plot(var_item, off_screen=None, full_screen=False, screenshot=None,\n     else:\n         plotter.camera_position = cpos\n \n-    eye_dome_lighting = kwargs.pop(\"edl\", eye_dome_lighting)\n     if eye_dome_lighting:\n         plotter.enable_eye_dome_lighting()\n \n@@ -136,10 +139,10 @@ def plot(var_item, off_screen=None, full_screen=False, screenshot=None,\n                           screenshot=screenshot,\n                           return_img=return_img,\n                           use_panel=use_panel,\n-                          height=kwargs.get('height', 400))\n+                          height=height)\n \n     # close and return camera position and maybe image\n-    if kwargs.get('auto_close', rcParams['auto_close']):\n+    if auto_close:\n         plotter.close()\n \n     # Result will be handled by plotter.show(): cpos or [cpos, img]\ndiff --git a/pyvista/plotting/plotting.py b/pyvista/plotting/plotting.py\n--- a/pyvista/plotting/plotting.py\n+++ b/pyvista/plotting/plotting.py\n@@ -15,8 +15,9 @@\n import warnings\n \n import pyvista\n-from pyvista.utilities import (convert_array, convert_string_array,\n-                               get_array, is_pyvista_dataset, numpy_to_texture,\n+from pyvista.utilities import (assert_empty_kwargs, convert_array,\n+                               convert_string_array, get_array,\n+                               is_pyvista_dataset, numpy_to_texture,\n                                raise_not_matching, try_callback, wrap)\n \n from .colors import get_cmap_safe\n@@ -580,7 +581,7 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n                  specular_power=100.0, nan_color=None, nan_opacity=1.0,\n                  loc=None, culling=None, rgb=False, categories=False,\n                  use_transparency=False, below_color=None, above_color=None,\n-                 annotations=None, pickable=True, **kwargs):\n+                 annotations=None, pickable=True, preference=\"point\", **kwargs):\n         \"\"\"\n         Adds any PyVista/VTK mesh or dataset that PyVista can wrap to the\n         scene. This method using a mesh representation to view the surfaces\n@@ -804,8 +805,10 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n         if lighting is None:\n             lighting = rcParams['lighting']\n \n-        if clim is None:\n-            clim = kwargs.get('rng', None)\n+        # supported aliases\n+        clim = kwargs.pop('rng', clim)\n+        cmap = kwargs.pop('colormap', cmap)\n+        culling = kwargs.pop(\"backface_culling\", culling)\n \n         if render_points_as_spheres is None:\n             render_points_as_spheres = rcParams['render_points_as_spheres']\n@@ -815,18 +818,22 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n \n         if nan_color is None:\n             nan_color = rcParams['nan_color']\n-        nanr, nanb, nang = parse_color(nan_color)\n-        nan_color = nanr, nanb, nang, nan_opacity\n+        nan_color = list(parse_color(nan_color))\n+        nan_color.append(nan_opacity)\n         if color is True:\n             color = rcParams['color']\n \n         if texture is False:\n             texture = None\n \n-        if culling is None:\n-            culling = kwargs.get(\"backface_culling\", False)\n-            if culling is True:\n-                culling = 'backface'\n+        if culling is True:\n+            culling = 'backface'\n+\n+        rgb = kwargs.pop('rgba', rgb)\n+\n+        if \"scalar\" in kwargs:\n+            raise TypeError(\"`scalar` is an invalid keyword argument for `add_mesh`. Perhaps you mean `scalars` with an s?\")\n+        assert_empty_kwargs(**kwargs)\n \n         ##### Handle composite datasets #####\n \n@@ -846,9 +853,9 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n                     raise RuntimeError('Scalar array must be given as a string name for multiblock datasets.')\n \n             the_arguments = locals()\n-            the_arguments.update(kwargs)\n             the_arguments.pop('self')\n             the_arguments.pop('mesh')\n+            the_arguments.pop('kwargs')\n \n             if multi_colors:\n                 # Compute unique colors for each index of the block\n@@ -965,7 +972,7 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n             self.mapper.SetArrayName(scalars)\n             original_scalar_name = scalars\n             scalars = get_array(mesh, scalars,\n-                                preference=kwargs.get('preference', 'cell'), err=True)\n+                                preference=preference, err=True)\n             if stitle is None:\n                 stitle = original_scalar_name\n \n@@ -995,7 +1002,7 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n             try:\n                 # Get array from mesh\n                 opacity = get_array(mesh, opacity,\n-                                    preference=kwargs.get('preference', 'cell'), err=True)\n+                                    preference=preference, err=True)\n                 opacity = normalize(opacity)\n                 _custom_opac = True\n             except:\n@@ -1018,8 +1025,6 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n             opacity = 255 - opacity\n \n         # Scalar formatting ===================================================\n-        if cmap is None: # grab alias for cmaps: colormap\n-            cmap = kwargs.get('colormap', None)\n         if cmap is None: # Set default map if matplotlib is avaialble\n             if has_matplotlib:\n                 cmap = rcParams['cmap']\n@@ -1049,8 +1054,6 @@ def add_mesh(self, mesh, color=None, style=None, scalars=None,\n                 scalar_bar_args.setdefault('n_labels', 0)\n                 _using_labels = True\n \n-            if rgb is False or rgb is None:\n-                rgb = kwargs.get('rgba', False)\n             if rgb:\n                 if scalars.ndim != 2 or scalars.shape[1] < 3 or scalars.shape[1] > 4:\n                     raise ValueError('RGB array must be n_points/n_cells by 3/4 in shape.')\n@@ -1229,7 +1232,7 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n                    loc=None, culling=False, multi_colors=False,\n                    blending='composite', mapper='fixed_point',\n                    stitle=None, scalar_bar_args=None, show_scalar_bar=None,\n-                   annotations=None, pickable=True, **kwargs):\n+                   annotations=None, pickable=True, preference=\"point\", **kwargs):\n         \"\"\"\n         Adds a volume, rendered using a fixed point ray cast mapper by default.\n \n@@ -1355,8 +1358,14 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n         if name is None:\n             name = '{}({})'.format(type(volume).__name__, str(hex(id(volume))))\n \n-        if clim is None:\n-            clim = kwargs.get('rng', None)\n+        # Supported aliases\n+        clim = kwargs.pop('rng', clim)\n+        cmap = kwargs.pop('colormap', cmap)\n+        culling = kwargs.pop(\"backface_culling\", culling)\n+\n+        if \"scalar\" in kwargs:\n+            raise TypeError(\"`scalar` is an invalid keyword argument for `add_mesh`. Perhaps you mean `scalars` with an s?\")\n+        assert_empty_kwargs(**kwargs)\n \n         if scalar_bar_args is None:\n             scalar_bar_args = {}\n@@ -1364,10 +1373,8 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n         if show_scalar_bar is None:\n             show_scalar_bar = rcParams['show_scalar_bar']\n \n-        if culling is None:\n-            culling = kwargs.get(\"backface_culling\", False)\n-            if culling is True:\n-                culling = 'backface'\n+        if culling is True:\n+            culling = 'backface'\n \n         # Convert the VTK data object to a pyvista wrapped object if neccessary\n         if not is_pyvista_dataset(volume):\n@@ -1416,7 +1423,7 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n                                     reset_camera=reset_camera, name=next_name,\n                                     ambient=ambient, categories=categories, loc=loc,\n                                     culling=culling, clim=clim,\n-                                    mapper=mapper, pickable=pickable, **kwargs)\n+                                    mapper=mapper, pickable=pickable)\n \n                 actors.append(a)\n             return actors\n@@ -1444,7 +1451,7 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n         if isinstance(scalars, str):\n             title = scalars\n             scalars = get_array(volume, scalars,\n-                                preference=kwargs.get('preference', 'point'), err=True)\n+                                preference=preference, err=True)\n             if stitle is None:\n                 stitle = title\n         else:\n@@ -1512,11 +1519,9 @@ def add_volume(self, volume, scalars=None, clim=None, resolution=None,\n             for val, anno in annotations.items():\n                 table.SetAnnotation(float(val), str(anno))\n \n-        if cmap is None: # grab alias for cmaps: colormap\n-            cmap = kwargs.get('colormap', None)\n-            if cmap is None: # Set default map if matplotlib is avaialble\n-                if has_matplotlib:\n-                    cmap = rcParams['cmap']\n+        if cmap is None: # Set default map if matplotlib is avaialble\n+            if has_matplotlib:\n+                cmap = rcParams['cmap']\n \n         if cmap is not None:\n             if not has_matplotlib:\n@@ -2971,7 +2976,7 @@ def add_point_labels(self, points, labels, italic=False, bold=True,\n                          name=None, shape_color='grey', shape='rounded_rect',\n                          fill_shape=True, margin=3, shape_opacity=1.0,\n                          pickable=False, render_points_as_spheres=False,\n-                         tolerance=0.001, **kwargs):\n+                         tolerance=0.001):\n         \"\"\"\n         Creates a point actor with one label from list labels assigned to\n         each point.\n@@ -3060,9 +3065,6 @@ def add_point_labels(self, points, labels, italic=False, bold=True,\n             font_family = rcParams['font']['family']\n         if font_size is None:\n             font_size = rcParams['font']['size']\n-        if point_color is None and text_color is None and kwargs.get('color', None) is not None:\n-            point_color = kwargs.get('color', None)\n-            text_color = kwargs.get('color', None)\n         if point_color is None:\n             point_color = rcParams['color']\n         if text_color is None:\ndiff --git a/pyvista/plotting/qt_plotting.py b/pyvista/plotting/qt_plotting.py\n--- a/pyvista/plotting/qt_plotting.py\n+++ b/pyvista/plotting/qt_plotting.py\n@@ -311,7 +311,7 @@ def __init__(self, parent=None, title=None, shape=(1, 1), off_screen=None,\n                  border=None, border_color='k', border_width=2.0,\n                  multi_samples=None, line_smoothing=False,\n                  point_smoothing=False, polygon_smoothing=False,\n-                 splitting_position=None, **kwargs):\n+                 splitting_position=None):\n         \"\"\" Initialize Qt interactor \"\"\"\n         if not has_pyqt:\n             raise AssertionError('Requires PyQt5')\ndiff --git a/pyvista/plotting/widgets.py b/pyvista/plotting/widgets.py\n--- a/pyvista/plotting/widgets.py\n+++ b/pyvista/plotting/widgets.py\n@@ -13,7 +13,7 @@ class WidgetHelper(object):\n \n     def add_box_widget(self, callback, bounds=None, factor=1.25,\n                        rotation_enabled=True, color=None, use_planes=False,\n-                       outline_translation=True, pass_widget=False, **kwargs):\n+                       outline_translation=True, pass_widget=False):\n         \"\"\"Add a box widget to the scene. This is useless without a callback\n         function. You can pass a callable function that takes a single\n         argument, the PolyData box output from this widget, and performs a\n@@ -174,7 +174,7 @@ def add_plane_widget(self, callback, normal='x', origin=None,\n                          assign_to_axis=None, tubing=False,\n                          outline_translation=False,\n                          origin_translation=True, implicit=True,\n-                         pass_widget=False, test_callback=True, **kwargs):\n+                         pass_widget=False, test_callback=True):\n         \"\"\"Add a plane widget to the scene. This is useless without a callback\n         function. You can pass a callable function that takes two\n         arguments, the normal and origin of the plane in that order output\n@@ -492,7 +492,7 @@ def add_mesh_slice_orthogonal(self, mesh, generate_triangles=False,\n \n     def add_line_widget(self, callback, bounds=None, factor=1.25,\n                         resolution=100, color=None, use_vertices=False,\n-                        pass_widget=False, **kwargs):\n+                        pass_widget=False):\n         \"\"\"Add a line widget to the scene. This is useless without a callback\n         function. You can pass a callable function that takes a single\n         argument, the PolyData line output from this widget, and performs a\n@@ -825,7 +825,7 @@ def callback(value):\n     def add_spline_widget(self, callback, bounds=None, factor=1.25,\n                           n_hanldes=5, resolution=25, color=\"yellow\",\n                           show_ribbon=False, ribbon_color=\"pink\",\n-                          ribbon_opacity=0.5, pass_widget=False, **kwargs):\n+                          ribbon_opacity=0.5, pass_widget=False):\n         \"\"\"Create and add a spline widget to the scene. Use the bounds\n         argument to place this widget. Several \"handles\" are used to control a\n         parametric function for building this spline. Click directly on the\ndiff --git a/pyvista/utilities/__init__.py b/pyvista/utilities/__init__.py\n--- a/pyvista/utilities/__init__.py\n+++ b/pyvista/utilities/__init__.py\n@@ -1,8 +1,8 @@\n-from .errors import (Observer, Report, send_errors_to_logging,\n-                     set_error_output_file)\n+from .errors import (Observer, Report, assert_empty_kwargs,\n+                     send_errors_to_logging, set_error_output_file)\n from .features import *\n from .fileio import *\n from .geometric_objects import *\n from .parametric_objects import *\n from .sphinx_gallery import Scraper, _get_sg_image_scraper\n-from .utilities import *\n+from .helpers import *\ndiff --git a/pyvista/utilities/errors.py b/pyvista/utilities/errors.py\n--- a/pyvista/utilities/errors.py\n+++ b/pyvista/utilities/errors.py\n@@ -2,6 +2,7 @@\n import os\n import re\n import scooby\n+import sys\n \n import vtk\n \n@@ -125,3 +126,21 @@ def __init__(self, additional=None, ncol=3, text_width=80, sort=False):\n         scooby.Report.__init__(self, additional=additional, core=core,\n                                optional=optional, ncol=ncol,\n                                text_width=text_width, sort=sort)\n+\n+\n+def assert_empty_kwargs(**kwargs):\n+    \"\"\"An internal helper to assert that all keyword arguments have been used.\n+    If any keyword arguments are passed, a ``TypeError`` is raised.\n+    \"\"\"\n+    n = len(kwargs)\n+    if n == 0:\n+        return True\n+    caller = sys._getframe(1).f_code.co_name\n+    keys = list(kwargs.keys())\n+    bad_arguments = \"[\" + (\"{}, \" * (n - 1) + \"{}\").format(*keys) + \"]\"\n+    if n == 1:\n+        grammar = \"is an invalid keyword argument\"\n+    else:\n+        grammar = \"are invalid keyword arguments\"\n+    message = \"{} {} for `{}`\".format(bad_arguments, grammar, caller)\n+    raise TypeError(message)\ndiff --git a/pyvista/utilities/geometric_objects.py b/pyvista/utilities/geometric_objects.py\n--- a/pyvista/utilities/geometric_objects.py\n+++ b/pyvista/utilities/geometric_objects.py\n@@ -17,6 +17,7 @@\n import vtk\n \n import pyvista\n+from pyvista.utilities import assert_empty_kwargs\n \n \n NORMALS = {\n@@ -51,7 +52,7 @@ def translate(surf, center=[0., 0., 0.], direction=[1., 0., 0.]):\n \n \n def Cylinder(center=(0.,0.,0.), direction=(1.,0.,0.), radius=0.5, height=1.0,\n-             resolution=100, **kwargs):\n+             resolution=100, capping=True, **kwargs):\n \n     \"\"\"\n     Create the surface of a cylinder.\n@@ -91,7 +92,8 @@ def Cylinder(center=(0.,0.,0.), direction=(1.,0.,0.), radius=0.5, height=1.0,\n     >>> cylinder.plot() # doctest:+SKIP\n \n     \"\"\"\n-    capping = kwargs.get('capping', kwargs.get('cap_ends', True))\n+    capping = kwargs.pop('cap_ends', capping)\n+    assert_empty_kwargs(**kwargs)\n     cylinderSource = vtk.vtkCylinderSource()\n     cylinderSource.SetRadius(radius)\n     cylinderSource.SetHeight(height)\n@@ -528,12 +530,12 @@ def Text3D(string, depth=0.5):\n     return pyvista.wrap(tri_filter.GetOutput())\n \n \n-def SuperToroid(**kwargs):\n+def SuperToroid(*args, **kwargs):\n     \"\"\"DEPRECATED: use :func:`pyvista.ParametricSuperToroid`\"\"\"\n     raise RuntimeError('use `pyvista.ParametricSuperToroid` instead')\n \n \n-def Ellipsoid(**kwargs):\n+def Ellipsoid(*args, **kwargs):\n     \"\"\"DEPRECATED: use :func:`pyvista.ParametricEllipsoid`\"\"\"\n     raise RuntimeError('use `pyvista.ParametricEllipsoid` instead')\n \ndiff --git a/pyvista/utilities/utilities.py b/pyvista/utilities/helpers.py\nsimilarity index 100%\nrename from pyvista/utilities/utilities.py\nrename to pyvista/utilities/helpers.py\ndiff --git a/pyvista/utilities/parametric_objects.py b/pyvista/utilities/parametric_objects.py\n--- a/pyvista/utilities/parametric_objects.py\n+++ b/pyvista/utilities/parametric_objects.py\n@@ -3,6 +3,7 @@\n import vtk\n \n import pyvista\n+from pyvista.utilities import assert_empty_kwargs\n \n from .geometric_objects import translate\n \n@@ -361,7 +362,15 @@ def ParametricEllipsoid(xradius=None, yradius=None, zradius=None,\n     >>> mesh.plot(color='w', smooth_shading=True)  # doctest:+SKIP\n     \"\"\"\n     parametric_function = vtk.vtkParametricEllipsoid()\n-    parametric_keywords(parametric_function, **kwargs)\n+    parametric_keywords(parametric_function, min_u=kwargs.pop(\"min_u\", 0),\n+                        max_u=kwargs.pop(\"max_u\", 2*pi),\n+                        min_v=kwargs.pop(\"min_v\", 0.0),\n+                        max_v=kwargs.pop(\"max_v\", 2*pi),\n+                        join_u=kwargs.pop(\"join_u\", False),\n+                        join_v=kwargs.pop(\"join_v\", False),\n+                        twist_u=kwargs.pop(\"twist_u\", False),\n+                        twist_v=kwargs.pop(\"twist_v\", False),\n+                        clockwise=kwargs.pop(\"clockwise\", True),)\n \n     if xradius is not None:\n         parametric_function.SetXRadius(xradius)\n@@ -964,7 +973,7 @@ def ParametricTorus(ringradius=None, crosssectionradius=None, **kwargs):\n \n def parametric_keywords(parametric_function, min_u=0, max_u=2*pi,\n                         min_v=0.0, max_v=2*pi, join_u=False, join_v=False,\n-                        twist_u=False, twist_v=False, clockwise=True, **kwargs):\n+                        twist_u=False, twist_v=False, clockwise=True):\n     \"\"\"Applys keyword arguments to a parametric function.\n \n     Parameters\n@@ -1013,7 +1022,7 @@ def parametric_keywords(parametric_function, min_u=0, max_u=2*pi,\n \n \n def surface_from_para(parametric_function, u_res=100, v_res=100,\n-                      w_res=100, **kwargs):\n+                      w_res=100):\n     \"\"\"Construct a mesh from a parametric function.\n \n     Parameters\n", "test_patch": "diff --git a/tests/test_plotting.py b/tests/test_plotting.py\n--- a/tests/test_plotting.py\n+++ b/tests/test_plotting.py\n@@ -837,3 +837,21 @@ def test_fail_plot_table():\n     with pytest.raises(TypeError):\n         plotter = pyvista.Plotter(off_screen=OFF_SCREEN)\n         plotter.add_mesh(table)\n+\n+\n+@pytest.mark.skipif(NO_PLOTTING, reason=\"Requires system to support plotting\")\n+def test_bad_keyword_arguments():\n+    \"\"\"Make sure bad keyword arguments raise an error\"\"\"\n+    mesh = examples.load_uniform()\n+    with pytest.raises(TypeError):\n+        pyvista.plot(mesh, foo=5, off_screen=OFF_SCREEN)\n+    with pytest.raises(TypeError):\n+        pyvista.plot(mesh, scalar=mesh.active_scalar_name, off_screen=OFF_SCREEN)\n+    with pytest.raises(TypeError):\n+        plotter = pyvista.Plotter(off_screen=OFF_SCREEN)\n+        plotter.add_mesh(mesh, scalar=mesh.active_scalar_name)\n+        plotter.show()\n+    with pytest.raises(TypeError):\n+        plotter = pyvista.Plotter(off_screen=OFF_SCREEN)\n+        plotter.add_mesh(mesh, foo=\"bad\")\n+        plotter.show()\ndiff --git a/tests/test_utilities.py b/tests/test_utilities.py\n--- a/tests/test_utilities.py\n+++ b/tests/test_utilities.py\n@@ -6,8 +6,9 @@\n \n import pyvista\n from pyvista import examples as ex\n-from pyvista import utilities\n-from pyvista import fileio\n+from pyvista.utilities import helpers\n+from pyvista.utilities import fileio\n+from pyvista.utilities import errors\n \n # Only set this here just the once.\n pyvista.set_error_output_file(os.path.join(os.path.dirname(__file__), 'ERROR_OUTPUT.txt'))\n@@ -17,13 +18,13 @@ def test_createvectorpolydata_error():\n     orig = np.random.random((3, 1))\n     vec = np.random.random((3, 1))\n     with pytest.raises(Exception):\n-        utilities.vector_poly_data(orig, vec)\n+        helpers.vector_poly_data(orig, vec)\n \n \n def test_createvectorpolydata_1D():\n     orig = np.random.random(3)\n     vec = np.random.random(3)\n-    vdata = utilities.vector_poly_data(orig, vec)\n+    vdata = helpers.vector_poly_data(orig, vec)\n     assert np.any(vdata.points)\n     assert np.any(vdata.point_arrays['vectors'])\n \n@@ -31,7 +32,7 @@ def test_createvectorpolydata_1D():\n def test_createvectorpolydata():\n     orig = np.random.random((100, 3))\n     vec = np.random.random((100, 3))\n-    vdata = utilities.vector_poly_data(orig, vec)\n+    vdata = helpers.vector_poly_data(orig, vec)\n     assert np.any(vdata.points)\n     assert np.any(vdata.point_arrays['vectors'])\n \n@@ -72,12 +73,12 @@ def test_get_array():\n     grid._add_point_array(oarr, 'other')\n     farr = np.random.rand(grid.n_points * grid.n_cells)\n     grid._add_field_array(farr, 'field_data')\n-    assert np.allclose(carr, utilities.get_array(grid, 'test_data', preference='cell'))\n-    assert np.allclose(parr, utilities.get_array(grid, 'test_data', preference='point'))\n-    assert np.allclose(oarr, utilities.get_array(grid, 'other'))\n-    assert utilities.get_array(grid, 'foo') is None\n-    assert utilities.get_array(grid, 'test_data', preference='field') is None\n-    assert np.allclose(farr, utilities.get_array(grid, 'field_data', preference='field'))\n+    assert np.allclose(carr, helpers.get_array(grid, 'test_data', preference='cell'))\n+    assert np.allclose(parr, helpers.get_array(grid, 'test_data', preference='point'))\n+    assert np.allclose(oarr, helpers.get_array(grid, 'other'))\n+    assert helpers.get_array(grid, 'foo') is None\n+    assert helpers.get_array(grid, 'test_data', preference='field') is None\n+    assert np.allclose(farr, helpers.get_array(grid, 'field_data', preference='field'))\n \n \n \n@@ -85,11 +86,11 @@ def test_get_array():\n def test_is_inside_bounds():\n     data = ex.load_uniform()\n     bnds = data.bounds\n-    assert utilities.is_inside_bounds((0.5, 0.5, 0.5), bnds)\n-    assert not utilities.is_inside_bounds((12, 5, 5), bnds)\n-    assert not utilities.is_inside_bounds((5, 12, 5), bnds)\n-    assert not utilities.is_inside_bounds((5, 5, 12), bnds)\n-    assert not utilities.is_inside_bounds((12, 12, 12), bnds)\n+    assert helpers.is_inside_bounds((0.5, 0.5, 0.5), bnds)\n+    assert not helpers.is_inside_bounds((12, 5, 5), bnds)\n+    assert not helpers.is_inside_bounds((5, 12, 5), bnds)\n+    assert not helpers.is_inside_bounds((5, 5, 12), bnds)\n+    assert not helpers.is_inside_bounds((12, 12, 12), bnds)\n \n \n def test_get_sg_image_scraper():\n@@ -166,3 +167,13 @@ def test_transform_vectors_sph_to_cart():\n         [uu[-1, -1], vv[-1, -1], ww[-1, -1]],\n         [67.80403533828323, 360.8359915416445, -70000.0],\n     )\n+\n+def test_assert_empty_kwargs():\n+    kwargs = {}\n+    assert errors.assert_empty_kwargs(**kwargs)\n+    with pytest.raises(TypeError):\n+        kwargs = {\"foo\":6}\n+        errors.assert_empty_kwargs(**kwargs)\n+    with pytest.raises(TypeError):\n+        kwargs = {\"foo\":6, \"goo\":\"bad\"}\n+        errors.assert_empty_kwargs(**kwargs)\n", "problem_statement": "Strictly enforce keyword arguments\nI see folks quite often forget the s in the `scalars` argument for the `BasePlotter.add_mesh()` method. We should allow `scalar` as an alias much like how we allow `rng` and `clim` for the colorbar range/limits\n", "hints_text": "I would disagree. 2 arguments:\r\n\r\n`There should be one-- and preferably only one --obvious way to do it.` as says https://www.python.org/dev/peps/pep-0020/\r\n\r\nit makes documentation harder.\nRaise an exception to tell the user to use `scalars` when `scalar` is used as a keyword arg. \nOkay, I'm convinced. \r\n\r\nWe may want to do some keyword refactoring and not just allow an endless dict of `**kwargs` to be passed to `add_mesh`. For example, you can pass all types of typos and not have an error thrown:\r\n\r\n```py\r\nimport pyvista as pv\r\np = pv.Plotter()\r\np.add_mesh(pv.Cube(), my_crazy_argument=\"foooooooo\")\r\np.show()\r\n```\r\n\r\nThere are several places in the API where this can happen that will need to be refactored \nWhat we could do is pop out the allowed aliases like `rng` and then raise an exception if the `kwargs` dict is not empty that might say something like:\r\n\r\n> \"[list of arguments still in the kwargs] are not valid keyword arguments.\"", "created_at": "2019-10-31T03:34:40Z", "version": "0.22", "FAIL_TO_PASS": ["tests/test_plotting.py::test_invalid_color", "tests/test_plotting.py::test_invalid_font", "tests/test_plotting.py::test_themes", "tests/test_utilities.py::test_createvectorpolydata_error", "tests/test_utilities.py::test_createvectorpolydata_1D", "tests/test_utilities.py::test_createvectorpolydata", "tests/test_utilities.py::test_read", "tests/test_utilities.py::test_is_inside_bounds", "tests/test_utilities.py::test_get_sg_image_scraper", "tests/test_utilities.py::test_report", "tests/test_utilities.py::test_grid_from_sph_coords", "tests/test_utilities.py::test_transform_vectors_sph_to_cart", "tests/test_utilities.py::test_assert_empty_kwargs"], "PASS_TO_PASS": [], "environment_setup_commit": "1affc211d8b93024fa3e163584f235975d3536ed"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1674", "base_commit": "e77e6586fa38e6f7e98efca80d560d0fea8a9669", "patch": "diff --git a/pydicom/cli/main.py b/pydicom/cli/main.py\n--- a/pydicom/cli/main.py\n+++ b/pydicom/cli/main.py\n@@ -10,10 +10,11 @@\n import argparse\n import pkg_resources\n import re\n+import sys\n from typing import Tuple, cast, List, Any, Dict, Optional, Callable\n \n from pydicom import dcmread\n-from pydicom.data.data_manager import get_testdata_file\n+from pydicom.data.data_manager import get_charset_files, get_testdata_file\n from pydicom.dataset import Dataset\n \n \n@@ -132,6 +133,14 @@ def filespec_parser(filespec: str) -> List[Tuple[Dataset, Any]]:\n     except NotImplementedError:  # will get this if absolute path passed\n         pydicom_filename = \"\"\n \n+    # Check if filename is in charset files\n+    if not pydicom_filename:\n+        try:\n+            char_filenames = get_charset_files(filename)\n+            pydicom_filename = char_filenames[0]\n+        except NotImplementedError:  # will get this if absolute path passed\n+            pass\n+\n     if prefix == \"pydicom\":\n         filename = pydicom_filename\n \n@@ -201,8 +210,11 @@ def main(args: Optional[List[str]] = None) -> None:\n     \"\"\"\n     global subparsers\n \n+    py_version = sys.version.split()[0]\n+\n     parser = argparse.ArgumentParser(\n-        prog=\"pydicom\", description=\"pydicom command line utilities\"\n+        prog=\"pydicom\",\n+        description=f\"pydicom command line utilities (Python {py_version})\"\n     )\n     subparsers = parser.add_subparsers(help=\"subcommand help\")\n \ndiff --git a/pydicom/util/codify.py b/pydicom/util/codify.py\n--- a/pydicom/util/codify.py\n+++ b/pydicom/util/codify.py\n@@ -18,7 +18,8 @@\n import os.path\n import re\n import sys\n-from typing import Optional, List, Callable\n+from typing import Optional, List, Callable, cast\n+from collections import deque\n \n import pydicom\n from pydicom.datadict import dictionary_keyword\n@@ -80,7 +81,8 @@ def code_dataelem(\n     dataelem: DataElement,\n     dataset_name: str = \"ds\",\n     exclude_size: Optional[int] = None,\n-    include_private: bool = False\n+    include_private: bool = False,\n+    var_names: Optional[deque] = None\n ) -> str:\n     \"\"\"Code lines for a single DICOM data element\n \n@@ -96,7 +98,8 @@ def code_dataelem(\n         will only have a commented string for a value,\n         causing a syntax error when the code is run,\n         and thus prompting the user to remove or fix that line.\n-\n+    var_names: Union[deque, None]\n+        Used internally to ensure unique variable names in nested sequences.\n     Returns\n     -------\n     str\n@@ -106,7 +109,8 @@ def code_dataelem(\n \n     if dataelem.VR == VR.SQ:\n         return code_sequence(\n-            dataelem, dataset_name, exclude_size, include_private\n+            dataelem, dataset_name, exclude_size, include_private,\n+            var_names=var_names\n         )\n \n     # If in DICOM dictionary, set using the keyword\n@@ -143,6 +147,7 @@ def code_sequence(\n     exclude_size: Optional[int] = None,\n     include_private: bool = False,\n     name_filter: Callable[[str], str] = default_name_filter,\n+    var_names: Optional[deque] = None,\n ) -> str:\n     \"\"\"Code lines for recreating a Sequence data element\n \n@@ -162,6 +167,8 @@ def code_sequence(\n     name_filter: Callable[[str], str]\n         A callable taking a sequence name or sequence item name, and returning\n         a shorter name for easier code reading\n+    var_names: Union[deque, None]\n+        Used internally to ensure unique variable names in nested sequences.\n \n     Returns\n     -------\n@@ -169,6 +176,15 @@ def code_sequence(\n         A string containing code lines to recreate a DICOM sequence\n     \"\"\"\n \n+    # Normally var_names is given from code_dataset, but for some tests need\n+    #   to initialize it\n+    if var_names is None:\n+        var_names = deque()\n+\n+    def unique_name(name: str) -> str:\n+        name_count = cast(deque, var_names).count(name) - 1\n+        return name if name_count == 0 else name + f\"_{name_count}\"\n+\n     lines = []\n     seq = dataelem.value\n     seq_name = dataelem.name\n@@ -183,8 +199,11 @@ def code_sequence(\n     lines.append(\"# \" + seq_name)\n \n     # Code line to create a new Sequence object\n-    if name_filter:\n-        seq_var = name_filter(seq_keyword)\n+    seq_var = name_filter(seq_keyword)\n+    var_names.append(seq_var)\n+    orig_seq_var = seq_var\n+    seq_var = unique_name(seq_var)\n+\n     lines.append(seq_var + \" = Sequence()\")\n \n     # Code line to add the sequence to its parent\n@@ -208,14 +227,29 @@ def code_sequence(\n         lines.append(\"# \" + seq_name + \": \" + seq_item_name + \" \" + index_str)\n \n         # Determine the variable name to use for the sequence item (dataset)\n-        ds_name = seq_var.replace(\"_sequence\", \"\") + index_str\n+        ds_name = orig_seq_var.replace(\"_sequence\", \"\") + index_str\n \n-        # Code the sequence item\n-        code_item = code_dataset(ds, ds_name, exclude_size, include_private)\n-        lines.append(code_item)\n+        # Append \"_#\" if name already in use (in parent sequences)\n+        var_names.append(ds_name)\n+        ds_name = unique_name(ds_name)\n+\n+        # Code the sequence item dataset\n+        code_item = code_dataset(\n+            ds, ds_name, exclude_size, include_private, var_names=var_names\n+        )\n \n-        # Code the line to append the item to its parent sequence\n-        lines.append(seq_var + \".append(\" + ds_name + \")\")\n+        # Remove variable name from stored list, this dataset complete\n+        var_names.pop()\n+\n+        # Code dataset creation and appending that to sequence, then the rest\n+        # This keeps the logic close together, rather than after many items set\n+        code_split = code_item.splitlines()\n+        lines.append(code_split[0])  # \"<ds_name> = Dataset()\"\n+        lines.append(f\"{seq_var}.append({ds_name})\")\n+        lines.extend(code_split[1:])\n+\n+    # Remove sequence variable name we've used\n+    var_names.pop()\n \n     # Join the lines and return a single string\n     return line_term.join(lines)\n@@ -227,6 +261,7 @@ def code_dataset(\n     exclude_size: Optional[int] = None,\n     include_private: bool = False,\n     is_file_meta: bool = False,\n+    var_names: Optional[deque] = None\n ) -> str:\n     \"\"\"Return Python code for creating `ds`.\n \n@@ -245,6 +280,8 @@ def code_dataset(\n         data elements will be coded.\n     is_file_meta : bool, optional\n         ``True`` if `ds` contains file meta information elements.\n+    var_names: deque, optional\n+        Used internally to ensure unique variable names in nested sequences.\n \n     Returns\n     -------\n@@ -252,8 +289,12 @@ def code_dataset(\n         The codified dataset.\n     \"\"\"\n \n+    if var_names is None:\n+        var_names = deque()\n     lines = []\n+\n     ds_class = \" = FileMetaDataset()\" if is_file_meta else \" = Dataset()\"\n+\n     lines.append(dataset_name + ds_class)\n     for dataelem in ds:\n         # If a private data element and flag says so, skip it and go to next\n@@ -261,7 +302,8 @@ def code_dataset(\n             continue\n         # Otherwise code the line and add it to the lines list\n         code_line = code_dataelem(\n-            dataelem, dataset_name, exclude_size, include_private\n+            dataelem, dataset_name, exclude_size, include_private,\n+            var_names=var_names\n         )\n         lines.append(code_line)\n         # Add blank line if just coded a sequence\n@@ -270,6 +312,7 @@ def code_dataset(\n     # If sequence was end of this dataset, remove the extra blank line\n     if len(lines) and lines[-1] == \"\":\n         lines.pop()\n+\n     # Join all the code lines and return them\n     return line_term.join(lines)\n \n@@ -313,8 +356,8 @@ def code_file_from_dataset(\n \n     Parameters\n     ----------\n-    filename : str\n-        Complete path and filename of a DICOM file to convert\n+    ds : Dataset\n+        A pydicom Dataset to convert\n     exclude_size : Union[int,None]\n         If not None, values longer than this (in bytes)\n         will only have a commented string for a value,\n@@ -336,6 +379,7 @@ def code_file_from_dataset(\n     filename = ds.get(\"filename\")\n     identifier = f\"DICOM file '{filename}'\" if filename else \"non-file dataset\"\n \n+    lines.append(\"# -*- coding: utf-8 -*-\")\n     lines.append(f\"# Coded version of {identifier}\")\n     lines.append(\"# Produced by pydicom codify utility script\")\n \n@@ -385,7 +429,7 @@ def set_parser_arguments(\n     parser.add_argument(\n         \"outfile\",\n         nargs=\"?\",\n-        type=argparse.FileType(\"w\"),\n+        type=argparse.FileType(\"w\", encoding=\"UTF-8\"),\n         help=(\n             \"Filename to write Python code to, if not specified then code is \"\n             \"written to stdout\"\n", "test_patch": "diff --git a/pydicom/tests/test_cli.py b/pydicom/tests/test_cli.py\n--- a/pydicom/tests/test_cli.py\n+++ b/pydicom/tests/test_cli.py\n@@ -1,3 +1,4 @@\n+# -*- coding: utf-8 -*-\n # Copyright 2020 pydicom authors. See LICENSE file for details.\n \"\"\"Tests for command-line interface\"\"\"\n \n@@ -138,6 +139,13 @@ def test_codify_data_element(self, capsys):\n         with pytest.raises(NotImplementedError):\n             main(\"codify pydicom::rtplan.dcm::RTPlanLabel\".split())\n \n+    def test_codify_UTF8(self, capsys):\n+        \"\"\"CLI `codify` command creates code with utf-8 characters\"\"\"\n+        main(f\"codify pydicom::chrFren.dcm\".split())\n+        out, _ = capsys.readouterr()\n+        assert out.startswith(\"# -*- coding: utf-8 -*-\")\n+        assert \"Buc^J\u00e9r\u00f4me\" in out\n+\n     def test_help(self, capsys):\n         \"\"\"CLI `help` command gives expected output\"\"\"\n         # With subcommand\ndiff --git a/pydicom/tests/test_util.py b/pydicom/tests/test_util.py\n--- a/pydicom/tests/test_util.py\n+++ b/pydicom/tests/test_util.py\n@@ -5,12 +5,11 @@\n \n import pytest\n \n-from pydicom import config, dcmread\n+from pydicom import config, dcmread, Dataset, Sequence\n from pydicom import filereader\n from pydicom._private_dict import private_dictionaries\n from pydicom.data import get_testdata_file\n from pydicom.dataelem import DataElement\n-from pydicom.dataset import Dataset\n from pydicom.tag import Tag\n from pydicom.uid import (\n     ImplicitVRLittleEndian, ExplicitVRBigEndian, ExplicitVRLittleEndian\n@@ -23,6 +22,7 @@\n     default_name_filter,\n     code_imports,\n     code_dataelem,\n+    code_dataset,\n     main as codify_main,\n )\n from pydicom.util.dump import *\n@@ -134,15 +134,34 @@ def test_code_sequence(self):\n             \"\\n\"\n             \"# Control Point Sequence: Control Point 1\\n\"\n             \"cp1 = Dataset()\\n\"\n-            \"cp1.PatientID = '1234'\\n\"\n-            \"cp_sequence.append(cp1)\"\n+            \"cp_sequence.append(cp1)\\n\"\n+            \"cp1.PatientID = '1234'\"\n         )\n \n         assert out == code_dataelem(elem)\n \n-    def test_code_dataset(self):\n-        \"\"\"Test utils.codify.code_dataset\"\"\"\n-        pass\n+    def test_codify_recurring_keyword(self):\n+        \"\"\"Test utils.codify.code_dataset with same keyword nested\"\"\"\n+        # Create fake Dataset with repeated DICOM keyword nested\n+        # (0040, a730)  Content Sequence  1 item(s) ----\n+        #    (0040, a040) Value Type                          CS: 'CODE'\n+        #    (0040, a730)  Content Sequence  1 item(s) ----\n+        #       (0040, a040) Value Type                          CS: 'CODE'\n+\n+        ds = Dataset()\n+        ds.ContentSequence = seq1 = Sequence()\n+        seq1.append(Dataset())\n+        seq1[0].ValueType = \"CODE\"\n+        seq1[0].ContentSequence = seq2 = Sequence()\n+        seq2.append(Dataset())\n+        seq2[0].ValueType = \"CODE_1\"\n+        ds_code = code_dataset(ds)\n+\n+        # normal 1st use of var name\n+        assert \"content1.ValueType = 'CODE'\" in ds_code\n+\n+        # Nested item of same name should have subscript\n+        assert \"content1_1.ValueType = 'CODE_1'\" in ds_code\n \n     def test_code_file(self, capsys):\n         \"\"\"Test utils.codify.code_file\"\"\"\n", "problem_statement": "Codify not generating content sequences correctly\n**Describe the bug**\r\nI am trying to generate a radiation dose structure report. I ran Codify on an existing RDSR to generate a template. The sequence content is reproduced but does not seem to be attached  to the base dataset. When I run the generated python file the dicom file it saves has no sequence content information.\r\n\r\n**Expected behavior**\r\nI expect the dicom file generated by the python code from Codify to be similar to the original file.\r\n\r\n**Steps To Reproduce**\r\n$ python codify X-RayRadiationDoseReport001_ESR.dcm rdsr.py\r\n$ python rsdr.py\r\n\r\nI am not able to attached the above files but can supply them.\r\n\r\n**Your environment**\r\nmodule       | version\r\n------       | -------\r\nplatform     | Linux-5.18.7-200.fc36.x86_64-x86_64-with-glibc2.35\r\nPython       | 3.10.5 (main, Jun  9 2022, 00:00:00) [GCC 12.1.1 20220507 (Red Hat 12.1.1-1)]\r\npydicom      | 2.3.0\r\ngdcm         | _module not found_\r\njpeg_ls      | _module not found_\r\nnumpy        | 1.22.4\r\nPIL          | 9.2.0\r\npylibjpeg    | _module not found_\r\nopenjpeg     | _module not found_\r\nlibjpeg      | _module not found_\r\n\r\nRegards\r\nAlan\n", "hints_text": "Hi Alan,\r\n\r\nI'm happy to look into this, but may need a file to work with.  But perhaps we can first have a look at the generated code, which normally immediately attaches any sequence it creates.  For example:\r\n\r\n```\r\n$ pydicom codify pydicom::rtplan.dcm\r\n```\r\nGives lines like\r\n\r\n```python\r\n# Beam Sequence\r\nbeam_sequence = Sequence()\r\nds.BeamSequence = beam_sequence\r\n```\r\n\r\nand then proceeds to create a new Dataset, set its data elements, and eventually `append` it to the sequence:\r\n\r\n```python\r\n# Beam Sequence: Beam 1\r\nbeam1 = Dataset()\r\nbeam1.Manufacturer = 'Linac co.'\r\nbeam1.InstitutionName = 'Here'\r\n.\r\n.\r\n.\r\nbeam_sequence.append(beam1)\r\n```\r\nCan you check your generated code for those kinds of patterns and let us know what you see?\r\n\r\n\nHi Darcy\r\n\r\nThanks for the guidance. Stepping through the code I've picked up two problems. The first has to do with nested content. Eg. if the DICOM file has:\r\n![NestedContent](https://user-images.githubusercontent.com/34305581/185054806-ed2559c6-2e0f-46e5-a9de-085ada6e7753.png)\r\n\r\nThe second Content1 overwrites the first. See lines 121-191 in the attached file.\r\n\r\n[rdsr.txt](https://github.com/pydicom/pydicom/files/9356766/rdsr.txt) Rename this file to rsdr.py.\r\n\r\nThe second problem is more puzzling. In lines 121-191 we have:\r\n\r\n```\r\ncontent_sequence = Sequence()\r\nds.ContentSequence = content_sequence\r\n\r\n# Content Sequence: Content 1\r\ncontent1 = Dataset()\r\ncontent1.add_new((0x0040, 0x0000), 'UL', 512)\r\ncontent1.RelationshipType = 'HAS CONCEPT MOD'\r\ncontent1.ValueType = 'CODE'\r\n.\r\n.\r\n.\r\ncontent_sequence.append(content1)\r\n```\r\nAfter stepping though this ds.ContentSequence is still empty even though there are definitely items in content1 and the items were added to content_sequence. It is not related to the nested content as content2 (lines 193 - 223 in rdsr.py) which has no nested content shows the same behaviour.\r\n\r\nIf I shift the line \"ds.ContentSequence = content_sequence\" to just before the save statement at the end of the file the script crashes while saving with:\r\nRecursionError: maximum recursion depth exceeded while calling a Python object\r\nThis suggests two things:\r\n1) The previous code is generating a nested loop somewhere.\r\n2) The line \"ds.ContentSequence = content_sequence\" appears to be creating a copy of content_sequence. My understanding of python is that an assigning objects should merely update pointers, but this does not appear to be the case.\r\n\r\nI would suggest generating content names that reflect the content level if possible, eg. content1-1.\r\n\r\nRegards\r\nAlan\nAlan, if you are unable to share your RDSR for testing, do you want to check the behaviour is the same with one of the OpenREM test RDSR files and use that for Darcy or anyone else to work on? One of the more straight forward ones, maybe [this Siemens Flash one](https://bitbucket.org/openrem/openrem/src/develop/openrem/remapp/tests/test_files/CT-RDSR-Siemens_Flash-TAP-SS.dcm)?\nHi Ed\r\nI ran Codify on  CT-RDSR-Siemens_Flash-TAP-SS.dcm as you suggested. This gave the attached file:\r\n[CT-RDSR-Siemens_Flash-TAP-SS.py.txt](https://github.com/pydicom/pydicom/files/9359044/CT-RDSR-Siemens_Flash-TAP-SS.py.txt) rename to .py\r\nRunning this file produces a dicom file 10 times smaller than the original file (attached)\r\n[CT-RDSR-Siemens_Flash-TAP-SS_from_codify.dcm.txt](https://github.com/pydicom/pydicom/files/9359049/CT-RDSR-Siemens_Flash-TAP-SS_from_codify.dcm.txt) Rename to .dcm\r\nLooking at this in a dicom viewer the sequence content is gone.\r\nHope this helps\r\nRegards\r\nAlan\r\n\r\n\r\n\n> The second Content1 overwrites the first.\r\n> ...\r\n> I would suggest generating content names that reflect the content level if possible, eg. content1-1.\r\n\r\nRight, it makes sense - the code didn't consider that a nested item could have the same name.  I can update that.\r\n\r\n\r\n> RecursionError: maximum recursion depth exceeded while calling a Python object\r\n\r\nNot too surprising - I think I hit recursion problems when I first wrote codify.\r\n\r\nThanks for the detailed investigation and example files (thanks also @edmcdonagh  for the suggestion).  I can set up some failing tests with those and then see about fixing them.  I've got some time in the next couple of days, hopefully it isn't too difficult and I can get a PR soon.", "created_at": "2022-08-18T17:35:33Z", "version": "2.3", "FAIL_TO_PASS": ["pydicom/tests/test_cli.py::TestCLIcall::test_codify_UTF8", "pydicom/tests/test_util.py::TestCodify::test_code_sequence", "pydicom/tests/test_util.py::TestCodify::test_codify_recurring_keyword"], "PASS_TO_PASS": ["pydicom/tests/test_cli.py::TestFilespec::test_syntax[extra:colon]", "pydicom/tests/test_cli.py::TestFilespec::test_syntax[no_callable()]", "pydicom/tests/test_cli.py::TestFilespec::test_syntax[no_equals", "pydicom/tests/test_cli.py::TestFilespec::test_syntax[BeamSequence[0]extra]", "pydicom/tests/test_cli.py::TestFilespec::test_syntax[BeamSequence[x]]", "pydicom/tests/test_cli.py::TestFilespec::test_elem_not_exists[NotThere]", "pydicom/tests/test_cli.py::TestFilespec::test_elem_not_exists[BeamSequenceXX]", "pydicom/tests/test_cli.py::TestFilespec::test_elem_not_exists[BeamDose]", "pydicom/tests/test_cli.py::TestFilespec::test_bad_index[BeamSequence[42]]", "pydicom/tests/test_cli.py::TestFilespec::test_bad_index[BeamSequence[-42]]", "pydicom/tests/test_cli.py::TestFilespec::test_offers_pydicom_testfile", "pydicom/tests/test_cli.py::TestFilespec::test_colons", "pydicom/tests/test_cli.py::TestFilespecElementEval::test_correct_values", "pydicom/tests/test_cli.py::TestCLIcall::test_bare_command", "pydicom/tests/test_cli.py::TestCLIcall::test_codify_command", "pydicom/tests/test_cli.py::TestCLIcall::test_codify_data_element", "pydicom/tests/test_cli.py::TestCLIcall::test_help", "pydicom/tests/test_cli.py::TestCLIcall::test_show_command", "pydicom/tests/test_cli.py::TestCLIcall::test_show_options", "pydicom/tests/test_util.py::TestCodify::test_camel_to_underscore", "pydicom/tests/test_util.py::TestCodify::test_tag_repr", "pydicom/tests/test_util.py::TestCodify::test_default_name_filter", "pydicom/tests/test_util.py::TestCodify::test_code_imports", "pydicom/tests/test_util.py::TestCodify::test_code_dataelem_standard", "pydicom/tests/test_util.py::TestCodify::test_code_dataelem_exclude_size", "pydicom/tests/test_util.py::TestCodify::test_code_dataelem_private", "pydicom/tests/test_util.py::TestCodify::test_code_dataelem_sequence", "pydicom/tests/test_util.py::TestCodify::test_code_file", "pydicom/tests/test_util.py::TestDump::test_print_character", "pydicom/tests/test_util.py::TestDump::test_filedump", "pydicom/tests/test_util.py::TestDump::test_datadump", "pydicom/tests/test_util.py::TestDump::test_hexdump", "pydicom/tests/test_util.py::TestDump::test_pretty_print", "pydicom/tests/test_util.py::TestFixer::test_fix_separator_callback", "pydicom/tests/test_util.py::TestFixer::test_fix_separator", "pydicom/tests/test_util.py::TestFixer::test_mismatch_callback", "pydicom/tests/test_util.py::TestFixer::test_fix_mismatch", "pydicom/tests/test_util.py::TestHexUtil::test_hex_to_bytes", "pydicom/tests/test_util.py::TestHexUtil::test_bytes_to_hex", "pydicom/tests/test_util.py::TestDataElementCallbackTests::test_bad_separator", "pydicom/tests/test_util.py::TestDataElementCallbackTests::test_impl_vr_comma", "pydicom/tests/test_util.py::TestDataElementCallbackTests::test_null_value_for_fixed_vr", "pydicom/tests/test_util.py::TestDataElementCallbackTests::test_space_delimiter", "pydicom/tests/test_util.py::TestDataElementCallbackTests::test_process_unknown_vr", "pydicom/tests/test_util.py::TestLeanRead::test_explicit_little", "pydicom/tests/test_util.py::TestLeanRead::test_implicit_little", "pydicom/tests/test_util.py::TestLeanRead::test_explicit_big", "pydicom/tests/test_util.py::TestLeanRead::test_no_tsyntax", "pydicom/tests/test_util.py::TestLeanRead::test_no_meta", "pydicom/tests/test_util.py::TestLeanRead::test_UN_sequence"], "environment_setup_commit": "a8be738418dee0a2b93c241fbd5e0bc82f4b8680"}]