[{"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1241", "base_commit": "9d69811e539774f296c2f289839147e741251716", "patch": "diff --git a/pydicom/encaps.py b/pydicom/encaps.py\n--- a/pydicom/encaps.py\n+++ b/pydicom/encaps.py\n@@ -77,13 +77,15 @@ def get_frame_offsets(fp: DicomFileLike) -> Tuple[bool, List[int]]:\n     tag = Tag(fp.read_tag())\n \n     if tag != 0xfffee000:\n-        raise ValueError(\"Unexpected tag '{}' when parsing the Basic Table \"\n-                         \"Offset item.\".format(tag))\n+        raise ValueError(\n+            f\"Unexpected tag '{tag}' when parsing the Basic Table Offset item\"\n+        )\n \n     length = fp.read_UL()\n     if length % 4:\n-        raise ValueError(\"The length of the Basic Offset Table item is not \"\n-                         \"a multiple of 4.\")\n+        raise ValueError(\n+            \"The length of the Basic Offset Table item is not a multiple of 4\"\n+        )\n \n     offsets = []\n     # Always return at least a 0 offset\n@@ -116,18 +118,20 @@ def get_nr_fragments(fp: DicomFileLike) -> int:\n             # Item\n             length = fp.read_UL()\n             if length == 0xFFFFFFFF:\n-                raise ValueError(\"Undefined item length at offset {} when \"\n-                                 \"parsing the encapsulated pixel data \"\n-                                 \"fragments.\".format(fp.tell() - 4))\n+                raise ValueError(\n+                    f\"Undefined item length at offset {fp.tell() - 4} when \"\n+                    \"parsing the encapsulated pixel data fragments\"\n+                )\n             fp.seek(length, 1)\n             nr_fragments += 1\n         elif tag == 0xFFFEE0DD:\n             # Sequence Delimiter\n             break\n         else:\n-            raise ValueError(\"Unexpected tag '{}' at offset {} when parsing \"\n-                             \"the encapsulated pixel data fragment items.\"\n-                             .format(tag, fp.tell() - 4))\n+            raise ValueError(\n+                f\"Unexpected tag '{tag}' at offset {fp.tell() - 4} when \"\n+                \"parsing the encapsulated pixel data fragment items\"\n+            )\n \n     fp.seek(start)\n     return nr_fragments\n@@ -206,9 +210,10 @@ def generate_pixel_data_fragment(\n             # Item\n             length = fp.read_UL()\n             if length == 0xFFFFFFFF:\n-                raise ValueError(\"Undefined item length at offset {} when \"\n-                                 \"parsing the encapsulated pixel data \"\n-                                 \"fragments.\".format(fp.tell() - 4))\n+                raise ValueError(\n+                    f\"Undefined item length at offset {fp.tell() - 4} when \"\n+                    \"parsing the encapsulated pixel data fragments\"\n+                )\n             yield fp.read(length)\n         elif tag == 0xFFFEE0DD:\n             # Sequence Delimiter\n@@ -216,9 +221,10 @@ def generate_pixel_data_fragment(\n             fp.seek(-4, 1)\n             break\n         else:\n-            raise ValueError(\"Unexpected tag '{0}' at offset {1} when parsing \"\n-                             \"the encapsulated pixel data fragment items.\"\n-                             .format(tag, fp.tell() - 4))\n+            raise ValueError(\n+                f\"Unexpected tag '{tag}' at offset {fp.tell() - 4} when \"\n+                \"parsing the encapsulated pixel data fragment items\"\n+            )\n \n \n def generate_pixel_data_frame(\n@@ -549,8 +555,10 @@ def fragment_frame(\n     frame_length = len(frame)\n     # Add 1 to fix odd length frames not being caught\n     if nr_fragments > (frame_length + 1) / 2.0:\n-        raise ValueError('Too many fragments requested (the minimum fragment '\n-                         'size is 2 bytes)')\n+        raise ValueError(\n+            \"Too many fragments requested (the minimum fragment size is \"\n+            \"2 bytes)\"\n+        )\n \n     length = int(frame_length / nr_fragments)\n \n@@ -596,7 +604,7 @@ def itemize_fragment(fragment: bytes) -> bytes:\n       a 4 byte length.\n     \"\"\"\n     # item tag (fffe,e000)\n-    item = bytes(b'\\xFE\\xFF\\x00\\xE0')\n+    item = b'\\xFE\\xFF\\x00\\xE0'\n     # fragment length '<I' little endian, 4 byte unsigned int\n     item += pack('<I', len(fragment))\n     # fragment data\n@@ -640,7 +648,7 @@ def itemize_frame(\n     :dcm:`Annex A.4 <part05/sect_A.4.html>`\n     \"\"\"\n     for fragment in fragment_frame(frame, nr_fragments):\n-        yield itemise_fragment(fragment)\n+        yield itemize_fragment(fragment)\n \n \n itemise_frame = itemize_frame\n@@ -664,6 +672,16 @@ def encapsulate(\n     For multi-frame data each frame must be encoded separately and then all\n     encoded frames encapsulated together.\n \n+    When many large frames are to be encapsulated, the total length of\n+    encapsulated data may exceed the maximum length available with the\n+    :dcm:`Basic Offset Table<part05/sect_A.4.html>` (2**31 - 1 bytes). Under\n+    these circumstances you can:\n+\n+    * Pass ``has_bot=False`` to :func:`~pydicom.encaps.encapsulate`\n+    * Use :func:`~pydicom.encaps.encapsulate_extended` and add the\n+      :dcm:`Extended Offset Table<part03/sect_C.7.6.3.html>` elements to your\n+      dataset (recommended)\n+\n     Data will be encapsulated with a Basic Offset Table Item at the beginning,\n     then one or more fragment items. Each item will be of even length and the\n     final fragment of each frame may be padded with ``0x00`` if required.\n@@ -688,18 +706,33 @@ def encapsulate(\n     ----------\n     DICOM Standard, Part 5, :dcm:`Section 7.5 <part05/sect_7.5.html>` and\n     :dcm:`Annex A.4 <part05/sect_A.4.html>`\n+\n+    See Also\n+    --------\n+    :func:`~pydicom.encaps.encapsulate_extended`\n     \"\"\"\n-    no_frames = len(frames)\n+    nr_frames = len(frames)\n     output = bytearray()\n \n     # Add the Basic Offset Table Item\n     # Add the tag\n     output.extend(b'\\xFE\\xFF\\x00\\xE0')\n     if has_bot:\n+        # Check that the 2**32 - 1 limit in BOT item lengths won't be exceeded\n+        total = (nr_frames - 1) * 8 + sum([len(f) for f in frames[:-1]])\n+        if total > 2**32 - 1:\n+            raise ValueError(\n+                f\"The total length of the encapsulated frame data ({total} \"\n+                \"bytes) will be greater than the maximum allowed by the Basic \"\n+                f\"Offset Table ({2**32 - 1} bytes), it's recommended that you \"\n+                \"use the Extended Offset Table instead (see the \"\n+                \"'encapsulate_extended' function for more information)\"\n+            )\n+\n         # Add the length\n-        output.extend(pack('<I', 4 * no_frames))\n+        output.extend(pack('<I', 4 * nr_frames))\n         # Reserve 4 x len(frames) bytes for the offsets\n-        output.extend(b'\\xFF\\xFF\\xFF\\xFF' * no_frames)\n+        output.extend(b'\\xFF\\xFF\\xFF\\xFF' * nr_frames)\n     else:\n         # Add the length\n         output.extend(pack('<I', 0))\n@@ -708,7 +741,7 @@ def encapsulate(\n     for ii, frame in enumerate(frames):\n         # `itemised_length` is the total length of each itemised frame\n         itemised_length = 0\n-        for item in itemise_frame(frame, fragments_per_frame):\n+        for item in itemize_frame(frame, fragments_per_frame):\n             itemised_length += len(item)\n             output.extend(item)\n \n@@ -717,7 +750,67 @@ def encapsulate(\n \n     if has_bot:\n         # Go back and write the frame offsets - don't need the last offset\n-        output[8:8 + 4 * no_frames] = pack('<{}I'.format(no_frames),\n-                                           *bot_offsets[:-1])\n+        output[8:8 + 4 * nr_frames] = pack(f\"<{nr_frames}I\", *bot_offsets[:-1])\n \n     return bytes(output)\n+\n+\n+def encapsulate_extended(frames: List[bytes]) -> Tuple[bytes, bytes, bytes]:\n+    \"\"\"Return encapsulated image data and values for the Extended Offset Table\n+    elements.\n+\n+    When using a compressed transfer syntax (such as RLE Lossless or one of\n+    JPEG formats) then any *Pixel Data* must be :dcm:`encapsulated\n+    <part05/sect_A.4.html>`. When many large frames are to be encapsulated, the\n+    total length of encapsulated data may exceed the maximum length available\n+    with the :dcm:`Basic Offset Table<part05/sect_A.4.html>` (2**32 - 1 bytes).\n+    Under these circumstances you can:\n+\n+    * Pass ``has_bot=False`` to :func:`~pydicom.encaps.encapsulate`\n+    * Use :func:`~pydicom.encaps.encapsulate_extended` and add the\n+      :dcm:`Extended Offset Table<part03/sect_C.7.6.3.html>` elements to your\n+      dataset (recommended)\n+\n+    Examples\n+    --------\n+\n+    .. code-block:: python\n+\n+        from pydicom.encaps import encapsulate_extended\n+\n+        # 'frames' is a list of image frames that have been each been encoded\n+        # separately using the compression method corresponding to the Transfer\n+        # Syntax UID\n+        frames: List[bytes] = [...]\n+        out: Tuple[bytes, bytes, bytes] = encapsulate_extended(frames)\n+\n+        ds.PixelData = out[0]\n+        ds.ExtendedOffsetTable = out[1]\n+        ds.ExtendedOffsetTableLengths = out[2]\n+\n+    Parameters\n+    ----------\n+    frames : list of bytes\n+        The compressed frame data to encapsulate, one frame per item.\n+\n+    Returns\n+    -------\n+    bytes, bytes, bytes\n+        The (encapsulated frames, extended offset table, extended offset\n+        table lengths).\n+\n+    See Also\n+    --------\n+    :func:`~pydicom.encaps.encapsulate`\n+    \"\"\"\n+    nr_frames = len(frames)\n+    frame_lengths = [len(frame) for frame in frames]\n+    frame_offsets = [0]\n+    for ii, length in enumerate(frame_lengths[:-1]):\n+        # Extra 8 bytes for the Item tag and length\n+        frame_offsets.append(frame_offsets[ii] + length + 8)\n+\n+    offsets = pack(f\"<{nr_frames}Q\", *frame_offsets)\n+    lengths = pack(f\"<{nr_frames}Q\", *frame_lengths)\n+\n+    return encapsulate(frames, has_bot=False), offsets, lengths\n", "test_patch": "diff --git a/pydicom/tests/test_encaps.py b/pydicom/tests/test_encaps.py\n--- a/pydicom/tests/test_encaps.py\n+++ b/pydicom/tests/test_encaps.py\n@@ -1,6 +1,8 @@\n-# Copyright 2008-2018 pydicom authors. See LICENSE file for details.\n+# Copyright 2008-2020 pydicom authors. See LICENSE file for details.\n \"\"\"Test for encaps.py\"\"\"\n \n+from struct import unpack\n+\n import pytest\n \n from pydicom import dcmread\n@@ -16,7 +18,8 @@\n     read_item,\n     fragment_frame,\n     itemise_frame,\n-    encapsulate\n+    encapsulate,\n+    encapsulate_extended\n )\n from pydicom.filebase import DicomBytesIO\n \n@@ -36,7 +39,7 @@ def test_bad_tag(self):\n         fp.is_little_endian = True\n         with pytest.raises(ValueError,\n                            match=r\"Unexpected tag '\\(fffe, e100\\)' when \"\n-                                 r\"parsing the Basic Table Offset item.\"):\n+                                 r\"parsing the Basic Table Offset item\"):\n             get_frame_offsets(fp)\n \n     def test_bad_length_multiple(self):\n@@ -49,7 +52,7 @@ def test_bad_length_multiple(self):\n         fp.is_little_endian = True\n         with pytest.raises(ValueError,\n                            match=\"The length of the Basic Offset Table item\"\n-                                 \" is not a multiple of 4.\"):\n+                                 \" is not a multiple of 4\"):\n             get_frame_offsets(fp)\n \n     def test_zero_length(self):\n@@ -138,7 +141,7 @@ def test_item_bad_tag(self):\n         fp.is_little_endian = True\n         msg = (\n             r\"Unexpected tag '\\(0010, 0010\\)' at offset 12 when parsing the \"\n-            r\"encapsulated pixel data fragment items.\"\n+            r\"encapsulated pixel data fragment items\"\n         )\n         with pytest.raises(ValueError, match=msg):\n             get_nr_fragments(fp)\n@@ -212,7 +215,7 @@ def test_item_undefined_length(self):\n         with pytest.raises(ValueError,\n                            match=\"Undefined item length at offset 4 when \"\n                                  \"parsing the encapsulated pixel data \"\n-                                 \"fragments.\"):\n+                                 \"fragments\"):\n             next(fragments)\n         pytest.raises(StopIteration, next, fragments)\n \n@@ -249,8 +252,7 @@ def test_item_bad_tag(self):\n         with pytest.raises(ValueError,\n                            match=r\"Unexpected tag '\\(0010, 0010\\)' at offset \"\n                                  r\"12 when parsing the encapsulated pixel \"\n-                                 r\"data \"\n-                                 r\"fragment items.\"):\n+                                 r\"data fragment items\"):\n             next(fragments)\n         pytest.raises(StopIteration, next, fragments)\n \n@@ -1199,3 +1201,63 @@ def test_encapsulate_bot(self):\n             b'\\xfe\\xff\\x00\\xe0'  # Next item tag\n             b'\\xe6\\x0e\\x00\\x00'  # Next item length\n         )\n+\n+    def test_encapsulate_bot_large_raises(self):\n+        \"\"\"Test exception raised if too much pixel data for BOT.\"\"\"\n+\n+        class FakeBytes(bytes):\n+            length = -1\n+\n+            def __len__(self):\n+                return self.length\n+\n+            def __getitem__(self, s):\n+                return b'\\x00' * 5\n+\n+        frame_a = FakeBytes()\n+        frame_a.length = 2**32 - 1 - 8  # 8 for first BOT item tag/length\n+        frame_b = FakeBytes()\n+        frame_b.length = 10\n+        data = encapsulate([frame_a, frame_b], has_bot=True)\n+\n+        frame_a.length = 2**32 - 1 - 7\n+        msg = (\n+            r\"The total length of the encapsulated frame data \\(4294967296 \"\n+            r\"bytes\\) will be greater than the maximum allowed by the Basic \"\n+        )\n+        with pytest.raises(ValueError, match=msg):\n+            encapsulate([frame_a, frame_b], has_bot=True)\n+\n+\n+class TestEncapsulateExtended:\n+    \"\"\"Tests for encaps.encapsulate_extended.\"\"\"\n+    def test_encapsulate(self):\n+        ds = dcmread(JP2K_10FRAME_NOBOT)\n+        frames = decode_data_sequence(ds.PixelData)\n+        assert len(frames) == 10\n+\n+        out = encapsulate_extended(frames)\n+        # Pixel Data encapsulated OK\n+        assert isinstance(out[0], bytes)\n+        test_frames = decode_data_sequence(out[0])\n+        for a, b in zip(test_frames, frames):\n+            assert a == b\n+\n+        # Extended Offset Table is OK\n+        assert isinstance(out[1], bytes)\n+        assert [\n+            0x0000,  # 0\n+            0x0eee,  # 3822\n+            0x1df6,  # 7670\n+            0x2cf8,  # 11512\n+            0x3bfc,  # 15356\n+            0x4ade,  # 19166\n+            0x59a2,  # 22946\n+            0x6834,  # 26676\n+            0x76e2,  # 30434\n+            0x8594  # 34196\n+        ] == list(unpack('<10Q', out[1]))\n+\n+        # Extended Offset Table Lengths are OK\n+        assert isinstance(out[2], bytes)\n+        assert [len(f) for f in frames] == list(unpack('<10Q', out[2]))\n", "problem_statement": "Add support for Extended Offset Table to encaps module\n[CP1818](http://webcache.googleusercontent.com/search?q=cache:xeWXtrAs9G4J:ftp://medical.nema.org/medical/dicom/final/cp1818_ft_whenoffsettabletoosmall.pdf) added the use of an Extended Offset Table for encapsulated pixel data when the Basic Offset Table isn't suitable.\n", "hints_text": "Some notes for myself:\r\n\r\n* The EOT is optional and intended to support users where there are multiple compressed frames and the total length of all the frames exceeds the `2**32 - 1` limit available in the BOT item lengths. \r\n* Only 1 fragment per frame is allowed with an EOT\r\n* The *Extended Table Offset Lengths* is the length (in bytes) of each compressed frame\r\n* It's not usable where each compressed frame is larger than `2**32 - 1` due to the limit in the 4-byte (FFFE,E000) Item Tag length (who has 4 GB compressed images anyway?)\r\n\r\nRelevant links: [google groups issue](https://groups.google.com/forum/?nomobile=true#!searchin/comp.protocols.dicom/extended$20offset|sort:date/comp.protocols.dicom/piMk2TmcyEg/daCk33zDBwAJ), [Image Pixel module](http://dicom.nema.org/medical/dicom/current/output/chtml/part03/sect_C.7.6.3.html), [BOT](http://dicom.nema.org/medical/dicom/current/output/chtml/part05/sect_A.4.html)\r\n\r\nImplementation shouldn't be difficult, something along the lines of:\r\n```python\r\nfrom struct import pack\r\nfrom pydicom.encaps import encapsulate\r\n\r\nframe_data: List[bytes] = [...]\r\nds.PixelData = encapsulate(frame_data, has_bot=False, fragments_per_frame=1)\r\n\r\nframe_lengths = [len(f) for f in frame_data]\r\nframe_offsets = [0]\r\nfor ii, length in enumerate(frame_lengths[:-1]):\r\n    frame_offsets .append(frame_offsets [ii] + length + 8)\r\nds.ExtendedOffsetTable =  pack(\"<{len(frame_offsets )}Q\", *frame_offsets )\r\nds.ExtendedOffsetTableLengths = pack(\"<{len(frame_lengths)}Q\", *frame_lengths)\r\n```\r\n\r\nMaybe just add a helper function to ensure conformance? And add a check to `encapsulate()` for data that's too long.\r\n```python\r\nfrom pydicom.encaps import encapsulate_extended\r\n\r\nout: Tuple[bytes, bytes, bytes] = encapsulate_extended(frame_data)\r\nds.PixelData = out[0]\r\nds.ExtendedOffsetTable = out[1]\r\nds.ExtendedOffsetTableLengths = out[2]\r\n```\r\n\r\nParsing should already be handled correctly by `generate_pixel_data()` since it's 1 frame per fragment.", "created_at": "2020-10-30T01:59:32Z", "version": "2.0", "FAIL_TO_PASS": ["pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_bad_tag", "pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_bad_length_multiple", "pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_zero_length", "pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_multi_frame", "pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_single_frame", "pydicom/tests/test_encaps.py::TestGetFrameOffsets::test_not_little_endian", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_item_undefined_length", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_item_sequence_delimiter", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_item_bad_tag", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_single_fragment_no_delimiter", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_multi_fragments_no_delimiter", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_single_fragment_delimiter", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_multi_fragments_delimiter", "pydicom/tests/test_encaps.py::TestGetNrFragments::test_not_little_endian", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_item_undefined_length", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_item_sequence_delimiter", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_item_bad_tag", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_single_fragment_no_delimiter", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_multi_fragments_no_delimiter", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_single_fragment_delimiter", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_multi_fragments_delimiter", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFragment::test_not_little_endian", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_empty_bot_single_fragment", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_empty_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_bot_single_fragment", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_multi_frame_one_to_one", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_multi_frame_three_to_one", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_multi_frame_varied_ratio", "pydicom/tests/test_encaps.py::TestGeneratePixelDataFrames::test_empty_bot_multi_fragments_per_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_single_fragment", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_no_nr_frames_raises", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_too_few_fragments", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_multi_fragments_per_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_no_marker", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_empty_bot_missing_marker", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_bot_single_fragment", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_multi_frame_one_to_one", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_multi_frame_three_to_one", "pydicom/tests/test_encaps.py::TestGeneratePixelData::test_multi_frame_varied_ratio", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_empty_bot_single_fragment", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_empty_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_bot_single_fragment", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_bot_triple_fragment_single_frame", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_multi_frame_one_to_one", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_multi_frame_three_to_one", "pydicom/tests/test_encaps.py::TestDecodeDataSequence::test_multi_frame_varied_ratio", "pydicom/tests/test_encaps.py::TestDefragmentData::test_defragment", "pydicom/tests/test_encaps.py::TestReadItem::test_item_undefined_length", "pydicom/tests/test_encaps.py::TestReadItem::test_item_sequence_delimiter", "pydicom/tests/test_encaps.py::TestReadItem::test_item_sequence_delimiter_zero_length", "pydicom/tests/test_encaps.py::TestReadItem::test_item_bad_tag", "pydicom/tests/test_encaps.py::TestReadItem::test_single_fragment_no_delimiter", "pydicom/tests/test_encaps.py::TestReadItem::test_multi_fragments_no_delimiter", "pydicom/tests/test_encaps.py::TestReadItem::test_single_fragment_delimiter", "pydicom/tests/test_encaps.py::TestReadItem::test_multi_fragments_delimiter", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_single_fragment_even_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_single_fragment_odd_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_even_fragment_even_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_even_fragment_odd_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_odd_fragments_even_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_odd_fragments_odd_data", "pydicom/tests/test_encaps.py::TestFragmentFrame::test_too_many_fragments_raises", "pydicom/tests/test_encaps.py::TestEncapsulateFrame::test_single_item", "pydicom/tests/test_encaps.py::TestEncapsulateFrame::test_two_items", "pydicom/tests/test_encaps.py::TestEncapsulate::test_encapsulate_single_fragment_per_frame_no_bot", "pydicom/tests/test_encaps.py::TestEncapsulate::test_encapsulate_single_fragment_per_frame_bot", "pydicom/tests/test_encaps.py::TestEncapsulate::test_encapsulate_bot", "pydicom/tests/test_encaps.py::TestEncapsulate::test_encapsulate_bot_large_raises", "pydicom/tests/test_encaps.py::TestEncapsulateExtended::test_encapsulate"], "PASS_TO_PASS": [], "environment_setup_commit": "9d69811e539774f296c2f289839147e741251716"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1682", "base_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91", "patch": "diff --git a/benchmarks/benchmarks/infinite_sheds.py b/benchmarks/benchmarks/infinite_sheds.py\n--- a/benchmarks/benchmarks/infinite_sheds.py\n+++ b/benchmarks/benchmarks/infinite_sheds.py\n@@ -10,7 +10,11 @@\n \n class InfiniteSheds:\n \n-    def setup(self):\n+    # benchmark variant parameters (run both vectorize=True and False)\n+    params = [True, False]\n+    param_names = ['vectorize']\n+\n+    def setup(self, vectorize):\n         self.times = pd.date_range(start='20180601', freq='1min',\n                                    periods=1440)\n         self.location = location.Location(40, -80)\n@@ -38,7 +42,7 @@ def setup(self):\n                 gcr=self.gcr\n             )\n \n-    def time_get_irradiance_poa_fixed(self):\n+    def time_get_irradiance_poa_fixed(self, vectorize):\n         infinite_sheds.get_irradiance_poa(\n             surface_tilt=self.surface_tilt,\n             surface_azimuth=self.surface_azimuth,\n@@ -51,10 +55,11 @@ def time_get_irradiance_poa_fixed(self):\n             dhi=self.clearsky_irradiance['dhi'],\n             dni=self.clearsky_irradiance['dni'],\n             albedo=self.albedo,\n-            npoints=self.npoints\n+            npoints=self.npoints,\n+            vectorize=vectorize,\n         )\n \n-    def time_get_irradiance_poa_tracking(self):\n+    def time_get_irradiance_poa_tracking(self, vectorize):\n         infinite_sheds.get_irradiance_poa(\n             surface_tilt=self.tracking['surface_tilt'],\n             surface_azimuth=self.tracking['surface_azimuth'],\n@@ -67,10 +72,11 @@ def time_get_irradiance_poa_tracking(self):\n             dhi=self.clearsky_irradiance['dhi'],\n             dni=self.clearsky_irradiance['dni'],\n             albedo=self.albedo,\n-            npoints=self.npoints\n+            npoints=self.npoints,\n+            vectorize=vectorize,\n         )\n \n-    def time_get_irradiance_fixed(self):\n+    def time_get_irradiance_fixed(self, vectorize):\n         infinite_sheds.get_irradiance(\n             surface_tilt=self.surface_tilt,\n             surface_azimuth=self.surface_azimuth,\n@@ -83,10 +89,11 @@ def time_get_irradiance_fixed(self):\n             dhi=self.clearsky_irradiance['dhi'],\n             dni=self.clearsky_irradiance['dni'],\n             albedo=self.albedo,\n-            npoints=self.npoints\n+            npoints=self.npoints,\n+            vectorize=vectorize,\n         )\n \n-    def time_get_irradiance_tracking(self):\n+    def time_get_irradiance_tracking(self, vectorize):\n         infinite_sheds.get_irradiance(\n             surface_tilt=self.tracking['surface_tilt'],\n             surface_azimuth=self.tracking['surface_azimuth'],\n@@ -99,5 +106,6 @@ def time_get_irradiance_tracking(self):\n             dhi=self.clearsky_irradiance['dhi'],\n             dni=self.clearsky_irradiance['dni'],\n             albedo=self.albedo,\n-            npoints=self.npoints\n+            npoints=self.npoints,\n+            vectorize=vectorize,\n         )\ndiff --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@@ -10,10 +10,10 @@\n from pvlib.irradiance import beam_component, aoi, haydavies\n \n def _vf_ground_sky_integ(surface_tilt, surface_azimuth, gcr, height,\n-                         pitch, max_rows=10, npoints=100):\n+                         pitch, max_rows=10, npoints=100, vectorize=False):\n     \"\"\"\n-    Integrated and per-point view factors from the ground to the sky at points\n-    between interior rows of the array.\n+    Integrated view factor to the sky from the ground underneath\n+    interior rows of the array.\n \n     Parameters\n     ----------\n@@ -35,20 +35,16 @@ def _vf_ground_sky_integ(surface_tilt, surface_azimuth, gcr, height,\n         Maximum number of rows to consider in front and behind the current row.\n     npoints : int, default 100\n         Number of points used to discretize distance along the ground.\n+    vectorize : bool, default False\n+        If True, vectorize the view factor calculation across ``surface_tilt``.\n+        This increases speed with the cost of increased memory usage.\n \n     Returns\n     -------\n-    fgnd_sky : float\n+    fgnd_sky : numeric\n         Integration of view factor over the length between adjacent, interior\n-        rows. [unitless]\n-    fz : ndarray\n-        Fraction of distance from the previous row to the next row. [unitless]\n-    fz_sky : ndarray\n-        View factors at discrete points between adjacent, interior rows.\n-        [unitless]\n-\n+        rows.  Shape matches that of ``surface_tilt``. [unitless]\n     \"\"\"\n-    # TODO: vectorize over surface_tilt\n     # Abuse utils._vf_ground_sky_2d by supplying surface_tilt in place\n     # of a signed rotation. This is OK because\n     # 1) z span the full distance between 2 rows, and\n@@ -57,12 +53,16 @@ def _vf_ground_sky_integ(surface_tilt, surface_azimuth, gcr, height,\n     # The VFs to the sky will thus be symmetric around z=0.5\n     z = np.linspace(0, 1, npoints)\n     rotation = np.atleast_1d(surface_tilt)\n-    fz_sky = np.zeros((len(rotation), npoints))\n-    for k, r in enumerate(rotation):\n-        vf, _ = utils._vf_ground_sky_2d(z, r, gcr, pitch, height, max_rows)\n-        fz_sky[k, :] = vf\n+    if vectorize:\n+        fz_sky = utils._vf_ground_sky_2d(z, rotation, gcr, pitch, height,\n+                                         max_rows)\n+    else:\n+        fz_sky = np.zeros((npoints, len(rotation)))\n+        for k, r in enumerate(rotation):\n+            vf = utils._vf_ground_sky_2d(z, r, gcr, pitch, height, max_rows)\n+            fz_sky[:, k] = vf[:, 0]  # remove spurious rotation dimension\n     # calculate the integrated view factor for all of the ground between rows\n-    return np.trapz(fz_sky, z, axis=1)\n+    return np.trapz(fz_sky, z, axis=0)\n \n \n def _poa_ground_shadows(poa_ground, f_gnd_beam, df, vf_gnd_sky):\n@@ -401,7 +401,7 @@ def _shaded_fraction(solar_zenith, solar_azimuth, surface_tilt,\n def get_irradiance_poa(surface_tilt, surface_azimuth, solar_zenith,\n                        solar_azimuth, gcr, height, pitch, ghi, dhi, dni,\n                        albedo, model='isotropic', dni_extra=None, iam=1.0,\n-                       npoints=100):\n+                       npoints=100, vectorize=False):\n     r\"\"\"\n     Calculate plane-of-array (POA) irradiance on one side of a row of modules.\n \n@@ -469,7 +469,12 @@ def get_irradiance_poa(surface_tilt, surface_azimuth, solar_zenith,\n         on the surface that is not reflected away. [unitless]\n \n     npoints : int, default 100\n-        Number of points used to discretize distance along the ground.\n+        Number of discretization points for calculating integrated view\n+        factors.\n+\n+    vectorize : bool, default False\n+        If True, vectorize the view factor calculation across ``surface_tilt``.\n+        This increases speed with the cost of increased memory usage.\n \n     Returns\n     -------\n@@ -537,7 +542,8 @@ def get_irradiance_poa(surface_tilt, surface_azimuth, solar_zenith,\n     # method differs from [1], Eq. 7 and Eq. 8; height is defined at row\n     # center rather than at row lower edge as in [1].\n     vf_gnd_sky = _vf_ground_sky_integ(\n-        surface_tilt, surface_azimuth, gcr, height, pitch, max_rows, npoints)\n+        surface_tilt, surface_azimuth, gcr, height, pitch, max_rows, npoints,\n+        vectorize)\n     # fraction of row slant height that is shaded from direct irradiance\n     f_x = _shaded_fraction(solar_zenith, solar_azimuth, surface_tilt,\n                            surface_azimuth, gcr)\n@@ -610,7 +616,7 @@ def get_irradiance(surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n                    gcr, height, pitch, ghi, dhi, dni,\n                    albedo, model='isotropic', dni_extra=None, iam_front=1.0,\n                    iam_back=1.0, bifaciality=0.8, shade_factor=-0.02,\n-                   transmission_factor=0, npoints=100):\n+                   transmission_factor=0, npoints=100, vectorize=False):\n     \"\"\"\n     Get front and rear irradiance using the infinite sheds model.\n \n@@ -701,7 +707,12 @@ def get_irradiance(surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n         etc. A negative value is a reduction in back irradiance. [unitless]\n \n     npoints : int, default 100\n-        Number of points used to discretize distance along the ground.\n+        Number of discretization points for calculating integrated view\n+        factors.\n+\n+    vectorize : bool, default False\n+        If True, vectorize the view factor calculation across ``surface_tilt``.\n+        This increases speed with the cost of increased memory usage.\n \n     Returns\n     -------\n@@ -756,14 +767,14 @@ def get_irradiance(surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n         solar_zenith=solar_zenith, solar_azimuth=solar_azimuth,\n         gcr=gcr, height=height, pitch=pitch, ghi=ghi, dhi=dhi, dni=dni,\n         albedo=albedo, model=model, dni_extra=dni_extra, iam=iam_front,\n-        npoints=npoints)\n+        npoints=npoints, vectorize=vectorize)\n     # back side POA irradiance\n     irrad_back = get_irradiance_poa(\n         surface_tilt=backside_tilt, surface_azimuth=backside_sysaz,\n         solar_zenith=solar_zenith, solar_azimuth=solar_azimuth,\n         gcr=gcr, height=height, pitch=pitch, ghi=ghi, dhi=dhi, dni=dni,\n         albedo=albedo, model=model, dni_extra=dni_extra, iam=iam_back,\n-        npoints=npoints)\n+        npoints=npoints, vectorize=vectorize)\n \n     colmap_front = {\n         'poa_global': 'poa_front',\ndiff --git a/pvlib/bifacial/utils.py b/pvlib/bifacial/utils.py\n--- a/pvlib/bifacial/utils.py\n+++ b/pvlib/bifacial/utils.py\n@@ -5,7 +5,6 @@\n import numpy as np\n from pvlib.tools import sind, cosd, tand\n \n-\n def _solar_projection_tangent(solar_zenith, solar_azimuth, surface_azimuth):\n     \"\"\"\n     Tangent of the angle between the zenith vector and the sun vector\n@@ -104,7 +103,7 @@ def _vf_ground_sky_2d(x, rotation, gcr, pitch, height, max_rows=10):\n         Position on the ground between two rows, as a fraction of the pitch.\n         x = 0 corresponds to the point on the ground directly below the\n         center point of a row. Positive x is towards the right. [unitless]\n-    rotation : float\n+    rotation : numeric\n         Rotation angle of the row's right edge relative to row center.\n         [degree]\n     gcr : float\n@@ -120,30 +119,53 @@ def _vf_ground_sky_2d(x, rotation, gcr, pitch, height, max_rows=10):\n \n     Returns\n     -------\n-    vf : numeric\n-        Fraction of sky dome visible from each point on the ground. [unitless]\n-    wedge_angles : array\n-        Angles defining each wedge of sky that is blocked by a row. Shape is\n-        (2, len(x), 2*max_rows+1). ``wedge_angles[0,:,:]`` is the\n-        starting angle of each wedge, ``wedge_angles[1,:,:]`` is the end angle.\n-        [degree]\n+    vf : array\n+        Fraction of sky dome visible from each point on the ground.\n+        Shape is (len(x), len(rotation)). [unitless]\n     \"\"\"\n-    x = np.atleast_1d(x)  # handle float\n+    # This function creates large float64 arrays of size\n+    # (2*len(x)*len(rotation)*len(max_rows)) or ~100 MB for\n+    # typical time series inputs.  This function makes heavy\n+    # use of numpy's out parameter to avoid allocating new\n+    # memory.  Unfortunately that comes at the cost of some\n+    # readability: because arrays get reused to avoid new allocations,\n+    # variable names don't always match what they hold.\n+\n+    # handle floats:\n+    x = np.atleast_1d(x)[:, np.newaxis, np.newaxis]\n+    rotation = np.atleast_1d(rotation)[np.newaxis, :, np.newaxis]\n     all_k = np.arange(-max_rows, max_rows + 1)\n     width = gcr * pitch / 2.\n+    distance_to_row_centers = (all_k - x) * pitch\n+    dy = width * sind(rotation)\n+    dx = width * cosd(rotation)\n+\n+    phi = np.empty((2, x.shape[0], rotation.shape[1], len(all_k)))\n+\n     # angles from x to right edge of each row\n-    a1 = height + width * sind(rotation)\n-    b1 = (all_k - x[:, np.newaxis]) * pitch + width * cosd(rotation)\n-    phi_1 = np.degrees(np.arctan2(a1, b1))\n+    a1 = height + dy\n+    # temporarily store one leg of the triangle in phi:\n+    np.add(distance_to_row_centers, dx, out=phi[0])\n+    np.arctan2(a1, phi[0], out=phi[0])\n+\n     # angles from x to left edge of each row\n-    a2 = height - width * sind(rotation)\n-    b2 = (all_k - x[:, np.newaxis]) * pitch - width * cosd(rotation)\n-    phi_2 = np.degrees(np.arctan2(a2, b2))\n-    phi = np.stack([phi_1, phi_2])\n-    swap = phi[0, :, :] > phi[1, :, :]\n-    # swap where phi_1 > phi_2 so that phi_1[0,:,:] is the lesser angle\n-    phi = np.where(swap, phi[::-1], phi)\n-    # right edge of next row - left edge of previous row\n-    wedge_vfs = 0.5 * (cosd(phi[1, :, 1:]) - cosd(phi[0, :, :-1]))\n-    vf = np.sum(np.where(wedge_vfs > 0, wedge_vfs, 0.), axis=1)\n-    return vf, phi\n+    a2 = height - dy\n+    np.subtract(distance_to_row_centers, dx, out=phi[1])\n+    np.arctan2(a2, phi[1], out=phi[1])\n+\n+    # swap angles so that phi[0,:,:,:] is the lesser angle\n+    phi.sort(axis=0)\n+\n+    # now re-use phi's memory again, this time storing cos(phi).\n+    next_edge = phi[1, :, :, 1:]\n+    np.cos(next_edge, out=next_edge)\n+    prev_edge = phi[0, :, :, :-1]\n+    np.cos(prev_edge, out=prev_edge)\n+    # right edge of next row - left edge of previous row, again\n+    # reusing memory so that the difference is stored in next_edge.\n+    # Note that the 0.5 view factor coefficient is applied after summing\n+    # as a minor speed optimization.\n+    np.subtract(next_edge, prev_edge, out=next_edge)\n+    np.clip(next_edge, a_min=0., a_max=None, out=next_edge)\n+    vf = np.sum(next_edge, axis=-1) / 2\n+    return vf\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@@ -42,7 +42,8 @@ def test_system():\n     return syst, pts, vfs_ground_sky\n \n \n-def test__vf_ground_sky_integ(test_system):\n+@pytest.mark.parametrize(\"vectorize\", [True, False])\n+def test__vf_ground_sky_integ(test_system, vectorize):\n     ts, pts, vfs_gnd_sky = test_system\n     # pass rotation here since max_rows=1 for the hand-solved case in\n     # the fixture test_system, which means the ground-to-sky view factor\n@@ -50,7 +51,7 @@ def test__vf_ground_sky_integ(test_system):\n     vf_integ = infinite_sheds._vf_ground_sky_integ(\n         ts['rotation'], ts['surface_azimuth'],\n         ts['gcr'], ts['height'], ts['pitch'],\n-        max_rows=1, npoints=3)\n+        max_rows=1, npoints=3, vectorize=vectorize)\n     expected_vf_integ = np.trapz(vfs_gnd_sky, pts)\n     assert np.isclose(vf_integ, expected_vf_integ, rtol=0.1)\n \n@@ -262,7 +263,8 @@ def test__backside_tilt():\n     assert np.allclose(back_az, np.array([0., 330., 90., 180.]))\n \n \n-def test_get_irradiance():\n+@pytest.mark.parametrize(\"vectorize\", [True, False])\n+def test_get_irradiance(vectorize):\n     # singleton inputs\n     solar_zenith = 0.\n     solar_azimuth = 180.\n@@ -282,7 +284,7 @@ def test_get_irradiance():\n         surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n         gcr, height, pitch, ghi, dhi, dni, albedo, iam_front, iam_back,\n         bifaciality=0.8, shade_factor=-0.02, transmission_factor=0,\n-        npoints=npoints)\n+        npoints=npoints, vectorize=vectorize)\n     expected_front_diffuse = np.array([300.])\n     expected_front_direct = np.array([700.])\n     expected_front_global = expected_front_diffuse + expected_front_direct\n@@ -300,11 +302,11 @@ def test_get_irradiance():\n         surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n         gcr, height, pitch, ghi, dhi, dni, albedo, iam_front, iam_back,\n         bifaciality=0.8, shade_factor=-0.02, transmission_factor=0,\n-        npoints=npoints)\n+        npoints=npoints, vectorize=vectorize)\n     result_front = infinite_sheds.get_irradiance_poa(\n         surface_tilt, surface_azimuth, solar_zenith, solar_azimuth,\n         gcr, height, pitch, ghi, dhi, dni,\n-        albedo, iam=iam_front)\n+        albedo, iam=iam_front, vectorize=vectorize)\n     assert isinstance(result, pd.DataFrame)\n     expected_poa_global = pd.Series(\n         [1000., 500., result_front['poa_global'][2] * (1 + 0.8 * 0.98),\ndiff --git a/pvlib/tests/bifacial/test_utils.py b/pvlib/tests/bifacial/test_utils.py\n--- a/pvlib/tests/bifacial/test_utils.py\n+++ b/pvlib/tests/bifacial/test_utils.py\n@@ -35,7 +35,7 @@ def test_system_fixed_tilt():\n     c22 = (-2 - sqr3) / np.sqrt(1.25**2 + (2 + sqr3)**2)  # right edge row 0\n     c23 = (0 - sqr3) / np.sqrt(1.25**2 + (0 - sqr3)**2)  # right edge row 1\n     vf_2 = 0.5 * (c23 - c22 + c21 - c20)  # vf at point 1\n-    vfs_ground_sky = np.array([vf_0, vf_1, vf_2])\n+    vfs_ground_sky = np.array([[vf_0], [vf_1], [vf_2]])\n     return syst, pts, vfs_ground_sky\n \n \n@@ -79,10 +79,10 @@ def test__unshaded_ground_fraction(\n def test__vf_ground_sky_2d(test_system_fixed_tilt):\n     # vector input\n     ts, pts, vfs_gnd_sky = test_system_fixed_tilt\n-    vfs, _ = utils._vf_ground_sky_2d(pts, ts['rotation'], ts['gcr'],\n-                                     ts['pitch'], ts['height'], max_rows=1)\n+    vfs = utils._vf_ground_sky_2d(pts, ts['rotation'], ts['gcr'],\n+                                  ts['pitch'], ts['height'], max_rows=1)\n     assert np.allclose(vfs, vfs_gnd_sky, rtol=0.1)  # middle point vf is off\n     # test with singleton x\n-    vf, _ = utils._vf_ground_sky_2d(pts[0], ts['rotation'], ts['gcr'],\n-                                    ts['pitch'], ts['height'], max_rows=1)\n+    vf = utils._vf_ground_sky_2d(pts[0], ts['rotation'], ts['gcr'],\n+                                 ts['pitch'], ts['height'], max_rows=1)\n     assert np.isclose(vf, vfs_gnd_sky[0])\n", "problem_statement": "Infinite sheds perf improvement: vectorize over surface_tilt\nInfinite sheds is quite a bit slower than the modelchain POA modeling we use for frontside (as expected). I see a TODO comment in the code for _vf_ground_sky_integ (`_TODO: vectorize over surface_tilt_`) that could potentially result in some perf improvement for Infinite sheds calls with tracking systems.\n", "hints_text": "I haven't profiled it but I suspect you are right: [this line](https://github.com/pvlib/pvlib-python/blob/7bb30ad6e9d599d10510c7b37d95d56f14e846b4/pvlib/bifacial/infinite_sheds.py#L61) is the likely bottleneck.\n@wholmgren in #1627 you mentioned looking into optimizations for infinite sheds.  I wonder if there is anything in the works?\r\n\r\nAlso here's a quick and dirty profile, for reference:\r\n\r\n![image](https://user-images.githubusercontent.com/57452607/222270556-923deda6-6b9a-4006-b729-6270f0235d90.png)\r\n\r\n```python\r\nimport pvlib\r\nimport pandas as pd\r\n\r\ntimes = pd.date_range('2019-01-01', '2019-02-01', freq='5T', 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\ntr = pvlib.tracking.singleaxis(sp.zenith, sp.azimuth)\r\n\r\ngcr = 0.5\r\nheight = 1.5\r\npitch = 3.0\r\nalbedo = 0.2\r\n\r\nresult = pvlib.bifacial.infinite_sheds.get_irradiance(\r\n    surface_tilt=tr.surface_tilt, surface_azimuth=tr.surface_azimuth,\r\n    solar_zenith=sp.zenith, solar_azimuth=sp.azimuth,\r\n    gcr=0.5, height=1.5, pitch=3.0,\r\n    ghi=cs.ghi, dhi=cs.dhi, dni=cs.dni, albedo=0.2\r\n)\r\n```\nI identified a few places with repeated calculations and started thinking about larger changes to mostly private functions that would avoid more repeated calculations. So all of that is largely complementary to vectorization.\nThis is a great conversation, thanks all for responding so quickly. Is there a timeline by which we could expect some of these perf changes to go in? Excited to use a faster Infinite Sheds. :)\nI tried out vectorizing over `surface_tilt` in `utils._vf_ground_sky_2d` and got a ~15-20% speed improvement for `infinite_sheds._vf_ground_sky_integ`.  After those changes (and some minor unrelated optimizations), 80% of remaining runtime in `_vf_ground_sky_2d` is in the arctan and cosine calculations, so there might not be much room for additional improvement without changing how we do the math.  I'll open a PR with some more formal benchmark results. \r\n\r\n@aturabi, separately from these code optimizations, you might take a look at reducing the `npoints` parameter.  That parameter offers a trade-off between model resolution and speed, and I don't think we've done a real sensitivity analysis but I bet the default of 100 is overkill a lot of the time and you could get essentially the same simulation results faster with a lower (perhaps much lower) number.  ", "created_at": "2023-03-03T00:30:38Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_ground_sky_integ[True]", "pvlib/tests/bifacial/test_infinite_sheds.py::test__vf_ground_sky_integ[False]", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance[True]", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance[False]", "pvlib/tests/bifacial/test_utils.py::test__vf_ground_sky_2d"], "PASS_TO_PASS": ["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__ground_angle_zero_gcr", "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_limiting_gcr", "pvlib/tests/bifacial/test_infinite_sheds.py::test_get_irradiance_with_haydavies", "pvlib/tests/bifacial/test_utils.py::test__solar_projection_tangent", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.5-0.0-180.0-0.0-180.0-0.5]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[1.0-0.0-180.0-0.0-180.0-0.0]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[1.0-90.0-180.0-0.0-180.0-1.0]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.5-45.0-180.0-45.0-270.0-0.6464466094067263]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.5-45.0-180.0-90.0-180.0-0.0]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.7071067811865476-45-180-0-180-0.5]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.7071067811865476-45-180-45-180-0.0]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.7071067811865476-45-180-45-90-0.5]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.7071067811865476-45-180-45-0-1.0]", "pvlib/tests/bifacial/test_utils.py::test__unshaded_ground_fraction[0.7071067811865476-45-180-45-135-0.1464466094067262]"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1555", "base_commit": "9db89e1d8f5e82dc617f9c8cbf303fe23a0632b9", "patch": "diff --git a/pydicom/dataset.py b/pydicom/dataset.py\n--- a/pydicom/dataset.py\n+++ b/pydicom/dataset.py\n@@ -1488,7 +1488,7 @@ def _convert_pixel_data_without_handler(self) -> None:\n         possible_handlers = [\n             hh for hh in pydicom.config.pixel_data_handlers\n             if hh is not None\n-            and hh.supports_transfer_syntax(ts)  # type: ignore[attr-defined]\n+            and hh.supports_transfer_syntax(ts)\n         ]\n \n         # No handlers support the transfer syntax\n@@ -1504,7 +1504,7 @@ def _convert_pixel_data_without_handler(self) -> None:\n         #   dependencies met\n         available_handlers = [\n             hh for hh in possible_handlers\n-            if hh.is_available()  # type: ignore[attr-defined]\n+            if hh.is_available()\n         ]\n \n         # There are handlers that support the transfer syntax but none of them\n@@ -1518,13 +1518,13 @@ def _convert_pixel_data_without_handler(self) -> None:\n             )\n             pkg_msg = []\n             for hh in possible_handlers:\n-                hh_deps = hh.DEPENDENCIES  # type: ignore[attr-defined]\n+                hh_deps = hh.DEPENDENCIES\n                 # Missing packages\n                 missing = [dd for dd in hh_deps if have_package(dd) is None]\n                 # Package names\n                 names = [hh_deps[name][1] for name in missing]\n                 pkg_msg.append(\n-                    f\"{hh.HANDLER_NAME} \"  # type: ignore[attr-defined]\n+                    f\"{hh.HANDLER_NAME} \"\n                     f\"(req. {', '.join(names)})\"\n                 )\n \n@@ -1819,7 +1819,7 @@ def overlay_array(self, group: int) -> \"numpy.ndarray\":\n \n         available_handlers = [\n             hh for hh in overlay_data_handlers\n-            if hh.is_available()  # type: ignore[attr-defined]\n+            if hh.is_available()\n         ]\n         if not available_handlers:\n             # For each of the handlers we want to find which\n@@ -1830,13 +1830,13 @@ def overlay_array(self, group: int) -> \"numpy.ndarray\":\n             )\n             pkg_msg = []\n             for hh in overlay_data_handlers:\n-                hh_deps = hh.DEPENDENCIES  # type: ignore[attr-defined]\n+                hh_deps = hh.DEPENDENCIES\n                 # Missing packages\n                 missing = [dd for dd in hh_deps if have_package(dd) is None]\n                 # Package names\n                 names = [hh_deps[name][1] for name in missing]\n                 pkg_msg.append(\n-                    f\"{hh.HANDLER_NAME} \"  # type: ignore[attr-defined]\n+                    f\"{hh.HANDLER_NAME} \"\n                     f\"(req. {', '.join(names)})\"\n                 )\n \n@@ -1846,7 +1846,7 @@ def overlay_array(self, group: int) -> \"numpy.ndarray\":\n         for handler in available_handlers:\n             try:\n                 # Use the handler to get an ndarray of the pixel data\n-                func = handler.get_overlay_array  # type: ignore[attr-defined]\n+                func = handler.get_overlay_array\n                 return cast(\"numpy.ndarray\", func(self, group))\n             except Exception as exc:\n                 logger.debug(\ndiff --git a/pydicom/encoders/base.py b/pydicom/encoders/base.py\n--- a/pydicom/encoders/base.py\n+++ b/pydicom/encoders/base.py\n@@ -84,11 +84,11 @@ def add_plugin(self, label: str, import_path: Tuple[str, str]) -> None:\n         module = import_module(import_path[0])\n \n         # `is_available(UID)` is required for plugins\n-        if module.is_available(self.UID):  # type: ignore[attr-defined]\n+        if module.is_available(self.UID):\n             self._available[label] = getattr(module, import_path[1])\n         else:\n             # `ENCODER_DEPENDENCIES[UID]` is required for plugins\n-            deps = module.ENCODER_DEPENDENCIES  # type: ignore[attr-defined]\n+            deps = module.ENCODER_DEPENDENCIES\n             self._unavailable[label] = deps[self.UID]\n \n     @staticmethod\ndiff --git a/pydicom/filebase.py b/pydicom/filebase.py\n--- a/pydicom/filebase.py\n+++ b/pydicom/filebase.py\n@@ -182,15 +182,15 @@ def __init__(\n         self.close = file_like_obj.close\n         self.name: str = getattr(file_like_obj, 'name', '<no filename>')\n \n-    def no_write(self, bytes_read: bytes) -> None:\n+    def no_write(self, bytes_read: bytes) -> int:\n         \"\"\"Used for file-like objects where no write is available\"\"\"\n         raise IOError(\"This DicomFileLike object has no write() method\")\n \n-    def no_read(self, bytes_read: Optional[int] = None) -> None:\n+    def no_read(self, size: int = -1) -> bytes:\n         \"\"\"Used for file-like objects where no read is available\"\"\"\n         raise IOError(\"This DicomFileLike object has no read() method\")\n \n-    def no_seek(self, offset: int, from_what: int = 0) -> None:\n+    def no_seek(self, offset: int, whence: int = 0) -> int:\n         \"\"\"Used for file-like objects where no seek is available\"\"\"\n         raise IOError(\"This DicomFileLike object has no seek() method\")\n \ndiff --git a/pydicom/fileset.py b/pydicom/fileset.py\n--- a/pydicom/fileset.py\n+++ b/pydicom/fileset.py\n@@ -2865,10 +2865,10 @@ def my_recorder(ds: Dataset) -> Dataset:\n \n def _single_level_record_type(ds: Dataset) -> str:\n     \"\"\"Return a single-level *Directory Record Type* for `ds`.\"\"\"\n-    sop_class = getattr(ds, \"SOPClassUID\", None)\n+    sop_class = cast(Optional[UID], getattr(ds, \"SOPClassUID\", None))\n \n     try:\n-        return _SINGLE_LEVEL_SOP_CLASSES[sop_class]\n+        return _SINGLE_LEVEL_SOP_CLASSES[sop_class]  # type: ignore[index]\n     except KeyError:\n         return \"PATIENT\"\n \n@@ -2888,9 +2888,9 @@ def _four_level_record_type(ds: Dataset) -> str:\n     if \"RTPlanLabel\" in ds:\n         return \"RT PLAN\"\n \n-    sop_class = getattr(ds, \"SOPClassUID\", None)\n+    sop_class = cast(Optional[UID], getattr(ds, \"SOPClassUID\", None))\n \n     try:\n-        return _FOUR_LEVEL_SOP_CLASSES[sop_class]\n+        return _FOUR_LEVEL_SOP_CLASSES[sop_class]  # type: ignore[index]\n     except KeyError:\n         return \"IMAGE\"\ndiff --git a/pydicom/filewriter.py b/pydicom/filewriter.py\n--- a/pydicom/filewriter.py\n+++ b/pydicom/filewriter.py\n@@ -1000,6 +1000,7 @@ def dcmwrite(\n         Write a DICOM file from a dataset that was read in with ``dcmread()``.\n         ``save_as()`` wraps ``dcmwrite()``.\n     \"\"\"\n+    tsyntax: Optional[UID]\n \n     # Ensure is_little_endian and is_implicit_VR are set\n     if None in (dataset.is_little_endian, dataset.is_implicit_VR):\n@@ -1090,7 +1091,7 @@ def dcmwrite(\n             fp.write(preamble)\n             fp.write(b'DICM')\n \n-        tsyntax: Optional[UID] = None  # type: ignore[no-redef]\n+        tsyntax = None\n         if dataset.file_meta:  # May be an empty Dataset\n             # If we want to `write_like_original`, don't enforce_standard\n             write_file_meta_info(\ndiff --git a/pydicom/pixel_data_handlers/util.py b/pydicom/pixel_data_handlers/util.py\n--- a/pydicom/pixel_data_handlers/util.py\n+++ b/pydicom/pixel_data_handlers/util.py\n@@ -244,7 +244,7 @@ def apply_modality_lut(arr: \"np.ndarray\", ds: \"Dataset\") -> \"np.ndarray\":\n     * DICOM Standard, Part 4, :dcm:`Annex N.2.1.1\n       <part04/sect_N.2.html#sect_N.2.1.1>`\n     \"\"\"\n-    if 'ModalityLUTSequence' in ds:\n+    if ds.get(\"ModalityLUTSequence\"):\n         item = cast(List[\"Dataset\"], ds.ModalityLUTSequence)[0]\n         nr_entries = cast(List[int], item.LUTDescriptor)[0] or 2**16\n         first_map = cast(List[int], item.LUTDescriptor)[1]\n@@ -342,7 +342,7 @@ def apply_voi_lut(\n       <part04/sect_N.2.html#sect_N.2.1.1>`\n     \"\"\"\n     valid_voi = False\n-    if 'VOILUTSequence' in ds:\n+    if ds.get('VOILUTSequence'):\n         ds.VOILUTSequence = cast(List[\"Dataset\"], ds.VOILUTSequence)\n         valid_voi = None not in [\n             ds.VOILUTSequence[0].get('LUTDescriptor', None),\n@@ -408,7 +408,7 @@ def apply_voi(\n     * DICOM Standard, Part 4, :dcm:`Annex N.2.1.1\n       <part04/sect_N.2.html#sect_N.2.1.1>`\n     \"\"\"\n-    if \"VOILUTSequence\" not in ds:\n+    if not ds.get('VOILUTSequence'):\n         return arr\n \n     if not np.issubdtype(arr.dtype, np.integer):\n@@ -529,7 +529,7 @@ def apply_windowing(\n     ds.BitsStored = cast(int, ds.BitsStored)\n     y_min: float\n     y_max: float\n-    if 'ModalityLUTSequence' in ds:\n+    if ds.get('ModalityLUTSequence'):\n         # Unsigned - see PS3.3 C.11.1.1.1\n         y_min = 0\n         item = cast(List[\"Dataset\"], ds.ModalityLUTSequence)[0]\ndiff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -55,7 +55,10 @@ def __getstate__(self) -> Dict[str, Any]:\n     def __setstate__(self, state: Dict[str, Any]) -> None:\n         self.__dict__.update(state)\n \n-    def __reduce_ex__(self, protocol: int) -> Tuple[Any, ...]:\n+    def __reduce_ex__(  # type: ignore[override]\n+        self, protocol: int\n+    ) -> Tuple[Any, ...]:\n+        # Python 3.8 - protocol: SupportsIndex (added in 3.8)\n         # datetime.time, and datetime.datetime return Tuple[Any, ...]\n         # datetime.date doesn't define __reduce_ex__\n         reduce_ex = cast(Tuple[Any, ...], super().__reduce_ex__(protocol))\n@@ -324,17 +327,17 @@ def __new__(  # type: ignore[misc]\n             if match.group('ms'):\n                 microsecond = int(match.group('ms').rstrip().ljust(6, '0'))\n \n-            return super().__new__(  # type: ignore[call-arg, no-any-return]\n+            return super().__new__(\n                 cls, hour, minute, second, microsecond\n             )\n \n         if isinstance(val, datetime.time):\n-            return super().__new__(  # type: ignore[call-arg, no-any-return]\n+            return super().__new__(\n                 cls, val.hour, val.minute, val.second, val.microsecond\n             )\n \n         try:\n-            return super().__new__(  # type: ignore[call-arg, no-any-return]\n+            return super().__new__(\n                 cls, *args, **kwargs\n             )\n         except Exception as exc:\n@@ -925,7 +928,7 @@ def __new__(  # type: ignore[misc]\n         if len(args) and args[0] is None:\n             return None\n \n-        return cast(\"PersonName\", super().__new__(cls))\n+        return super().__new__(cls)\n \n     def __init__(\n         self,\n", "test_patch": "diff --git a/pydicom/tests/test_handler_util.py b/pydicom/tests/test_handler_util.py\n--- a/pydicom/tests/test_handler_util.py\n+++ b/pydicom/tests/test_handler_util.py\n@@ -890,6 +890,10 @@ def test_unchanged(self):\n         out = apply_modality_lut(arr, ds)\n         assert arr is out\n \n+        ds.ModalityLUTSequence = []\n+        out = apply_modality_lut(arr, ds)\n+        assert arr is out\n+\n     def test_lutdata_ow(self):\n         \"\"\"Test LUT Data with VR OW.\"\"\"\n         ds = dcmread(MOD_16_SEQ)\n@@ -1839,6 +1843,10 @@ def test_unchanged(self):\n         out = apply_windowing(arr, ds)\n         assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n \n+        ds.ModalityLUTSequence = []\n+        out = apply_windowing(arr, ds)\n+        assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n+\n     def test_rescale_empty(self):\n         \"\"\"Test RescaleSlope and RescaleIntercept being empty.\"\"\"\n         ds = dcmread(WIN_12_1F)\n@@ -2051,6 +2059,11 @@ def test_unchanged(self):\n         out = apply_voi(arr, ds)\n         assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n \n+        ds.VOILUTSequence = []\n+        out = apply_voi(arr, ds)\n+        assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n+\n+\n     def test_voi_lutdata_ow(self):\n         \"\"\"Test LUT Data with VR OW.\"\"\"\n         ds = Dataset()\n@@ -2072,6 +2085,7 @@ def test_voi_lutdata_ow(self):\n \n @pytest.mark.skipif(not HAVE_NP, reason=\"Numpy is not available\")\n class TestNumpy_ApplyVOILUT:\n+    \"\"\"Tests for util.apply_voi_lut()\"\"\"\n     def test_unchanged(self):\n         \"\"\"Test input array is unchanged if no VOI LUT\"\"\"\n         ds = Dataset()\n@@ -2082,6 +2096,10 @@ def test_unchanged(self):\n         out = apply_voi_lut(arr, ds)\n         assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n \n+        ds.VOILUTSequence = []\n+        out = apply_voi_lut(arr, ds)\n+        assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n+\n     def test_only_windowing(self):\n         \"\"\"Test only windowing operation elements present.\"\"\"\n         ds = Dataset()\n", "problem_statement": "Converting Dicom image to Png\n**Describe the issue**\r\nhi, i am trying to convert Dicom image to png but in case of some particular file i am getting this \"list out of range error\".\r\n\r\n**Expected behavior**\r\ndicom image converted to png pne\r\n\r\n**Steps To Reproduce**\r\nHow to reproduce the issue. Please include:\r\n1. A minimum working code sample\r\n```\r\nfrom pydicom import dcmread\r\ndef read_xray(path, voi_lut = True, fix_monochrome = True):\r\n    dicom = dcmread(path, force=True)\r\n    \r\n    # VOI LUT (if available by DICOM device) is used to transform raw DICOM data to \"human-friendly\" view\r\n    if voi_lut:\r\n        data = apply_voi_lut(dicom.pixel_array, dicom)\r\n    else:\r\n        data = dicom.pixel_array\r\n               \r\n    # depending on this value, X-ray may look inverted - fix that:\r\n    if fix_monochrome and dicom.PhotometricInterpretation == \"MONOCHROME1\":\r\n        data = np.amax(data) - data\r\n        \r\n    data = data - np.min(data)\r\n    data = data / np.max(data)\r\n    data = (data * 255).astype(np.uint8)\r\n        \r\n    return data\r\n\r\nimg = read_xray('/content/a.5545da1153f57ff8425be6f4bc712c090e7e22efff194da525210c84aba2a947.dcm')\r\nplt.figure(figsize = (12,12))\r\nplt.imshow(img)\r\n```\r\n2. The traceback (if one occurred)\r\n```\r\nIndexError                                Traceback (most recent call last)\r\n<ipython-input-13-6e53d7d16b90> in <module>()\r\n     19     return data\r\n     20 \r\n---> 21 img = read_xray('/content/a.5545da1153f57ff8425be6f4bc712c090e7e22efff194da525210c84aba2a947.dcm')\r\n     22 plt.figure(figsize = (12,12))\r\n     23 plt.imshow(img)\r\n\r\n2 frames\r\n/usr/local/lib/python3.7/dist-packages/pydicom/multival.py in __getitem__(self, index)\r\n     93         self, index: Union[slice, int]\r\n     94     ) -> Union[MutableSequence[_ItemType], _ItemType]:\r\n---> 95         return self._list[index]\r\n     96 \r\n     97     def insert(self, position: int, val: _T) -> None:\r\n\r\nIndexError: list index out of range\r\n```\r\n\r\n3. Which of the following packages are available and their versions:\r\n  * Numpy : latest as of 29th dec\r\n  * Pillow : latest as of 29th dec\r\n  * JPEG-LS : latest as of 29th dec\r\n  * GDCM : latest as of 29th dec\r\n4. The anonymized DICOM dataset (if possible).\r\nimage link : https://drive.google.com/file/d/1j13XTTPCLX-8e7FE--1n5Staxz7GGNWm/view?usp=sharing\r\n\r\n**Your environment**\r\nIf you're using **pydicom 2 or later**, please use the `pydicom.env_info`\r\nmodule to gather information about your environment and paste it in the issue:\r\n\r\n```bash\r\n$ python -m pydicom.env_info\r\n```\r\n\r\nFor **pydicom 1.x**, please run the following code snippet and paste the\r\noutput.\r\n\r\n```python\r\nimport platform, sys, pydicom\r\nprint(platform.platform(),\r\n      \"\\nPython\", sys.version,\r\n      \"\\npydicom\", pydicom.__version__)\r\n```\r\n\n", "hints_text": "```\r\nTraceback (most recent call last):\r\n  File \"pyd1554.py\", line 29, in <module>\r\n    img = read_xray('datasets/pyd1554.dcm')\r\n  File \"...pyd1554.py\", line 14, in read_xray\r\n    data = apply_voi_lut(dicom.pixel_array, dicom)\r\n  File \".../pydicom/pixel_data_handlers/util.py\", line 348, in apply_voi_lut\r\n    ds.VOILUTSequence[0].get('LUTDescriptor', None),\r\n  File \".../pydicom/multival.py\", line 95, in __getitem__\r\n    return self._list[index]\r\nIndexError: list index out of range\r\n```\r\nThe *VOI LUT Sequence* is empty, which is probably non-conformant but I can't actually tell what the *SOP Class UID* is (your anonymiser is weird, but probably mammo).\r\n\r\nHere's the whole (also weird) *VOI LUT Sequence* (in hex):\r\n```\r\n       Tag | SQ  |     | Length    | Seq end delimiter     |\r\n28 00 10 30 53 51 00 00 FF FF FF FF FE FF DD E0 00 00 00 00\r\n```\r\n\r\nI might add a check for an empty sequence in `apply_voi_lut` (and the other visualisation functions).\r\n", "created_at": "2021-12-29T08:49:15Z", "version": "2.2", "FAIL_TO_PASS": ["pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_unchanged"], "PASS_TO_PASS": ["pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unknown_pixel_representation_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unknown_bits_allocated_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unsupported_dtypes", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[1-0-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[1-1-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[8-0-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[8-1-False-int8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[16-0-False-uint16]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[16-1-False-int16]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-0-False-uint32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-1-False-int32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-0-True-float32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[64-0-True-float64]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_byte_swapping", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_1frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_1frame_3sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_2frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_2frame_3sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_3sample_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_3sample_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_3sample_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_3sample_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_compressed_syntaxes_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_compressed_syntaxes_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_uncompressed_syntaxes", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_nr_frames_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_samples_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_planar_conf_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_unknown_current_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_unknown_desired_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[RGB-RGB]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL-YBR_FULL]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL-YBR_FULL_422]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL_422-YBR_FULL_422]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL_422-YBR_FULL]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_rgb_ybr_rgb_single_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_rgb_ybr_rgb_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_frame_by_frame", "pydicom/tests/test_handler_util.py::TestNumpy_DtypeCorrectedForEndianness::test_byte_swapping", "pydicom/tests/test_handler_util.py::TestNumpy_DtypeCorrectedForEndianness::test_no_endian_raises", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_slope_intercept", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lut_sequence", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lut_sequence_zero_entries", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lutdata_ow", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_neither_ds_nor_palette_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_palette_unknown_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_palette_unavailable_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_supplemental_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_invalid_bit_depth_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_invalid_lut_bit_depth_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_unequal_lut_length_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_no_palette_color", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint08_16", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint08_16_2frame", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint16_16_segmented_litle", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint16_16_segmented_big", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_16_allocated_8_entries", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_alpha", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_well_known_palette", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_first_map_positive", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_first_map_negative", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_discrete", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_linear", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_indirect_08", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_indirect_16", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_spring", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_summer", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_fall", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_winter", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_first_linear_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_first_indirect_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_unknown_opcode_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_single_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_multi_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint32", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int32", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_rescale", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_modality_lut", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_photometric_interp", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_parameters", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_index", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_rescale_empty", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_single_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_multi_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_zero_entries", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_int8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_int16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_bad_depth", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint16_array_float", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_lutdata_ow", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_only_windowing", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_only_voi", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_voi_windowing", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_voi_windowing_empty", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_precision", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_not_j2k", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_no_siz", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_short_bytestream", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_none", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_missing", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_existing"], "environment_setup_commit": "0fa18d2a2179c92efc22200ed6b3689e66cecf92"}, {"repo": "pyvista/pyvista", "instance_id": "pyvista__pyvista-4329", "base_commit": "b9ee9b9c0625d737dfd4a60e9ed10609e7e0b7fd", "patch": "diff --git a/pyvista/core/pointset.py b/pyvista/core/pointset.py\n--- a/pyvista/core/pointset.py\n+++ b/pyvista/core/pointset.py\n@@ -13,7 +13,7 @@\n import pyvista\n from pyvista import _vtk\n from pyvista.utilities import abstract_class\n-from pyvista.utilities.cells import CellArray, create_mixed_cells, get_mixed_cells\n+from pyvista.utilities.cells import CellArray, create_mixed_cells, get_mixed_cells, numpy_to_idarr\n \n from .._typing import BoundsLike\n from ..utilities.fileio import get_ext\n@@ -751,12 +751,32 @@ def lines(self, lines):\n \n     @property\n     def faces(self) -> np.ndarray:\n-        \"\"\"Return a pointer to the faces as a numpy array.\n+        \"\"\"Return the connectivity array of the faces of this PolyData.\n+\n+        The faces array is organized as::\n+\n+           [n0, p0_0, p0_1, ..., p0_n, n1, p1_0, p1_1, ..., p1_n, ...]\n+\n+        where ``n0`` is the number of points in face 0, and ``pX_Y`` is the\n+        Y'th point in face X.\n+\n+        For example, a triangle and a quadrilateral might be represented as::\n+\n+           [3, 0, 1, 2, 4, 0, 1, 3, 4]\n+\n+        Where the two individual faces would be ``[3, 0, 1, 2]`` and ``[4, 0, 1, 3, 4]``.\n \n         Returns\n         -------\n         numpy.ndarray\n-            Array of face indices.\n+            Array of face connectivity.\n+\n+        Notes\n+        -----\n+        The array returned cannot be modified in place and will raise a\n+        ``ValueError`` if attempted.\n+\n+        You can, however, set the faces directly. See the example.\n \n         Examples\n         --------\n@@ -773,8 +793,21 @@ def faces(self) -> np.ndarray:\n                [4, 1, 2, 5, 4],\n                [4, 3, 4, 7, 6],\n                [4, 4, 5, 8, 7]])\n+\n+        Set the faces directly. The following example creates a simple plane\n+        with a single square faces and modifies it to have two triangles\n+        instead.\n+\n+        >>> mesh = pv.Plane(i_resolution=1, j_resolution=1)\n+        >>> mesh.faces = [3, 0, 1, 2, 3, 3, 2, 1]\n+        >>> mesh.faces\n+        array([3, 0, 1, 2, 3, 3, 2, 1])\n+\n         \"\"\"\n-        return _vtk.vtk_to_numpy(self.GetPolys().GetData())\n+        array = _vtk.vtk_to_numpy(self.GetPolys().GetData())\n+        # Flag this array as read only to ensure users do not attempt to write to it.\n+        array.flags['WRITEABLE'] = False\n+        return array\n \n     @faces.setter\n     def faces(self, faces):\n@@ -1576,11 +1609,33 @@ def _check_for_consistency(self):\n \n     @property\n     def cells(self) -> np.ndarray:\n-        \"\"\"Return a pointer to the cells as a numpy object.\n+        \"\"\"Return the cell data as a numpy object.\n+\n+        This is the old style VTK data layout::\n+\n+           [n0, p0_0, p0_1, ..., p0_n, n1, p1_0, p1_1, ..., p1_n, ...]\n+\n+        where ``n0`` is the number of points in cell 0, and ``pX_Y`` is the\n+        Y'th point in cell X.\n+\n+        For example, a triangle and a line might be represented as::\n+\n+           [3, 0, 1, 2, 2, 0, 1]\n+\n+        Where the two individual cells would be ``[3, 0, 1, 2]`` and ``[2, 0, 1]``.\n \n         See Also\n         --------\n         pyvista.DataSet.get_cell\n+        pyvista.UnstructuredGrid.cell_connectivity\n+        pyvista.UnstructuredGrid.offset\n+\n+        Notes\n+        -----\n+        The array returned cannot be modified in place and will raise a\n+        ``ValueError`` if attempted.\n+\n+        You can, however, set the cells directly. See the example.\n \n         Examples\n         --------\n@@ -1590,13 +1645,25 @@ def cells(self) -> np.ndarray:\n \n         >>> import pyvista\n         >>> from pyvista import examples\n-        >>> hex_beam = pyvista.read(examples.hexbeamfile)\n-        >>> hex_beam.cells[:18]  # doctest:+SKIP\n-        array([ 8,  0,  2,  8,  7, 27, 36, 90, 81,  8,  2,  1,  4,\n-                8, 36, 18, 54, 90])\n+        >>> grid = examples.load_hexbeam()\n+        >>> grid.cells[:18]\n+        array([ 8,  0,  2,  8,  7, 27, 36, 90, 81,  8,  2,  1,  4,  8, 36, 18, 54,\n+               90])\n+\n+        While you cannot change the array inplace, you can overwrite it. For example:\n+\n+        >>> grid.cells = [8, 0, 1, 2, 3, 4, 5, 6, 7]\n \n         \"\"\"\n-        return _vtk.vtk_to_numpy(self.GetCells().GetData())\n+        # Flag this array as read only to ensure users do not attempt to write to it.\n+        array = _vtk.vtk_to_numpy(self.GetCells().GetData())\n+        array.flags['WRITEABLE'] = False\n+        return array\n+\n+    @cells.setter\n+    def cells(self, cells):\n+        vtk_idarr = numpy_to_idarr(cells, deep=False, return_ind=False)\n+        self.GetCells().ImportLegacyFormat(vtk_idarr)\n \n     @property\n     def cells_dict(self) -> dict:\ndiff --git a/pyvista/demos/logo.py b/pyvista/demos/logo.py\n--- a/pyvista/demos/logo.py\n+++ b/pyvista/demos/logo.py\n@@ -148,7 +148,7 @@ def plot_logo(\n     v_grid_atom = atomize(v_grid)\n     v_grid_atom['scalars'] = v_grid_atom.points[:, 0]\n     v_grid_atom_surf = v_grid_atom.extract_surface()\n-    faces = v_grid_atom_surf.faces.reshape(-1, 5)\n+    faces = v_grid_atom_surf.faces.reshape(-1, 5).copy()\n     faces[:, 1:] = faces[:, 1:][:, ::-1]\n     v_grid_atom_surf.faces = faces\n     plotter.add_mesh(\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@@ -146,6 +146,14 @@ def test_init_from_arrays(specify_offset):\n     assert np.allclose(cells, grid.cells)\n     assert np.allclose(grid.cell_connectivity, np.arange(16))\n \n+    # grid.cells is not mutable\n+    assert not grid.cells.flags['WRITEABLE']\n+\n+    # but attribute can be set\n+    new_cells = [8, 0, 1, 2, 3, 4, 5, 6, 7]\n+    grid.cells = [8, 0, 1, 2, 3, 4, 5, 6, 7]\n+    assert np.allclose(grid.cells, new_cells)\n+\n \n @pytest.mark.parametrize('multiple_cell_types', [False, True])\n @pytest.mark.parametrize('flat_cells', [False, True])\ndiff --git a/tests/test_polydata.py b/tests/test_polydata.py\n--- a/tests/test_polydata.py\n+++ b/tests/test_polydata.py\n@@ -82,6 +82,15 @@ def test_init_from_arrays():\n     with pytest.warns(Warning):\n         mesh = pyvista.PolyData(vertices.astype(np.int32), faces)\n \n+    # array must be immutable\n+    with pytest.raises(ValueError):\n+        mesh.faces[0] += 1\n+\n+    # attribute is mutable\n+    faces = [4, 0, 1, 2, 3]\n+    mesh.faces = faces\n+    assert np.allclose(faces, mesh.faces)\n+\n \n def test_init_from_arrays_with_vert():\n     vertices = np.array([[0, 0, 0], [1, 0, 0], [1, 1, 0], [0, 1, 0], [0.5, 0.5, -1], [0, 1.5, 1.5]])\n", "problem_statement": "PolyData faces array is not updatable in-place and has unexpected behavior\n### Describe the bug, what's wrong, and what you expected.\n\nWhen accessing `PolyData.faces` (and likely other cell data), we cannot update the array in place. Further, there is some unexpected behavior where accessing `PolyData.faces` will override existing, modified views of the array.\n\n### Steps to reproduce the bug.\n\n```python \r\n>>> import pyvista as pv\r\n>>> mesh = pv.Sphere()\r\n>>> f = mesh.faces\r\n>>> f\r\narray([  3,   2,  30, ..., 840,  29,  28])\r\n>>> a = f[1:4]\r\n>>> a\r\narray([ 2, 30,  0])\r\n>>> b = f[5:8]\r\n>>> b\r\narray([30, 58,  0])\r\n>>> f[1:4] = b\r\n>>> f[5:8] = a\r\n>>> f\r\narray([  3,  30,  58, ..., 840,  29,  28])\r\n>>> assert all(f[1:4] == b) and all(f[5:8] == a)\r\n>>> mesh.faces  # access overwrites `f` in place which is unexpected and causes the check above to now fail\r\n>>> assert all(f[1:4] == b) and all(f[5:8] == a)\r\n---------------------------------------------------------------------------\r\nAssertionError                            Traceback (most recent call last)\r\n<ipython-input-82-08205e08097f> in <cell line: 13>()\r\n     11 assert all(f[1:4] == b) and all(f[5:8] == a)\r\n     12 mesh.faces  # access overwrites `f` in place\r\n---> 13 assert all(f[1:4] == b) and all(f[5:8] == a)\r\n\r\nAssertionError: \r\n ```\n\n### System Information\n\n```shell\n--------------------------------------------------------------------------------\r\n  Date: Thu May 26 11:45:54 2022 MDT\r\n\r\n                OS : Darwin\r\n            CPU(s) : 16\r\n           Machine : x86_64\r\n      Architecture : 64bit\r\n               RAM : 64.0 GiB\r\n       Environment : Jupyter\r\n       File system : apfs\r\n        GPU Vendor : ATI Technologies Inc.\r\n      GPU Renderer : AMD Radeon Pro 5500M OpenGL Engine\r\n       GPU Version : 4.1 ATI-4.8.13\r\n\r\n  Python 3.8.8 | packaged by conda-forge | (default, Feb 20 2021, 16:12:38)\r\n  [Clang 11.0.1 ]\r\n\r\n           pyvista : 0.35.dev0\r\n               vtk : 9.1.0\r\n             numpy : 1.22.1\r\n           imageio : 2.9.0\r\n           appdirs : 1.4.4\r\n            scooby : 0.5.12\r\n        matplotlib : 3.5.2\r\n           IPython : 7.32.0\r\n          colorcet : 3.0.0\r\n           cmocean : 2.0\r\n        ipyvtklink : 0.2.2\r\n             scipy : 1.8.0\r\n        itkwidgets : 0.32.1\r\n              tqdm : 4.60.0\r\n            meshio : 5.3.4\r\n--------------------------------------------------------------------------------\n```\n\n\n### Screenshots\n\n_No response_\n\n### Code of Conduct\n\n- [X] I agree to follow this project's Code of Conduct\n", "hints_text": "Considering we are purely using VTK for this interface through the `vtk_to_numpy` function, I'm inclined to think this may be an upstream bug where the array doesn't interface completely\r\n\r\nhttps://github.com/pyvista/pyvista/blob/eaa71a640264765c8a9b40d485b6ccdb970c87b0/pyvista/core/pointset.py#L762 \r\n\r\nor perhaps there is a better way to access the Polys from the VTK object?\nA pure VTK demonstration of the issue:\r\n\r\n```py\r\nimport vtk\r\nfrom vtkmodules.util.numpy_support import vtk_to_numpy\r\n\r\n# https://github.com/pyvista/pyvista/issues/2705\r\nsphere = vtk.vtkSphereSource()\r\nsphere.Update()\r\nmesh = sphere.GetOutput()\r\n\r\n\r\nf = vtk_to_numpy(mesh.GetPolys().GetData())\r\n\r\n>>> f\r\narray([  3,   2,  30, ..., 840,  29,  28])\r\n>>> a = f[1:4]\r\n>>> a\r\narray([ 2, 30,  0])\r\n>>> b = f[5:8]\r\n>>> b\r\narray([30, 58,  0])\r\n>>> f[1:4] = b\r\n>>> f[5:8] = a\r\n>>> f\r\narray([  3,  30,  58, ..., 840,  29,  28])\r\n\r\nassert all(f[1:4] == b) and all(f[5:8] == a), 'if this fails, we have more serious problems'\r\n\r\n# access overwrites `f` in place which is unexpected and causes the check above to now fail\r\nvtk_to_numpy(mesh.GetPolys().GetData())\r\nassert all(f[1:4] == b) and all(f[5:8] == a), 'well, this is not expected'\r\n```\nWell, this was a joy to track down. The short answer: Use `vtkCellArray.GetConnectivityArray()` instead of `vtkCellArray.GetData()`.\r\n\r\nThe long explanation: A while ago, `vtkCellArray` was changed so that it can store either 32 bit or 64 bit connectivity data. This enabled saving memory when VTK is compiled with 64 bit ids (`vtkIdType`) AND if the mesh has less than 2 billion connectivity entries. This meant that `vtkCellArray` can store either a 32 bit or 64 bit array for connectivity. However, `GetData()` returns a `vtkIdTypeArray` which is usually compiled to be 64 bit so `vtkCellArray` could no longer return its internal array in this method. So, the placeholder implementation now makes a copy from its internal array to return. So changing this array does not actually affect the connectivity. The issue shown here is a side effect of this implementation. `vtkCellArray.GetConnectivityArray()` actually returns the internal array and can be safely changed to affect the connectivity. pyvista and VTK Python classes have to be changed to use `vtkCellArray.GetConnectivityArray()` and `vtkCellArray.GetOffsetsArray()`. Also don't use `vtkCellArray.SetCells()`. Use `vtkCellArray.SetData()` instead.\nUnfortunately for PyVista, this is not an easy fix as `GetConnectivityArray()` is not a direct stand-in for `GetData()`. \r\n\r\n`GetData()` array has connectivity offsets (`GetOffsetsArray()`) embedded in the array, whereas `GetConnectivityArray()` does not. This is quite nice for `vtkPolyData.GetPolys()` if and only if all of the polys are the same size (e.g., 3 for triangulated mesh), but if there are mixed polys, this is pretty tough to work with when trying to update the array in-place:\r\n\r\n```py\r\nimport numpy as np\r\nimport pyvista as pv\r\nfrom pyvista import _vtk\r\n\r\nvertices = np.array([[0, 0, 0], [1, 0, 0], [1, 1, 0], [0, 1, 0], [0.5, 0.5, -1]])\r\n\r\n# mesh faces\r\nfaces = np.hstack(\r\n    [\r\n        [4, 0, 1, 2, 3],  # square\r\n        [3, 0, 1, 4],  # triangle\r\n        [3, 1, 2, 4],  # triangle\r\n    ]\r\n)\r\n\r\nsurf = pv.PolyData(vertices, faces)\r\n```\r\n\r\n```py\r\n>>> _vtk.vtk_to_numpy(surf.GetPolys().GetData())\r\narray([4, 0, 1, 2, 3, 3, 0, 1, 4, 3, 1, 2, 4])\r\n```\r\n\r\n```py\r\n>>> _vtk.vtk_to_numpy(surf.GetPolys().GetConnectivityArray())\r\narray([0, 1, 2, 3, 0, 1, 4, 1, 2, 4])\r\n```\r\n\r\n```py\r\n>>> _vtk.vtk_to_numpy(surf.GetPolys().GetOffsetsArray())\r\narray([ 0,  4,  7, 10])\r\n```\r\n\r\nIn PyVista, we designed the `.cells`, `.faces`, `.strips`, `.lines`, etc. APIs around the assumption of having the connectivity offsets embedded in the array. This means we'd have to redesign these accessors entirely (breaking a lot of downstream code) just to enable this in-place operation. Further, changing the API to require users to deal with both connectivities and offsets as separate arrays will likely be too much of a burden.\r\n\r\n------\r\n\r\n> the placeholder implementation now makes a copy from its internal array to return. So changing this array does not actually affect the connectivity.\r\n\r\n@berkgeveci, I'm curious if you think this copy could/will be fixed upstream in VTK so that `GetData()` could work with in-place operations?\nAh I see. Unfortunately, what you want is not possible. The underlying structure for `vtkCellArray` changed and it does not store the number of vertices per cell anymore. This is the main reason why `GetData()` creates a new array (something that I forgot). For backwards compatibility. But it has the side effect of this array being \"detached\" from the actual topology of the mesh. There are good reasons for this change:\r\n\r\n- Most other codes and OpenGL prefer the new structure so there is more likelihood that VTK will be binary compatible with them.\r\n- When we get rid of the vertex count and always have the offsets, we can have random access at no additional memory cost. In the past, for polydata, the offsets were built when needed to save memory.\r\n\r\nNote that there is no loss of information here. You can simply recover the number of vertices of a cell by doing `offset[cellId+1]-offset[cellId]`. To make this work, the offsets array actually is of size `ncells+1` so that you can get the number of vertices for the last cell.\r\n\r\nI understand that this will be a backwards incompatible change for PyVista but there is no other way. You can still support\r\n\r\n```\r\n# mesh faces\r\nfaces = np.hstack(\r\n    [\r\n        [4, 0, 1, 2, 3],  # square\r\n        [3, 0, 1, 4],  # triangle\r\n        [3, 1, 2, 4],  # triangle\r\n    ]\r\n)\r\n\r\nsurf = pv.PolyData(vertices, faces)\r\n```\r\n\r\nbut it will have to create the connectivity and offsets array under the covers (or let VTK do it). But if you want to avoid the deep copy, you have to support `pv.PolyData(vertices, offsets, connectivity)`, which has to create a `vtkCellArray` and call `SetData()` on it.\r\nI suggest avoiding the use of `GetData()` all together.\r\n\r\nAlso note that there is a bug in how `SetData()` works currently and when called from Python, it does not cause `vtkCellArray()` to hold a reference to the underlying numpy buffer. Unless you are keeping a reference to that object from the Python side, the VTK object will hold a reference to a junk pointer once the numpy array goes out of scope. I am working on a fix for this issue.\r\n", "created_at": "2023-04-24T18:15:27Z", "version": "0.39", "FAIL_TO_PASS": ["tests/test_grid.py::test_init_from_arrays[False]", "tests/test_grid.py::test_init_from_arrays[True]", "tests/test_polydata.py::test_init_from_arrays"], "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_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_grid_from_specs", "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", "tests/test_polydata.py::test_init", "tests/test_polydata.py::test_init_from_pdata", "tests/test_polydata.py::test_init_from_arrays_with_vert", "tests/test_polydata.py::test_init_from_arrays_triangular", "tests/test_polydata.py::test_init_as_points", "tests/test_polydata.py::test_init_as_points_from_list", "tests/test_polydata.py::test_invalid_init", "tests/test_polydata.py::test_invalid_file", "tests/test_polydata.py::test_lines_on_init", "tests/test_polydata.py::test_polydata_repr_str", "tests/test_polydata.py::test_geodesic", "tests/test_polydata.py::test_geodesic_fail", "tests/test_polydata.py::test_geodesic_distance", "tests/test_polydata.py::test_ray_trace", "tests/test_polydata.py::test_edge_mask", "tests/test_polydata.py::test_boolean_union_intersection", "tests/test_polydata.py::test_boolean_difference", "tests/test_polydata.py::test_boolean_difference_fail", "tests/test_polydata.py::test_subtract", "tests/test_polydata.py::test_merge", "tests/test_polydata.py::test_add", "tests/test_polydata.py::test_intersection", "tests/test_polydata.py::test_curvature[mean]", "tests/test_polydata.py::test_curvature[gaussian]", "tests/test_polydata.py::test_curvature[maximum]", "tests/test_polydata.py::test_curvature[minimum]", "tests/test_polydata.py::test_invalid_curvature", "tests/test_polydata.py::test_save[.ply-True]", "tests/test_polydata.py::test_save[.ply-False]", "tests/test_polydata.py::test_save[.vtp-True]", "tests/test_polydata.py::test_save[.vtp-False]", "tests/test_polydata.py::test_save[.stl-True]", "tests/test_polydata.py::test_save[.stl-False]", "tests/test_polydata.py::test_save[.vtk-True]", "tests/test_polydata.py::test_save[.vtk-False]", "tests/test_polydata.py::test_save_ply_texture_array[3-True]", "tests/test_polydata.py::test_save_ply_texture_array[3-False]", "tests/test_polydata.py::test_save_ply_texture_array[4-True]", "tests/test_polydata.py::test_save_ply_texture_array[4-False]", "tests/test_polydata.py::test_save_ply_texture_array_catch[True]", "tests/test_polydata.py::test_save_ply_texture_array_catch[False]", "tests/test_polydata.py::test_pathlib_read_write", "tests/test_polydata.py::test_invalid_save", "tests/test_polydata.py::test_triangulate_filter", "tests/test_polydata.py::test_subdivision[butterfly]", "tests/test_polydata.py::test_subdivision[loop]", "tests/test_polydata.py::test_subdivision[linear]", "tests/test_polydata.py::test_invalid_subdivision", "tests/test_polydata.py::test_extract_feature_edges", "tests/test_polydata.py::test_extract_feature_edges_no_data", "tests/test_polydata.py::test_decimate", "tests/test_polydata.py::test_decimate_pro", "tests/test_polydata.py::test_compute_normals", "tests/test_polydata.py::test_compute_normals_inplace", "tests/test_polydata.py::test_compute_normals_split_vertices", "tests/test_polydata.py::test_point_normals", "tests/test_polydata.py::test_cell_normals", "tests/test_polydata.py::test_face_normals", "tests/test_polydata.py::test_clip_plane", "tests/test_polydata.py::test_extract_largest", "tests/test_polydata.py::test_clean", "tests/test_polydata.py::test_area", "tests/test_polydata.py::test_volume", "tests/test_polydata.py::test_remove_points_any", "tests/test_polydata.py::test_remove_points_all", "tests/test_polydata.py::test_remove_points_fail", "tests/test_polydata.py::test_vertice_cells_on_read", "tests/test_polydata.py::test_center_of_mass", "tests/test_polydata.py::test_project_points_to_plane", "tests/test_polydata.py::test_tube", "tests/test_polydata.py::test_smooth_inplace", "tests/test_polydata.py::test_delaunay_2d", "tests/test_polydata.py::test_lines", "tests/test_polydata.py::test_strips", "tests/test_polydata.py::test_ribbon_filter", "tests/test_polydata.py::test_is_all_triangles", "tests/test_polydata.py::test_extrude", "tests/test_polydata.py::test_extrude_capping_warnings", "tests/test_polydata.py::test_flip_normals", "tests/test_polydata.py::test_n_verts", "tests/test_polydata.py::test_n_lines", "tests/test_polydata.py::test_geodesic_disconnected"], "environment_setup_commit": "4c2d1aed10b1600d520271beba8579c71433e808"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-3608", "base_commit": "d783e421b714ed989d9e641977ea9b3b6ffaf807", "patch": "diff --git a/src/sqlfluff/cli/__init__.py b/src/sqlfluff/cli/__init__.py\n--- a/src/sqlfluff/cli/__init__.py\n+++ b/src/sqlfluff/cli/__init__.py\n@@ -1 +1,6 @@\n \"\"\"init py for cli.\"\"\"\n+\n+\n+EXIT_SUCCESS = 0\n+EXIT_FAIL = 1\n+EXIT_ERROR = 2\ndiff --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@@ -22,6 +22,7 @@\n from tqdm import tqdm\n from sqlfluff.cli.autocomplete import shell_completion_enabled, dialect_shell_complete\n \n+from sqlfluff.cli import EXIT_SUCCESS, EXIT_ERROR, EXIT_FAIL\n from sqlfluff.cli.formatters import (\n     format_linting_result_header,\n     OutputStreamFormatter,\n@@ -154,7 +155,7 @@ def __exit__(self, exc_type, exc_val, exc_tb):\n                     Color.red,\n                 )\n             )\n-            sys.exit(1)\n+            sys.exit(EXIT_ERROR)\n         elif exc_type is SQLFluffUserError:\n             click.echo(\n                 \"\\nUser Error: \"\n@@ -163,7 +164,7 @@ def __exit__(self, exc_type, exc_val, exc_tb):\n                     Color.red,\n                 )\n             )\n-            sys.exit(1)\n+            sys.exit(EXIT_ERROR)\n \n \n def common_options(f: Callable) -> Callable:\n@@ -335,7 +336,7 @@ def get_config(\n                     color=Color.red,\n                 )\n             )\n-            sys.exit(66)\n+            sys.exit(EXIT_ERROR)\n         except KeyError:\n             click.echo(\n                 OutputStreamFormatter.colorize_helper(\n@@ -344,7 +345,7 @@ def get_config(\n                     color=Color.red,\n                 )\n             )\n-            sys.exit(66)\n+            sys.exit(EXIT_ERROR)\n     from_root_kwargs = {}\n     if \"require_dialect\" in kwargs:\n         from_root_kwargs[\"require_dialect\"] = kwargs.pop(\"require_dialect\")\n@@ -365,7 +366,7 @@ def get_config(\n                 color=Color.red,\n             )\n         )\n-        sys.exit(66)\n+        sys.exit(EXIT_ERROR)\n \n \n def get_linter_and_formatter(\n@@ -380,7 +381,7 @@ def get_linter_and_formatter(\n             dialect_selector(dialect)\n     except KeyError:  # pragma: no cover\n         click.echo(f\"Error: Unknown dialect '{cfg.get('dialect')}'\")\n-        sys.exit(66)\n+        sys.exit(EXIT_ERROR)\n     formatter = OutputStreamFormatter(\n         output_stream=output_stream or make_output_stream(cfg),\n         nocolor=cfg.get(\"nocolor\"),\n@@ -635,7 +636,7 @@ def lint(\n             formatter.completion_message()\n         sys.exit(result.stats()[\"exit code\"])\n     else:\n-        sys.exit(0)\n+        sys.exit(EXIT_SUCCESS)\n \n \n def do_fixes(lnt, result, formatter=None, **kwargs):\n@@ -730,7 +731,7 @@ def fix(\n     verbose = config.get(\"verbose\")\n     progress_bar_configuration.disable_progress_bar = disable_progress_bar\n \n-    exit_code = 0\n+    exit_code = EXIT_SUCCESS\n \n     formatter.dispatch_config(lnt)\n \n@@ -780,7 +781,7 @@ def fix(\n             )\n \n         click.echo(stdout, nl=False)\n-        sys.exit(1 if templater_error or unfixable_error else exit_code)\n+        sys.exit(EXIT_FAIL if templater_error or unfixable_error else exit_code)\n \n     # Lint the paths (not with the fix argument at this stage), outputting as we go.\n     click.echo(\"==== finding fixable violations ====\")\n@@ -816,7 +817,7 @@ def fix(\n                 fixed_file_suffix=fixed_suffix,\n             )\n             if not success:\n-                sys.exit(1)  # pragma: no cover\n+                sys.exit(EXIT_FAIL)  # pragma: no cover\n         else:\n             click.echo(\n                 \"Are you sure you wish to attempt to fix these? [Y/n] \", nl=False\n@@ -833,16 +834,16 @@ def fix(\n                     fixed_file_suffix=fixed_suffix,\n                 )\n                 if not success:\n-                    sys.exit(1)  # pragma: no cover\n+                    sys.exit(EXIT_FAIL)  # pragma: no cover\n                 else:\n                     formatter.completion_message()\n             elif c == \"n\":\n                 click.echo(\"Aborting...\")\n-                exit_code = 1\n+                exit_code = EXIT_FAIL\n             else:  # pragma: no cover\n                 click.echo(\"Invalid input, please enter 'Y' or 'N'\")\n                 click.echo(\"Aborting...\")\n-                exit_code = 1\n+                exit_code = EXIT_FAIL\n     else:\n         click.echo(\"==== no fixable linting violations found ====\")\n         formatter.completion_message()\n@@ -851,7 +852,7 @@ def fix(\n         (\n             dict(types=SQLLintError, fixable=False),\n             \"  [{} unfixable linting violations found]\",\n-            1,\n+            EXIT_FAIL,\n         ),\n     ]\n     for num_violations_kwargs, message_format, error_level in error_types:\n@@ -986,7 +987,7 @@ def parse(\n             import cProfile\n         except ImportError:  # pragma: no cover\n             click.echo(\"The cProfiler is not available on your platform.\")\n-            sys.exit(1)\n+            sys.exit(EXIT_ERROR)\n         pr = cProfile.Profile()\n         pr.enable()\n \n@@ -1053,9 +1054,9 @@ def parse(\n         click.echo(\"\\n\".join(profiler_buffer.getvalue().split(\"\\n\")[:50]))\n \n     if violations_count > 0 and not nofail:\n-        sys.exit(66)  # pragma: no cover\n+        sys.exit(EXIT_FAIL)  # pragma: no cover\n     else:\n-        sys.exit(0)\n+        sys.exit(EXIT_SUCCESS)\n \n \n # This \"__main__\" handler allows invoking SQLFluff using \"python -m\", which\ndiff --git a/src/sqlfluff/cli/formatters.py b/src/sqlfluff/cli/formatters.py\n--- a/src/sqlfluff/cli/formatters.py\n+++ b/src/sqlfluff/cli/formatters.py\n@@ -6,6 +6,7 @@\n import click\n from colorama import Style\n \n+from sqlfluff.cli import EXIT_FAIL, EXIT_SUCCESS\n from sqlfluff.cli.helpers import (\n     get_package_version,\n     get_python_version,\n@@ -14,6 +15,7 @@\n     wrap_field,\n )\n from sqlfluff.cli.outputstream import OutputStream\n+\n from sqlfluff.core import SQLBaseError, FluffConfig, Linter, TimingSummary\n from sqlfluff.core.enums import Color\n from sqlfluff.core.linter import LintedFile, LintingResult, ParsedString\n@@ -517,7 +519,7 @@ def handle_files_with_tmp_or_prs_errors(self, lint_result: LintingResult) -> int\n                         color,\n                     )\n                 )\n-        return 1 if num_filtered_errors else 0\n+        return EXIT_FAIL if num_filtered_errors else EXIT_SUCCESS\n \n     def print_out_violations_and_timing(\n         self,\ndiff --git a/src/sqlfluff/core/linter/linting_result.py b/src/sqlfluff/core/linter/linting_result.py\n--- a/src/sqlfluff/core/linter/linting_result.py\n+++ b/src/sqlfluff/core/linter/linting_result.py\n@@ -11,6 +11,7 @@\n )\n from typing_extensions import Literal\n \n+from sqlfluff.cli import EXIT_FAIL, EXIT_SUCCESS\n \n from sqlfluff.core.errors import (\n     CheckTuple,\n@@ -23,8 +24,6 @@\n \n # Classes needed only for type checking\n from sqlfluff.core.parser.segments.base import BaseSegment\n-\n-\n from sqlfluff.core.linter.linted_dir import LintedDir\n \n \n@@ -133,7 +132,9 @@ def stats(self) -> Dict[str, Any]:\n             all_stats[\"unclean rate\"] = 0\n         all_stats[\"clean files\"] = all_stats[\"clean\"]\n         all_stats[\"unclean files\"] = all_stats[\"unclean\"]\n-        all_stats[\"exit code\"] = 65 if all_stats[\"violations\"] > 0 else 0\n+        all_stats[\"exit code\"] = (\n+            EXIT_FAIL if all_stats[\"violations\"] > 0 else EXIT_SUCCESS\n+        )\n         all_stats[\"status\"] = \"FAIL\" if all_stats[\"violations\"] > 0 else \"PASS\"\n         return all_stats\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@@ -71,7 +71,7 @@ def invoke_assert_code(\n def test__cli__command_directed():\n     \"\"\"Basic checking of lint functionality.\"\"\"\n     result = invoke_assert_code(\n-        ret_code=65,\n+        ret_code=1,\n         args=[\n             lint,\n             [\n@@ -95,7 +95,7 @@ def test__cli__command_dialect():\n     \"\"\"Check the script raises the right exception on an unknown dialect.\"\"\"\n     # The dialect is unknown should be a non-zero exit code\n     invoke_assert_code(\n-        ret_code=66,\n+        ret_code=2,\n         args=[\n             lint,\n             [\n@@ -112,7 +112,7 @@ def test__cli__command_no_dialect():\n     \"\"\"Check the script raises the right exception no dialect.\"\"\"\n     # The dialect is unknown should be a non-zero exit code\n     result = invoke_assert_code(\n-        ret_code=1,\n+        ret_code=2,\n         args=[\n             lint,\n             [\"-\"],\n@@ -129,7 +129,7 @@ def test__cli__command_parse_error_dialect_explicit_warning():\n     # and a human-readable warning should be dislayed.\n     # Dialect specified as commandline option.\n     result = invoke_assert_code(\n-        ret_code=66,\n+        ret_code=1,\n         args=[\n             parse,\n             [\n@@ -152,7 +152,7 @@ def test__cli__command_parse_error_dialect_implicit_warning():\n     # and a human-readable warning should be dislayed.\n     # Dialect specified in .sqlfluff config.\n     result = invoke_assert_code(\n-        ret_code=66,\n+        ret_code=1,\n         args=[\n             # Config sets dialect to tsql\n             parse,\n@@ -173,7 +173,7 @@ def test__cli__command_parse_error_dialect_implicit_warning():\n def test__cli__command_dialect_legacy():\n     \"\"\"Check the script raises the right exception on a legacy dialect.\"\"\"\n     result = invoke_assert_code(\n-        ret_code=66,\n+        ret_code=2,\n         args=[\n             lint,\n             [\n@@ -190,7 +190,7 @@ def test__cli__command_dialect_legacy():\n def test__cli__command_extra_config_fail():\n     \"\"\"Check the script raises the right exception non-existent extra config path.\"\"\"\n     result = invoke_assert_code(\n-        ret_code=66,\n+        ret_code=2,\n         args=[\n             lint,\n             [\n@@ -429,7 +429,7 @@ def test__cli__command_lint_parse(command):\n                 [\"test/fixtures/cli/unknown_jinja_tag/test.sql\", \"-vvvvvvv\"],\n                 \"y\",\n             ),\n-            65,\n+            1,\n         ),\n     ],\n )\n@@ -461,7 +461,7 @@ def test__cli__command_lint_skip_ignore_files():\n             \"--disregard-sqlfluffignores\",\n         ],\n     )\n-    assert result.exit_code == 65\n+    assert result.exit_code == 1\n     assert \"L009\" in result.output.strip()\n \n \n@@ -488,7 +488,7 @@ def test__cli__command_lint_ignore_local_config():\n             \"test/fixtures/cli/ignore_local_config/ignore_local_config_test.sql\",\n         ],\n     )\n-    assert result.exit_code == 65\n+    assert result.exit_code == 1\n     assert \"L012\" in result.output.strip()\n \n \n@@ -561,7 +561,7 @@ def generic_roundtrip_test(\n     old_mode = stat.S_IMODE(status.st_mode)\n     # Check that we first detect the issue\n     invoke_assert_code(\n-        ret_code=65, args=[lint, [\"--dialect=ansi\", \"--rules\", rulestring, filepath]]\n+        ret_code=1, args=[lint, [\"--dialect=ansi\", \"--rules\", rulestring, filepath]]\n     )\n     # Fix the file (in force mode)\n     if force:\n@@ -997,7 +997,7 @@ def test__cli__command_fix_stdin_error_exit_code(\n     \"rule,fname,prompt,exit_code,fix_exit_code\",\n     [\n         (\"L001\", \"test/fixtures/linter/indentation_errors.sql\", \"y\", 0, 0),\n-        (\"L001\", \"test/fixtures/linter/indentation_errors.sql\", \"n\", 65, 1),\n+        (\"L001\", \"test/fixtures/linter/indentation_errors.sql\", \"n\", 1, 1),\n     ],\n )\n def test__cli__command__fix_no_force(rule, fname, prompt, exit_code, fix_exit_code):\n@@ -1075,7 +1075,7 @@ def test__cli__command_parse_serialize_from_stdin(serialize, write_file, tmp_pat\n                     ],\n                 }\n             ],\n-            65,\n+            1,\n         ),\n     ],\n )\n@@ -1115,7 +1115,7 @@ def test__cli__command_lint_serialize_from_stdin(serialize, sql, expected, exit_\n )\n def test__cli__command_fail_nice_not_found(command):\n     \"\"\"Check commands fail as expected when then don't find files.\"\"\"\n-    result = invoke_assert_code(args=command, ret_code=1)\n+    result = invoke_assert_code(args=command, ret_code=2)\n     assert \"could not be accessed\" in result.output\n \n \n@@ -1180,7 +1180,7 @@ def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_\n     # note the file is in here twice. two files = two payloads.\n     result = invoke_assert_code(\n         args=[lint, cmd_args],\n-        ret_code=65,\n+        ret_code=1,\n     )\n \n     if write_file:\n@@ -1226,7 +1226,7 @@ def test__cli__command_lint_serialize_github_annotation():\n                 \"--disable_progress_bar\",\n             ),\n         ],\n-        ret_code=65,\n+        ret_code=1,\n     )\n     result = json.loads(result.output)\n     assert result == [\n@@ -1337,7 +1337,7 @@ def test__cli__command_lint_serialize_github_annotation_native():\n                 \"--disable_progress_bar\",\n             ),\n         ],\n-        ret_code=65,\n+        ret_code=1,\n     )\n \n     assert result.output == \"\\n\".join(\n@@ -1381,7 +1381,7 @@ def test__cli__command_lint_serialize_annotation_level_error_failure_equivalent(\n                 \"--disable_progress_bar\",\n             ),\n         ],\n-        ret_code=65,\n+        ret_code=1,\n     )\n \n     result_failure = invoke_assert_code(\n@@ -1396,7 +1396,7 @@ def test__cli__command_lint_serialize_annotation_level_error_failure_equivalent(\n                 \"--disable_progress_bar\",\n             ),\n         ],\n-        ret_code=65,\n+        ret_code=1,\n     )\n \n     assert result_error.output == result_failure.output\n@@ -1450,7 +1450,7 @@ def test_cli_encoding(encoding, method, expect_success, tmpdir):\n         shutil.copy(sql_path, tmpdir)\n         options = [str(tmpdir / \"encoding_test.sql\")]\n     result = invoke_assert_code(\n-        ret_code=65,\n+        ret_code=1,\n         args=[\n             lint,\n             options,\n@@ -1479,7 +1479,7 @@ def test_cli_no_disable_noqa_flag():\n def test_cli_disable_noqa_flag():\n     \"\"\"Test that --disable_noqa flag ignores inline noqa comments.\"\"\"\n     result = invoke_assert_code(\n-        ret_code=65,\n+        ret_code=1,\n         args=[\n             lint,\n             [\n@@ -1563,7 +1563,7 @@ def test_cli_lint_enabled_progress_bar_multiple_paths(\n     ) -> None:\n         \"\"\"When progress bar is enabled, there should be some tracks in output.\"\"\"\n         result = invoke_assert_code(\n-            ret_code=65,\n+            ret_code=1,\n             args=[\n                 lint,\n                 [\ndiff --git a/test/rules/std_roundtrip_test.py b/test/rules/std_roundtrip_test.py\n--- a/test/rules/std_roundtrip_test.py\n+++ b/test/rules/std_roundtrip_test.py\n@@ -34,7 +34,7 @@ def generic_roundtrip_test(source_file, rulestring):\n     runner = CliRunner()\n     # Check that we first detect the issue\n     result = runner.invoke(lint, [\"--rules\", rulestring, \"--dialect=ansi\", filepath])\n-    assert result.exit_code == 65\n+    assert result.exit_code == 1\n     # Fix the file (in force mode)\n     result = runner.invoke(\n         fix, [\"--rules\", rulestring, \"--dialect=ansi\", \"-f\", filepath]\n@@ -80,7 +80,7 @@ def jinja_roundtrip_test(\n     result = runner.invoke(\n         lint, [\"--rules\", rulestring, \"--dialect=ansi\", sql_filepath]\n     )\n-    assert result.exit_code == 65\n+    assert result.exit_code == 1\n     # Fix the file (in force mode)\n     result = runner.invoke(\n         fix, [\"--rules\", rulestring, \"-f\", \"--dialect=ansi\", sql_filepath]\n", "problem_statement": "Return codes are inconsistent\n### Search before asking\r\n\r\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\r\n\r\n\r\n### What Happened\r\n\r\nWorking on #3431 - I noticed that we're inconsistent in our return codes.\r\n\r\nIn `commands.py` we call `sys.exit()` in 15 places (currently).\r\n\r\n- Twice we call `sys.exit(0)` on success, at the end of `parse` and `lint` (`fix` is a handled differently, see below). \u2714\ufe0f \r\n- Six times we call `sys.exit(1)` for a selection of things:\r\n  - Not having `cProfiler` installed.\r\n  - Failing to apply fixes\r\n  - User Errors and OSError (in `PathAndUserErrorHandler`)\r\n- Five times we call `sys.exit(66)` for a selection of things:\r\n  - User Errors (including unknown dialect or failing to load a dialect or config)\r\n  - If parsing failed when calling `parse`.\r\n- Once we use `handle_files_with_tmp_or_prs_errors` to determine the exit code (which returns 1 or 0)\r\n- Once we use `LintingResult.stats` to determine the exit code (which returns either 65 or 0)\r\n- Once we do a mixture of the above (see end of `fix`)\r\n\r\nThis neither DRY, or consistent ... or helpful?\r\n\r\n### Expected Behaviour\r\n\r\nWe should have consistent return codes for specific scenarios. There are up for discussion, but I would suggest:\r\n\r\n- 0 for success (obviously)\r\n- 1 for a fail which is error related: not having libraries installed, user errors etc...\r\n- 65 for a linting fail (i.e. no errors in running, but issues were found in either parsing or linting).\r\n- 66 for a fixing fail (i.e. we tried to fix errors but failed to do so for some reason).\r\n\r\nThese would be defined as constants at the top of `commands.py`.\r\n\r\n### Observed Behaviour\r\n\r\nsee above\r\n\r\n### How to reproduce\r\n\r\nsee above\r\n\r\n### Dialect\r\n\r\nN/A\r\n\r\n### Version\r\n\r\nDescription is as per code in #3431\r\n\r\n### Configuration\r\n\r\n-\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [X] Yes I am willing to submit a PR!\r\n\r\n### Code of Conduct\r\n\r\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\r\n\n", "hints_text": "I'm happy to contribute the changes for this one, but would appreciate views on what the error codes we should align on first @barrywhart @tunetheweb \nI'm not familiar with any widespread conventions about exit codes, except to keep them below 256.\r\n\r\nThis Stack Overflow post has a lot of discussion, often self-contradictory. https://stackoverflow.com/questions/1101957/are-there-any-standard-exit-status-codes-in-linux\r\n\r\nOverall, your proposal sounds good to me.\r\n\r\nCan you also search the existing issues for any mention of exit codes? I think there may be one or two open issues, perhaps related to the behavior when \"fix\" finds issues but some are unfixable. Because of its multifaceted nature as a linter and fixer that is used both interactively (e.g. during pre-commit) and in batch (CICD), SQLFluff perhaps has more stringent requirements for precise exit codes than some other tools. Do you think it'd be useful to review existing (and or write some new) user documentation before starting the coding, to help get a better understanding of the various use cases?\r\n\r\n\r\n\nAgree with @barrywhart 's comments.\r\n\r\nOnly question is why 65/66 instead of just 2/3?\n> Only question is why 65/66 instead of just 2/3?\r\n\r\nThis was initially because I had read that codes 0-64 were reserved for system usage but it appears things aren't that consistent.\r\n\r\n> This Stack Overflow post has a lot of discussion, often self-contradictory...\r\n\r\nI'm wondering based on this post whether we should simplify things:\r\n- 0: success\r\n- 1: fail (on linting or fixing, but due to finding issues with code or unable to fix, no \"errors\")\r\n- 2: fail because misuse or error\r\n\r\nIt's slightly less granular but a little more consistent with the bash approach (from the most recent post on that SO question):\r\n\r\n> Exit status 0: success\r\n> Exit status 1: \"failure\", as defined by the program\r\n> Exit status 2: command line usage error\nCleaning up the exit codes seems sensible. How likely do we think it is to break things for users?\nRelatively unlikely I reckon - I'm not sure the existing codes are sufficiently granular to be useful right now.", "created_at": "2022-07-14T15:06:34Z", "version": "1.1", "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_parse_with_retcode[command3-1]", "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__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_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_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[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_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_disable_noqa_flag", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_paths", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L001-test/fixtures/linter/indentation_errors.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L008-test/fixtures/linter/whitespace_errors.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L008-test/fixtures/linter/indentation_errors.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L010-test/fixtures/linter/whitespace_errors.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L011-test/fixtures/dialects/ansi/select_simple_i.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix[L012-test/fixtures/dialects/ansi/select_simple_i.sql]", "test/rules/std_roundtrip_test.py::test__cli__command__fix_templated[L010]", "test/rules/std_roundtrip_test.py::test__cli__command__fix_templated[L001]"], "PASS_TO_PASS": ["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[command22]", "test/cli/commands_test.py::test__cli__command_lint_parse[command23]", "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_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__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_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_nocolor", "test/cli/commands_test.py::test___main___help", "test/cli/commands_test.py::test_cli_no_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_enabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_lint_enabled_progress_bar_multiple_files"], "environment_setup_commit": "d83ab36bbb21f62cf0780d095a8be8cd366735d7"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4041", "base_commit": "66fd602d0824138f212082c8fdf381266a9edad3", "patch": "diff --git a/src/sqlfluff/core/errors.py b/src/sqlfluff/core/errors.py\n--- a/src/sqlfluff/core/errors.py\n+++ b/src/sqlfluff/core/errors.py\n@@ -1,5 +1,5 @@\n \"\"\"Errors - these are closely linked to what used to be called violations.\"\"\"\n-from typing import Optional, Tuple, List\n+from typing import Optional, Tuple, Any, List\n \n CheckTuple = Tuple[str, int, int]\n \n@@ -85,6 +85,18 @@ def get_info_dict(self):\n             \"description\": self.desc(),\n         }\n \n+    def check_tuple(self) -> CheckTuple:\n+        \"\"\"Get a tuple representing this error. Mostly for testing.\"\"\"\n+        return (\n+            self.rule_code(),\n+            self.line_no,\n+            self.line_pos,\n+        )\n+\n+    def source_signature(self) -> Tuple[Any, ...]:\n+        \"\"\"Return hashable source signature for deduplication.\"\"\"\n+        return (self.check_tuple(), self.desc())\n+\n     def ignore_if_in(self, ignore_iterable: List[str]):\n         \"\"\"Ignore this violation if it matches the iterable.\"\"\"\n         if self._identifier in ignore_iterable:\n@@ -189,13 +201,21 @@ def fixable(self):\n             return True\n         return False\n \n-    def check_tuple(self) -> CheckTuple:\n-        \"\"\"Get a tuple representing this error. Mostly for testing.\"\"\"\n-        return (\n-            self.rule.code,\n-            self.line_no,\n-            self.line_pos,\n+    def source_signature(self) -> Tuple[Any, ...]:\n+        \"\"\"Return hashable source signature for deduplication.\n+\n+        For linting errors we need to dedupe on more than just location and\n+        description, we also need to check the edits potentially made, both\n+        in the templated file but also in the source.\n+        \"\"\"\n+        fix_raws = tuple(\n+            tuple(e.raw for e in f.edit) if f.edit else None for f in self.fixes\n+        )\n+        source_fixes = tuple(\n+            tuple(tuple(e.source_fixes) for e in f.edit) if f.edit else None\n+            for f in self.fixes\n         )\n+        return (self.check_tuple(), self.description, fix_raws, source_fixes)\n \n     def __repr__(self):\n         return \"<SQLLintError: rule {} pos:{!r}, #fixes: {}, description: {}>\".format(\ndiff --git a/src/sqlfluff/core/linter/linted_file.py b/src/sqlfluff/core/linter/linted_file.py\n--- a/src/sqlfluff/core/linter/linted_file.py\n+++ b/src/sqlfluff/core/linter/linted_file.py\n@@ -54,18 +54,43 @@ def check_tuples(self, raise_on_non_linting_violations=True) -> List[CheckTuple]\n         \"\"\"Make a list of check_tuples.\n \n         This assumes that all the violations found are\n-        linting violations (and therefore implement `check_tuple()`).\n-        If they don't then this function raises that error.\n+        linting violations. If they don't then this function\n+        raises that error.\n         \"\"\"\n         vs: List[CheckTuple] = []\n         v: SQLLintError\n         for v in self.get_violations():\n-            if hasattr(v, \"check_tuple\"):\n+            if isinstance(v, SQLLintError):\n                 vs.append(v.check_tuple())\n             elif raise_on_non_linting_violations:\n                 raise v\n         return vs\n \n+    @staticmethod\n+    def deduplicate_in_source_space(\n+        violations: List[SQLBaseError],\n+    ) -> List[SQLBaseError]:\n+        \"\"\"Removes duplicates in the source space.\n+\n+        This is useful for templated files with loops, where we'll\n+        get a violation for each pass around the loop, but the user\n+        only cares about it once and we're only going to fix it once.\n+\n+        By filtering them early we get a more a more helpful CLI\n+        output *and* and more efficient fixing routine (by handling\n+        fewer fixes).\n+        \"\"\"\n+        new_violations = []\n+        dedupe_buffer = set()\n+        for v in violations:\n+            signature = v.source_signature()\n+            if signature not in dedupe_buffer:\n+                new_violations.append(v)\n+                dedupe_buffer.add(signature)\n+            else:\n+                linter_logger.debug(\"Removing duplicate source violation: %s\", v)\n+        return new_violations\n+\n     def get_violations(\n         self,\n         rules: Optional[Union[str, Tuple[str, ...]]] = None,\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@@ -736,7 +736,8 @@ def lint_parsed(\n \n         linted_file = LintedFile(\n             parsed.fname,\n-            violations,\n+            # Deduplicate violations\n+            LintedFile.deduplicate_in_source_space(violations),\n             time_dict,\n             tree,\n             ignore_mask=ignore_buff,\n", "test_patch": "diff --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@@ -407,8 +407,9 @@ def test__linter__empty_file():\n         (\n             False,\n             [\n-                (\"L006\", 3, 16),\n-                (\"L006\", 3, 16),\n+                # there are still two of each because L006 checks\n+                # for both *before* and *after* the operator.\n+                # The deduplication filter makes sure there aren't 4.\n                 (\"L006\", 3, 16),\n                 (\"L006\", 3, 16),\n                 (\"L006\", 3, 39),\n@@ -418,7 +419,11 @@ def test__linter__empty_file():\n     ],\n )\n def test__linter__mask_templated_violations(ignore_templated_areas, check_tuples):\n-    \"\"\"Test linter masks files properly around templated content.\"\"\"\n+    \"\"\"Test linter masks files properly around templated content.\n+\n+    NOTE: this also tests deduplication of fixes which have the same\n+    source position. i.e. `LintedFile.deduplicate_in_source_space()`.\n+    \"\"\"\n     lntr = Linter(\n         config=FluffConfig(\n             overrides={\n", "problem_statement": "Deduplicate violations in the same position\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\nWhen linting jinja files with loops we get multiple output violations for each time around the loop. e.g.\r\n\r\n```sql\r\nselect\r\n    a,\r\n    {% for val in [1, 2, 3, 4, 5, 6] %}\r\n        d+ {{ val }},\r\n    {% endfor %}\r\n    b\r\n```\r\n\r\nwe get\r\n\r\n```\r\n== [test.sql] FAIL\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   4 | P:  10 | L006 | Missing whitespace before +\r\nL:   7 | P:   1 | L001 | Unnecessary trailing whitespace.\r\n```\r\n\r\nThe duplicated `Missing whitespace` isn't helpful for the user. Regardless of whether we keep them in the background (perhaps we should), they shouldn't be shown to the user here because we're showing the same issue multiple times.\n\n### Use case\n\nCLI linting\n\n### Dialect\n\nall\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": "", "created_at": "2022-11-08T09:57:07Z", "version": "1.3", "FAIL_TO_PASS": ["test/core/linter_test.py::test__linter__mask_templated_violations[False-check_tuples1]"], "PASS_TO_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__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"], "environment_setup_commit": "dc59c2a5672aacedaf91f0e6129b467eefad331b"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4151", "base_commit": "dc59c2a5672aacedaf91f0e6129b467eefad331b", "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@@ -680,7 +680,7 @@ def do_fixes(lnt, result, formatter=None, **kwargs):\n     return False  # pragma: no cover\n \n \n-@cli.command()\n+@cli.command(cls=DeprecatedOptionsCommand)\n @common_options\n @core_options\n @click.option(\n@@ -710,9 +710,12 @@ def do_fixes(lnt, result, formatter=None, **kwargs):\n     ),\n )\n @click.option(\n+    \"--disable_progress_bar\",\n     \"--disable-progress-bar\",\n     is_flag=True,\n     help=\"Disables progress bars.\",\n+    cls=DeprecatedOption,\n+    deprecated=[\"--disable_progress_bar\"],\n )\n @click.option(\n     \"--FIX-EVEN-UNPARSABLE\",\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@@ -1775,6 +1775,46 @@ def test_cli_lint_enabled_progress_bar_multiple_files(\n         assert r\"\\rrule L001:\" in raw_output\n         assert r\"\\rrule L049:\" in raw_output\n \n+    def test_cli_fix_disabled_progress_bar(\n+        self, mock_disable_progress_bar: MagicMock\n+    ) -> None:\n+        \"\"\"When progress bar is disabled, nothing should be printed into output.\"\"\"\n+        result = invoke_assert_code(\n+            args=[\n+                fix,\n+                [\n+                    \"--disable-progress-bar\",\n+                    \"test/fixtures/linter/passing.sql\",\n+                ],\n+            ],\n+        )\n+        raw_output = repr(result.output)\n+\n+        assert (\n+            \"DeprecationWarning: The option '--disable_progress_bar' is deprecated, \"\n+            \"use '--disable-progress-bar'\"\n+        ) not in raw_output\n+\n+    def test_cli_fix_disabled_progress_bar_deprecated_option(\n+        self, mock_disable_progress_bar: MagicMock\n+    ) -> None:\n+        \"\"\"Same as above but checks additionally if deprecation warning is printed.\"\"\"\n+        result = invoke_assert_code(\n+            args=[\n+                fix,\n+                [\n+                    \"--disable_progress_bar\",\n+                    \"test/fixtures/linter/passing.sql\",\n+                ],\n+            ],\n+        )\n+        raw_output = repr(result.output)\n+\n+        assert (\n+            \"DeprecationWarning: The option '--disable_progress_bar' is deprecated, \"\n+            \"use '--disable-progress-bar'\"\n+        ) in raw_output\n+\n \n multiple_expected_output = \"\"\"==== finding fixable violations ====\n == [test/fixtures/linter/multiple_sql_errors.sql] FAIL\n", "problem_statement": "--disable_progress_bar Flag Broken for Fix\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\nI ran `sqlfluff fix ${target} --dialect ansi --disable_progress_bar --force` on version 1.4.0 and got an error with exit code 2. Running with `--disable-progress-bar` appears to work fine, but it appears that compatibility with underscores was broken in version 1.4.0.\n\n### Expected Behaviour\n\nShould run as expected, with no error and no progress bar.\n\n### Observed Behaviour\n\nExit code 2 and stderr:\r\n```\r\nUsage: sqlfluff fix [OPTIONS] [PATHS]...\r\n        Try 'sqlfluff fix -h' for help.\r\n\r\n        Error: No such option: --disable_progress_bar (Possible options: --disable-noqa, --disable-progress-bar)\r\n```\n\n### How to reproduce\n\nSql file:\r\n```\r\nSELECT foo FROM bar;\r\n```\r\n\r\nCommand:\r\n```\r\nsqlfluff fix ${target} --dialect ansi --disable_progress_bar --force\r\n```\n\n### Dialect\n\nansi\n\n### Version\n\npython 3.10.3\r\nsqlfluff 1.4.0 and up appears to have this problem (tested through 1.4.2)\n\n### Configuration\n\nNo special configuration. Ran hermetically with `trunk`.\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--disable_progress_bar Flag Broken for Fix\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\nI ran `sqlfluff fix ${target} --dialect ansi --disable_progress_bar --force` on version 1.4.0 and got an error with exit code 2. Running with `--disable-progress-bar` appears to work fine, but it appears that compatibility with underscores was broken in version 1.4.0.\n\n### Expected Behaviour\n\nShould run as expected, with no error and no progress bar.\n\n### Observed Behaviour\n\nExit code 2 and stderr:\r\n```\r\nUsage: sqlfluff fix [OPTIONS] [PATHS]...\r\n        Try 'sqlfluff fix -h' for help.\r\n\r\n        Error: No such option: --disable_progress_bar (Possible options: --disable-noqa, --disable-progress-bar)\r\n```\n\n### How to reproduce\n\nSql file:\r\n```\r\nSELECT foo FROM bar;\r\n```\r\n\r\nCommand:\r\n```\r\nsqlfluff fix ${target} --dialect ansi --disable_progress_bar --force\r\n```\n\n### Dialect\n\nansi\n\n### Version\n\npython 3.10.3\r\nsqlfluff 1.4.0 and up appears to have this problem (tested through 1.4.2)\n\n### Configuration\n\nNo special configuration. Ran hermetically with `trunk`.\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": "Looks like #3904 made `lint` work with both but updated `fix` to only accept `--disable-progress-bar`. I assume that was by accident. Should be relatively straightforward to fix by updating to match `lint`. \nLooks like #3904 made `lint` work with both but updated `fix` to only accept `--disable-progress-bar`. I assume that was by accident. Should be relatively straightforward to fix by updating to match `lint`. ", "created_at": "2022-12-11T16:33:31Z", "version": "1.3", "FAIL_TO_PASS": ["test/cli/commands_test.py::TestProgressBars::test_cli_fix_disabled_progress_bar_deprecated_option"], "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::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::TestProgressBars::test_cli_fix_disabled_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-3354", "base_commit": "36e89cbf2d13d5d95d2430f905a2fd122cf103c7", "patch": "diff --git a/src/sqlfluff/core/templaters/placeholder.py b/src/sqlfluff/core/templaters/placeholder.py\n--- a/src/sqlfluff/core/templaters/placeholder.py\n+++ b/src/sqlfluff/core/templaters/placeholder.py\n@@ -21,6 +21,8 @@\n KNOWN_STYLES = {\n     # e.g. WHERE bla = :name\n     \"colon\": regex.compile(r\"(?<![:\\w\\x5c]):(?P<param_name>\\w+)(?!:)\", regex.UNICODE),\n+    # e.g. WHERE bla = table:name - use with caution as more prone to false positives\n+    \"colon_nospaces\": regex.compile(r\":(?P<param_name>\\w+)\", regex.UNICODE),\n     # e.g. WHERE bla = :2\n     \"numeric_colon\": regex.compile(\n         r\"(?<![:\\w\\x5c]):(?P<param_name>\\d+)\", regex.UNICODE\n@@ -29,8 +31,10 @@\n     \"pyformat\": regex.compile(\n         r\"(?<![:\\w\\x5c])%\\((?P<param_name>[\\w_]+)\\)s\", regex.UNICODE\n     ),\n-    # e.g. WHERE bla = $name\n-    \"dollar\": regex.compile(r\"(?<![:\\w\\x5c])\\$(?P<param_name>[\\w_]+)\", regex.UNICODE),\n+    # e.g. WHERE bla = $name or WHERE bla = ${name}\n+    \"dollar\": regex.compile(\n+        r\"(?<![:\\w\\x5c])\\${?(?P<param_name>[\\w_]+)}?\", regex.UNICODE\n+    ),\n     # e.g. WHERE bla = ?\n     \"question_mark\": regex.compile(r\"(?<![:\\w\\x5c])\\?\", regex.UNICODE),\n     # e.g. WHERE bla = $3\n@@ -146,7 +150,7 @@ def process(\n                 param_name = found_param[\"param_name\"]\n             last_literal_length = span[0] - last_pos_raw\n             try:\n-                replacement = context[param_name]\n+                replacement = str(context[param_name])\n             except KeyError as err:\n                 # TODO: Add a url here so people can get more help.\n                 raise SQLTemplaterError(\n", "test_patch": "diff --git a/test/core/templaters/placeholder_test.py b/test/core/templaters/placeholder_test.py\n--- a/test/core/templaters/placeholder_test.py\n+++ b/test/core/templaters/placeholder_test.py\n@@ -79,6 +79,20 @@ def test__templater_raw():\n                 city_ids=\"(1, 2, 3, 45)\",\n             ),\n         ),\n+        (\n+            \"\"\"\n+            SELECT user_mail, city_id\n+            FROM users_data:table_suffix\n+            \"\"\",\n+            \"colon_nospaces\",\n+            \"\"\"\n+            SELECT user_mail, city_id\n+            FROM users_data42\n+            \"\"\",\n+            dict(\n+                table_suffix=\"42\",\n+            ),\n+        ),\n         (\n             \"\"\"\n             SELECT user_mail, city_id\n@@ -123,6 +137,8 @@ def test__templater_raw():\n             FROM users_data\n             WHERE (city_id) IN %(city_id)s\n             AND date > %(date)s\n+            AND someflag = %(someflag)s\n+            LIMIT %(limit)s\n             \"\"\",\n             \"pyformat\",\n             \"\"\"\n@@ -130,10 +146,11 @@ def test__templater_raw():\n             FROM users_data\n             WHERE (city_id) IN (1, 2, 3, 45)\n             AND date > '2020-10-01'\n+            AND someflag = False\n+            LIMIT 15\n             \"\"\",\n             dict(\n-                city_id=\"(1, 2, 3, 45)\",\n-                date=\"'2020-10-01'\",\n+                city_id=\"(1, 2, 3, 45)\", date=\"'2020-10-01'\", limit=15, someflag=False\n             ),\n         ),\n         (\n@@ -142,6 +159,7 @@ def test__templater_raw():\n             FROM users_data\n             WHERE (city_id) IN $city_id\n             AND date > $date\n+            OR date = ${date}\n             \"\"\",\n             \"dollar\",\n             \"\"\"\n@@ -149,6 +167,7 @@ def test__templater_raw():\n             FROM users_data\n             WHERE (city_id) IN (1, 2, 3, 45)\n             AND date > '2020-10-01'\n+            OR date = '2020-10-01'\n             \"\"\",\n             dict(\n                 city_id=\"(1, 2, 3, 45)\",\n@@ -221,6 +240,7 @@ def test__templater_raw():\n         \"colon_simple_substitution\",\n         \"colon_accept_block_at_end\",\n         \"colon_tuple_substitution\",\n+        \"colon_nospaces\",\n         \"question_mark\",\n         \"numeric_colon\",\n         \"pyformat\",\n", "problem_statement": "TypeError when using integer placeholder\n### Search before asking\r\n\r\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\r\n\r\n\r\n### What Happened\r\n\r\nAn exception occurs when trying to use integer substituents.\r\n\r\n### Expected Behaviour\r\n\r\nWork without errors.\r\n\r\n### Observed Behaviour\r\n\r\n\r\nAn exception occurs:\r\n```\r\n  ...\r\n  File \"venv/lib/python3.9/site-packages/sqlfluff/core/linter/linter.py\", line 816, in render_file\r\n    return self.render_string(raw_file, fname, config, encoding)\r\n  File \"venv/lib/python3.9/site-packages/sqlfluff/core/linter/linter.py\", line 787, in render_string\r\n    templated_file, templater_violations = self.templater.process(\r\n  File \"venv/lib/python3.9/site-packages/sqlfluff/core/templaters/placeholder.py\", line 183, in process\r\n    start_template_pos, start_template_pos + len(replacement), None\r\nTypeError: object of type 'int' has no len()\r\n\r\n```\r\n\r\n### How to reproduce\r\n\r\n1. Create a file `example.sql`:\r\n```\r\nSELECT 1\r\nLIMIT %(capacity)s;\r\n```\r\n2. Copy `.sqlfluff` from the Configuration section\r\n3. Run `sqlfluff lint --dialect postgres example.sql`\r\n\r\n### Dialect\r\n\r\npostgres\r\n\r\n### Version\r\n\r\nsqlfluff, version 0.13.1\r\n\r\n### Configuration\r\n\r\n```\r\n[sqlfluff]\r\nexclude_rules = L031\r\ntemplater = placeholder\r\n\r\n[sqlfluff:templater:placeholder]\r\nparam_style = pyformat\r\ncapacity = 15\r\n```\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [ ] Yes I am willing to submit a PR!\r\n\r\n### Code of Conduct\r\n\r\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\r\n\nSupport Postgres-style variable substitution\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\nThe Postgres `psql` utility supports flavor of colon-style variable substitution that currently confuses sqlfluff.  E.g.,\r\n\r\n```sql\r\nALTER TABLE name:variable RENAME TO name;\r\n```\r\n\r\nRunning the above through sqlfluff produces this output:\r\n\r\n```\r\nsqlfluff lint --dialect postgres 2.sql\r\n== [2.sql] FAIL\r\nL:   1 | P:   1 |  PRS | Line 1, Position 1: Found unparsable section: 'ALTER\r\n                       | TABLE name:variable RENAME TO name...'\r\n```\n\n### Use case\n\nI would like it if in the above the string \"name:variable\" were considered a valid table name (and other identifiers similarly).\n\n### Dialect\n\nThis applies to the Postgres dialect.\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": "\nThis sounds more like a templater feature than a dialect feature. Does psql allow variables to contain SQL fragments, e.g.: `WHERE foo = '3'`?\n> This sounds more like a templater feature than a dialect feature.\r\n\r\nTrue!  After looking over the code some, that may well be the right place to implement this.\r\n\r\n> Does psql allow variables to contain SQL fragments, e.g.: WHERE foo = '3'?\r\n\r\nYes.  E.g.,\r\n\r\n```\r\n% psql -v expression='2 + 2'\r\npsql (14.2, server 10.18)\r\nSSL connection (protocol: TLSv1.3, cipher: TLS_AES_256_GCM_SHA384, bits: 256, compression: off)\r\nType \"help\" for help.\r\n\r\ndb=> select :expression;\r\n ?column?\r\n----------\r\n        4\r\n(1 row)\r\n\r\ndb=> select 5:expression;\r\n ?column?\r\n----------\r\n       54\r\n(1 row)\r\n```\r\n\r\nMore at the [docs](https://www.postgresql.org/docs/current/app-psql.html#APP-PSQL-VARIABLES).", "created_at": "2022-05-17T11:50:34Z", "version": "0.12", "FAIL_TO_PASS": ["test/core/templaters/placeholder_test.py::test__templater_param_style[colon_nospaces]", "test/core/templaters/placeholder_test.py::test__templater_param_style[pyformat]", "test/core/templaters/placeholder_test.py::test__templater_param_style[dollar]"], "PASS_TO_PASS": ["test/core/templaters/placeholder_test.py::test__templater_raw", "test/core/templaters/placeholder_test.py::test__templater_param_style[no_changes]", "test/core/templaters/placeholder_test.py::test__templater_param_style[colon_simple_substitution]", "test/core/templaters/placeholder_test.py::test__templater_param_style[colon_accept_block_at_end]", "test/core/templaters/placeholder_test.py::test__templater_param_style[colon_tuple_substitution]", "test/core/templaters/placeholder_test.py::test__templater_param_style[question_mark]", "test/core/templaters/placeholder_test.py::test__templater_param_style[numeric_colon]", "test/core/templaters/placeholder_test.py::test__templater_param_style[numeric_dollar]", "test/core/templaters/placeholder_test.py::test__templater_param_style[percent]", "test/core/templaters/placeholder_test.py::test__templater_param_style[ampersand]", "test/core/templaters/placeholder_test.py::test__templater_custom_regex", "test/core/templaters/placeholder_test.py::test__templater_exception", "test/core/templaters/placeholder_test.py::test__templater_setup", "test/core/templaters/placeholder_test.py::test__templater_styles"], "environment_setup_commit": "8f6fd1d8a8d69b2c463fbcf5bd1131c47f12ad88"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4997", "base_commit": "50bbffd4672aa17af2651f40d533cf55048b7524", "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@@ -1356,7 +1356,7 @@ def render(\n             fname = path\n \n     # Get file specific config\n-    file_config.process_raw_file_for_config(raw_sql)\n+    file_config.process_raw_file_for_config(raw_sql, fname)\n     rendered = lnt.render_string(raw_sql, fname, file_config, \"utf8\")\n \n     if rendered.templater_violations:\ndiff --git a/src/sqlfluff/core/config.py b/src/sqlfluff/core/config.py\n--- a/src/sqlfluff/core/config.py\n+++ b/src/sqlfluff/core/config.py\n@@ -34,6 +34,16 @@\n ConfigElemType = Tuple[Tuple[str, ...], Any]\n \n \n+ALLOWABLE_LAYOUT_CONFIG_KEYS = (\n+    \"spacing_before\",\n+    \"spacing_after\",\n+    \"spacing_within\",\n+    \"line_position\",\n+    \"align_within\",\n+    \"align_scope\",\n+)\n+\n+\n @dataclass\n class _RemovedConfig:\n     old_path: Tuple[str, ...]\n@@ -501,13 +511,19 @@ def _incorporate_vals(ctx: dict, vals: List[ConfigElemType]) -> dict:\n     def _validate_configs(\n         configs: Iterable[ConfigElemType], file_path\n     ) -> List[ConfigElemType]:\n-        \"\"\"Validate config elements against removed list.\"\"\"\n+        \"\"\"Validate config elements.\n+\n+        We validate in two ways:\n+        1. Are these config settings removed or deprecated.\n+        2. Are these config elements in the layout section _valid_.\n+        \"\"\"\n         config_map = {cfg.old_path: cfg for cfg in REMOVED_CONFIGS}\n         # Materialise the configs into a list to we can iterate twice.\n         new_configs = list(configs)\n         defined_keys = {k for k, _ in new_configs}\n         validated_configs = []\n         for k, v in new_configs:\n+            # First validate against the removed option list.\n             if k in config_map.keys():\n                 formatted_key = \":\".join(k)\n                 removed_option = config_map[k]\n@@ -549,12 +565,37 @@ def _validate_configs(\n                 else:\n                     # Raise an error.\n                     raise SQLFluffUserError(\n-                        f\"Config file {file_path} set an outdated config \"\n+                        f\"Config file {file_path!r} set an outdated config \"\n                         f\"value {formatted_key}.\\n\\n{removed_option.warning}\\n\\n\"\n                         \"See https://docs.sqlfluff.com/en/stable/configuration.html\"\n                         \" for more details.\"\n                     )\n \n+            # Second validate any layout configs for validity.\n+            # NOTE: For now we don't check that the \"type\" is a valid one\n+            # to reference, or that the values are valid. For the values,\n+            # these are likely to be rejected by the layout routines at\n+            # runtime. The last risk area is validating that the type is\n+            # a valid one.\n+            if k and k[0] == \"layout\":\n+                # Check for:\n+                # - Key length\n+                # - Key values\n+                if (\n+                    # Key length must be 4\n+                    (len(k) != 4)\n+                    # Second value must (currently) be \"type\"\n+                    or (k[1] != \"type\")\n+                    # Last key value must be one of the allowable options.\n+                    or (k[3] not in ALLOWABLE_LAYOUT_CONFIG_KEYS)\n+                ):\n+                    raise SQLFluffUserError(\n+                        f\"Config file {file_path!r} set an invalid `layout` option \"\n+                        f\"value {':'.join(k)}.\\n\"\n+                        \"See https://docs.sqlfluff.com/en/stable/layout.html\"\n+                        \"#configuring-layout for more details.\"\n+                    )\n+\n             validated_configs.append((k, v))\n         return validated_configs\n \n@@ -1094,7 +1135,7 @@ def iter_vals(self, cfg: Optional[dict] = None) -> Iterable[tuple]:\n                 for idnt, key, val in self.iter_vals(cfg=cfg[k]):\n                     yield (idnt + 1, key, val)\n \n-    def process_inline_config(self, config_line: str):\n+    def process_inline_config(self, config_line: str, fname: str):\n         \"\"\"Process an inline config command and update self.\"\"\"\n         # Strip preceding comment marks\n         if config_line.startswith(\"--\"):\n@@ -1108,19 +1149,23 @@ def process_inline_config(self, config_line: str):\n         config_line = config_line[9:].strip()\n         # Divide on colons\n         config_path = [elem.strip() for elem in config_line.split(\":\")]\n+        config_val = (tuple(config_path[:-1]), config_path[-1])\n+        # Validate the value\n+        ConfigLoader._validate_configs([config_val], fname)\n         # Set the value\n-        self.set_value(config_path[:-1], config_path[-1])\n+        self.set_value(*config_val)\n         # If the config is for dialect, initialise the dialect\n         if config_path[:-1] == [\"dialect\"]:\n             self._initialise_dialect(config_path[-1])\n \n-    def process_raw_file_for_config(self, raw_str: str):\n+    def process_raw_file_for_config(self, raw_str: str, fname: str):\n         \"\"\"Process a full raw file for inline config and update self.\"\"\"\n         # Scan the raw file for config commands.\n         for raw_line in raw_str.splitlines():\n-            if raw_line.startswith(\"-- sqlfluff\"):\n+            # With or without a space.\n+            if raw_line.startswith((\"-- sqlfluff\", \"--sqlfluff\")):\n                 # Found a in-file config command\n-                self.process_inline_config(raw_line)\n+                self.process_inline_config(raw_line, fname)\n \n \n class ProgressBarConfiguration:\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@@ -141,7 +141,7 @@ def load_raw_file_and_config(\n         with open(fname, encoding=encoding, errors=\"backslashreplace\") as target_file:\n             raw_file = target_file.read()\n         # Scan the raw file for config commands.\n-        file_config.process_raw_file_for_config(raw_file)\n+        file_config.process_raw_file_for_config(raw_file, fname)\n         # Return the raw file and config\n         return raw_file, file_config, encoding\n \n@@ -897,7 +897,7 @@ def parse_string(\n         config = config or self.config\n \n         # Scan the raw file for config commands.\n-        config.process_raw_file_for_config(in_str)\n+        config.process_raw_file_for_config(in_str, fname)\n         rendered = self.render_string(in_str, fname, config, encoding)\n         violations += rendered.templater_violations\n \n", "test_patch": "diff --git a/test/core/config_test.py b/test/core/config_test.py\n--- a/test/core/config_test.py\n+++ b/test/core/config_test.py\n@@ -459,6 +459,50 @@ def test__config__validate_configs_indirect():\n         )\n \n \n+@pytest.mark.parametrize(\n+    \"raw_sql\",\n+    [\n+        (\n+            # \"types\" not \"type\"\n+            \"-- sqlfluff:layout:types:comma:line_position:leading\\n\"\n+            \"SELECT 1\"\n+        ),\n+        (\n+            # Unsupported layout config length\n+            \"-- sqlfluff:layout:foo\\n\"\n+            \"SELECT 1\"\n+        ),\n+        (\n+            # Unsupported layout config length\n+            \"-- sqlfluff:layout:type:comma:bar\\n\"\n+            \"SELECT 1\"\n+        ),\n+        (\n+            # Unsupported layout config key (\"foo\")\n+            \"-- sqlfluff:layout:type:comma:foo:bar\\n\"\n+            \"SELECT 1\"\n+        ),\n+        (\n+            # Unsupported layout config key (\"foo\") [no space]\n+            \"--sqlfluff:layout:type:comma:foo:bar\\n\"\n+            \"SELECT 1\"\n+        ),\n+    ],\n+)\n+def test__config__validate_configs_inline_layout(raw_sql):\n+    \"\"\"Test _validate_configs method of FluffConfig when used on a file.\n+\n+    This test covers both the validation of inline config\n+    directives but also the validation of layout configs.\n+    \"\"\"\n+    # Instantiate config object.\n+    cfg = FluffConfig(configs={\"core\": {\"dialect\": \"ansi\"}})\n+\n+    # Try to process an invalid inline config. Make sure we get an error.\n+    with pytest.raises(SQLFluffUserError):\n+        cfg.process_raw_file_for_config(raw_sql, \"test.sql\")\n+\n+\n def test__config__validate_configs_precedence_same_file():\n     \"\"\"Test _validate_configs method of FluffConfig where there's a conflict.\"\"\"\n     # Check with a known conflicted value\n@@ -528,19 +572,19 @@ def test__process_inline_config():\n     cfg = FluffConfig(config_b)\n     assert cfg.get(\"rules\") == \"LT03\"\n \n-    cfg.process_inline_config(\"-- sqlfluff:rules:LT02\")\n+    cfg.process_inline_config(\"-- sqlfluff:rules:LT02\", \"test.sql\")\n     assert cfg.get(\"rules\") == \"LT02\"\n \n     assert cfg.get(\"tab_space_size\", section=\"indentation\") == 4\n-    cfg.process_inline_config(\"-- sqlfluff:indentation:tab_space_size:20\")\n+    cfg.process_inline_config(\"-- sqlfluff:indentation:tab_space_size:20\", \"test.sql\")\n     assert cfg.get(\"tab_space_size\", section=\"indentation\") == 20\n \n     assert cfg.get(\"dialect\") == \"ansi\"\n     assert cfg.get(\"dialect_obj\").name == \"ansi\"\n-    cfg.process_inline_config(\"-- sqlfluff:dialect:postgres\")\n+    cfg.process_inline_config(\"-- sqlfluff:dialect:postgres\", \"test.sql\")\n     assert cfg.get(\"dialect\") == \"postgres\"\n     assert cfg.get(\"dialect_obj\").name == \"postgres\"\n \n     assert cfg.get(\"rulez\") is None\n-    cfg.process_inline_config(\"-- sqlfluff:rulez:LT06\")\n+    cfg.process_inline_config(\"-- sqlfluff:rulez:LT06\", \"test.sql\")\n     assert cfg.get(\"rulez\") == \"LT06\"\n", "problem_statement": "Validate layout configurations on load\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\nAs raised in this comment: https://github.com/sqlfluff/sqlfluff/pull/4558#discussion_r1142745101\r\n\r\nAt the moment, the layout configs are being validated _on use_ which is potentially flaky and convoluted. Better would be to validate configs _on load_.\n\n### Use case\n\n_No response_\n\n### Dialect\n\nall\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": "", "created_at": "2023-07-24T14:00:41Z", "version": "2.0", "FAIL_TO_PASS": ["test/core/config_test.py::test__config__validate_configs_inline_layout[--", "test/core/config_test.py::test__config__validate_configs_inline_layout[--sqlfluff:layout:type:comma:foo:bar\\nSELECT", "test/core/config_test.py::test__process_inline_config"], "PASS_TO_PASS": ["test/core/config_test.py::test__config__nested_combine", "test/core/config_test.py::test__config__dict_diff", "test/core/config_test.py::test__config__load_file_dir", "test/core/config_test.py::test__config__load_file_f", "test/core/config_test.py::test__config__load_nested", "test/core/config_test.py::test__config__iter_config_elems_from_dict", "test/core/config_test.py::test__config__load_toml", "test/core/config_test.py::test__config__load_placeholder_cfg", "test/core/config_test.py::test__config__iter_config_paths_right_order", "test/core/config_test.py::test__config__find_sqlfluffignore_in_same_directory", "test/core/config_test.py::test__config__nested_config_tests", "test/core/config_test.py::test__config__load_user_appdir_config", "test/core/config_test.py::test__config__split_comma_separated_string[AL01,LT08,AL07-expected0]", "test/core/config_test.py::test__config__split_comma_separated_string[\\nAL01,\\nLT08,\\nAL07,-expected1]", "test/core/config_test.py::test__config__split_comma_separated_string[raw_str2-expected2]", "test/core/config_test.py::test__config__split_comma_separated_string_correct_type", "test/core/config_test.py::test__config__templater_selection", "test/core/config_test.py::test__config__glob_exclude_config_tests", "test/core/config_test.py::test__config__glob_include_config_tests", "test/core/config_test.py::test__config__rules_set_to_none", "test/core/config_test.py::test__config__rules_group_with_exclude", "test/core/config_test.py::test__config__get_section", "test/core/config_test.py::test__config__get", "test/core/config_test.py::test__config__from_kwargs", "test/core/config_test.py::test__config_missing_dialect", "test/core/config_test.py::test__config__validate_configs_direct", "test/core/config_test.py::test__config__validate_configs_indirect", "test/core/config_test.py::test__config__validate_configs_precedence_same_file", "test/core/config_test.py::test__config__toml_list_config", "test/core/config_test.py::test__config__warn_unknown_rule"], "environment_setup_commit": "3629c3e702939c07264cc5ea903566ddc9ea2bb0"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1417", "base_commit": "da745538c7236028a22cdf0405f6829fcf6886bc", "patch": "diff --git a/astroid/decorators.py b/astroid/decorators.py\n--- a/astroid/decorators.py\n+++ b/astroid/decorators.py\n@@ -52,6 +52,8 @@ def cached(func, instance, args, kwargs):\n         return result\n \n \n+# TODO: Remove when support for 3.7 is dropped\n+# TODO: astroid 3.0 -> move class behind sys.version_info < (3, 8) guard\n class cachedproperty:\n     \"\"\"Provides a cached property equivalent to the stacking of\n     @cached and @property, but more efficient.\n@@ -70,6 +72,12 @@ class cachedproperty:\n     __slots__ = (\"wrapped\",)\n \n     def __init__(self, wrapped):\n+        if sys.version_info >= (3, 8):\n+            warnings.warn(\n+                \"cachedproperty has been deprecated and will be removed in astroid 3.0 for Python 3.8+. \"\n+                \"Use functools.cached_property instead.\",\n+                DeprecationWarning,\n+            )\n         try:\n             wrapped.__name__\n         except AttributeError as exc:\ndiff --git a/astroid/mixins.py b/astroid/mixins.py\n--- a/astroid/mixins.py\n+++ b/astroid/mixins.py\n@@ -18,6 +18,7 @@\n \"\"\"This module contains some mixins for the different nodes.\n \"\"\"\n import itertools\n+import sys\n from typing import TYPE_CHECKING, Optional\n \n from astroid import decorators\n@@ -26,11 +27,16 @@\n if TYPE_CHECKING:\n     from astroid import nodes\n \n+if sys.version_info >= (3, 8) or TYPE_CHECKING:\n+    from functools import cached_property\n+else:\n+    from astroid.decorators import cachedproperty as cached_property\n+\n \n class BlockRangeMixIn:\n     \"\"\"override block range\"\"\"\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         return self.lineno\n \n@@ -135,7 +141,7 @@ class MultiLineBlockMixin:\n     Assign nodes, etc.\n     \"\"\"\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def _multi_line_blocks(self):\n         return tuple(getattr(self, field) for field in self._multi_line_block_fields)\n \ndiff --git a/astroid/nodes/node_classes.py b/astroid/nodes/node_classes.py\n--- a/astroid/nodes/node_classes.py\n+++ b/astroid/nodes/node_classes.py\n@@ -80,6 +80,12 @@\n     from astroid import nodes\n     from astroid.nodes import LocalsDictNodeNG\n \n+if sys.version_info >= (3, 8) or TYPE_CHECKING:\n+    # pylint: disable-next=ungrouped-imports\n+    from functools import cached_property\n+else:\n+    from astroid.decorators import cachedproperty as cached_property\n+\n \n def _is_const(value):\n     return isinstance(value, tuple(CONST_CLS))\n@@ -824,7 +830,7 @@ def _infer_name(self, frame, name):\n             return name\n         return None\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def fromlineno(self):\n         \"\"\"The first line that this node appears on in the source code.\n \n@@ -833,7 +839,7 @@ def fromlineno(self):\n         lineno = super().fromlineno\n         return max(lineno, self.parent.fromlineno or 0)\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def arguments(self):\n         \"\"\"Get all the arguments for this node, including positional only and positional and keyword\"\"\"\n         return list(itertools.chain((self.posonlyargs or ()), self.args or ()))\n@@ -2601,7 +2607,7 @@ def postinit(\n         if body is not None:\n             self.body = body\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \n@@ -2734,7 +2740,7 @@ def postinit(\n     See astroid/protocols.py for actual implementation.\n     \"\"\"\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \n@@ -3093,7 +3099,7 @@ def postinit(\n         if isinstance(self.parent, If) and self in self.parent.orelse:\n             self.is_orelse = True\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \n@@ -3762,7 +3768,7 @@ def _wrap_attribute(self, attr):\n             return const\n         return attr\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def _proxied(self):\n         builtins = AstroidManager().builtins_module\n         return builtins.getattr(\"slice\")[0]\n@@ -4384,7 +4390,7 @@ def postinit(\n         if orelse is not None:\n             self.orelse = orelse\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \n@@ -4500,7 +4506,7 @@ def postinit(\n     See astroid/protocols.py for actual implementation.\n     \"\"\"\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \ndiff --git a/astroid/nodes/node_ng.py b/astroid/nodes/node_ng.py\n--- a/astroid/nodes/node_ng.py\n+++ b/astroid/nodes/node_ng.py\n@@ -38,6 +38,12 @@\n else:\n     from typing_extensions import Literal\n \n+if sys.version_info >= (3, 8) or TYPE_CHECKING:\n+    # pylint: disable-next=ungrouped-imports\n+    from functools import cached_property\n+else:\n+    # pylint: disable-next=ungrouped-imports\n+    from astroid.decorators import cachedproperty as cached_property\n \n # Types for 'NodeNG.nodes_of_class()'\n T_Nodes = TypeVar(\"T_Nodes\", bound=\"NodeNG\")\n@@ -435,14 +441,14 @@ def previous_sibling(self):\n     # these are lazy because they're relatively expensive to compute for every\n     # single node, and they rarely get looked at\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def fromlineno(self) -> Optional[int]:\n         \"\"\"The first line that this node appears on in the source code.\"\"\"\n         if self.lineno is None:\n             return self._fixed_source_line()\n         return self.lineno\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def tolineno(self) -> Optional[int]:\n         \"\"\"The last line that this node appears on in the source code.\"\"\"\n         if self.end_lineno is not None:\ndiff --git a/astroid/nodes/scoped_nodes/scoped_nodes.py b/astroid/nodes/scoped_nodes/scoped_nodes.py\n--- a/astroid/nodes/scoped_nodes/scoped_nodes.py\n+++ b/astroid/nodes/scoped_nodes/scoped_nodes.py\n@@ -52,7 +52,7 @@\n import sys\n import typing\n import warnings\n-from typing import Dict, List, Optional, Set, TypeVar, Union, overload\n+from typing import TYPE_CHECKING, Dict, List, Optional, Set, TypeVar, Union, overload\n \n from astroid import bases\n from astroid import decorators as decorators_mod\n@@ -93,6 +93,12 @@\n else:\n     from typing_extensions import Literal\n \n+if sys.version_info >= (3, 8) or TYPE_CHECKING:\n+    from functools import cached_property\n+else:\n+    # pylint: disable-next=ungrouped-imports\n+    from astroid.decorators import cachedproperty as cached_property\n+\n \n ITER_METHODS = (\"__iter__\", \"__getitem__\")\n EXCEPTION_BASE_CLASSES = frozenset({\"Exception\", \"BaseException\"})\n@@ -1611,7 +1617,7 @@ def postinit(\n         self.position = position\n         self.doc_node = doc_node\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def extra_decorators(self) -> List[node_classes.Call]:\n         \"\"\"The extra decorators that this function can have.\n \n@@ -1652,7 +1658,7 @@ def extra_decorators(self) -> List[node_classes.Call]:\n                             decorators.append(assign.value)\n         return decorators\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def type(\n         self,\n     ):  # pylint: disable=invalid-overridden-method,too-many-return-statements\n@@ -1726,7 +1732,7 @@ def type(\n                 pass\n         return type_name\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def fromlineno(self) -> Optional[int]:\n         \"\"\"The first line that this node appears on in the source code.\"\"\"\n         # lineno is the line number of the first decorator, we want the def\n@@ -1739,7 +1745,7 @@ def fromlineno(self) -> Optional[int]:\n \n         return lineno\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \n@@ -2337,7 +2343,7 @@ def _newstyle_impl(self, context=None):\n         doc=(\"Whether this is a new style class or not\\n\\n\" \":type: bool or None\"),\n     )\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def fromlineno(self) -> Optional[int]:\n         \"\"\"The first line that this node appears on in the source code.\"\"\"\n         if not PY38_PLUS:\n@@ -2352,7 +2358,7 @@ def fromlineno(self) -> Optional[int]:\n             return lineno\n         return super().fromlineno\n \n-    @decorators_mod.cachedproperty\n+    @cached_property\n     def blockstart_tolineno(self):\n         \"\"\"The line on which the beginning of this block ends.\n \ndiff --git a/astroid/objects.py b/astroid/objects.py\n--- a/astroid/objects.py\n+++ b/astroid/objects.py\n@@ -22,8 +22,10 @@\n     Call(func=Name('frozenset'), args=Tuple(...))\n \"\"\"\n \n+import sys\n+from typing import TYPE_CHECKING\n \n-from astroid import bases, decorators, util\n+from astroid import bases, util\n from astroid.exceptions import (\n     AttributeInferenceError,\n     InferenceError,\n@@ -35,6 +37,11 @@\n \n objectmodel = util.lazy_import(\"interpreter.objectmodel\")\n \n+if sys.version_info >= (3, 8) or TYPE_CHECKING:\n+    from functools import cached_property\n+else:\n+    from astroid.decorators import cachedproperty as cached_property\n+\n \n class FrozenSet(node_classes.BaseContainer):\n     \"\"\"class representing a FrozenSet composite node\"\"\"\n@@ -45,7 +52,7 @@ def pytype(self):\n     def _infer(self, context=None):\n         yield self\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def _proxied(self):  # pylint: disable=method-hidden\n         ast_builtins = AstroidManager().builtins_module\n         return ast_builtins.getattr(\"frozenset\")[0]\n@@ -114,7 +121,7 @@ def super_mro(self):\n         index = mro.index(self.mro_pointer)\n         return mro[index + 1 :]\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def _proxied(self):\n         ast_builtins = AstroidManager().builtins_module\n         return ast_builtins.getattr(\"super\")[0]\n@@ -218,7 +225,7 @@ class ExceptionInstance(bases.Instance):\n     the case of .args.\n     \"\"\"\n \n-    @decorators.cachedproperty\n+    @cached_property\n     def special_attributes(self):\n         qname = self.qname()\n         instance = objectmodel.BUILTIN_EXCEPTIONS.get(\n", "test_patch": "diff --git a/tests/unittest_decorators.py b/tests/unittest_decorators.py\n--- a/tests/unittest_decorators.py\n+++ b/tests/unittest_decorators.py\n@@ -1,7 +1,8 @@\n import pytest\n from _pytest.recwarn import WarningsRecorder\n \n-from astroid.decorators import deprecate_default_argument_values\n+from astroid.const import PY38_PLUS\n+from astroid.decorators import cachedproperty, deprecate_default_argument_values\n \n \n class SomeClass:\n@@ -97,3 +98,18 @@ def test_deprecated_default_argument_values_ok(recwarn: WarningsRecorder) -> Non\n         instance = SomeClass(name=\"some_name\")\n         instance.func(name=\"\", var=42)\n         assert len(recwarn) == 0\n+\n+\n+@pytest.mark.skipif(not PY38_PLUS, reason=\"Requires Python 3.8 or higher\")\n+def test_deprecation_warning_on_cachedproperty() -> None:\n+    \"\"\"Check the DeprecationWarning on cachedproperty.\"\"\"\n+\n+    with pytest.warns(DeprecationWarning) as records:\n+\n+        class MyClass:  # pylint: disable=unused-variable\n+            @cachedproperty\n+            def my_property(self):\n+                return 1\n+\n+        assert len(records) == 1\n+        assert \"functools.cached_property\" in records[0].message.args[0]\n", "problem_statement": "Replace `cachedproperty` with `functools.cached_property` (>= 3.8)\nI thought about this PR recently again. Typing `cachedproperty` might not work, but it can be replaced with `functools.cached_property`. We only need to `sys` guard it for `< 3.8`. This should work\r\n```py\r\nif sys.version_info >= (3, 8):\r\n    from functools import cached_property\r\nelse:\r\n    from astroid.decorators import cachedproperty as cached_property\r\n```\r\n\r\nAdditionally, the deprecation warning can be limited to `>= 3.8`.\r\n\r\n_Originally posted by @cdce8p in https://github.com/PyCQA/astroid/issues/1243#issuecomment-1052834322_\n", "hints_text": "@cdce8p Just thinking out loud: can we also use a type guard to define `cached_property`? Would `mypy` pick up on that? \n> @cdce8p Just thinking out loud: can we also use a type guard to define `cached_property`? Would `mypy` pick up on that?\r\n\r\nNot completely sure what you want to do with that.\r\n\r\nOn other thing, I just saw that we don't set the `python-version` for mypy. If we do that, we probably need to do some more workarounds to tell mypy `cachedproperty` is equal to `cached_property`. Adding `TYPE_CHECKING` could work\r\n```py\r\nif sys.version_info >= (3, 8) or TYPE_CHECKING:\r\n    from functools import cached_property\r\nelse:\r\n    from astroid.decorators import cachedproperty as cached_property\r\n```", "created_at": "2022-03-01T18:24:29Z", "version": "2.10", "FAIL_TO_PASS": ["tests/unittest_decorators.py::test_deprecation_warning_on_cachedproperty"], "PASS_TO_PASS": ["tests/unittest_decorators.py::TestDeprecationDecorators::test_deprecated_default_argument_values_one_arg", "tests/unittest_decorators.py::TestDeprecationDecorators::test_deprecated_default_argument_values_two_args", "tests/unittest_decorators.py::TestDeprecationDecorators::test_deprecated_default_argument_values_ok"], "environment_setup_commit": "da745538c7236028a22cdf0405f6829fcf6886bc"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-5206", "base_commit": "895e668a0047bd31fb87865fcab50d846bfb88ce", "patch": "diff --git a/src/sqlfluff/core/parser/segments/base.py b/src/sqlfluff/core/parser/segments/base.py\n--- a/src/sqlfluff/core/parser/segments/base.py\n+++ b/src/sqlfluff/core/parser/segments/base.py\n@@ -13,7 +13,6 @@\n import logging\n import weakref\n from collections import defaultdict\n-from copy import copy\n from dataclasses import dataclass\n from io import StringIO\n from itertools import chain\n@@ -965,25 +964,55 @@ def to_tuple(\n             )\n \n     def copy(\n-        self, segments: Optional[Tuple[\"BaseSegment\", ...]] = None\n+        self,\n+        segments: Optional[Tuple[\"BaseSegment\", ...]] = None,\n+        parent: Optional[\"BaseSegment\"] = None,\n     ) -> \"BaseSegment\":\n         \"\"\"Copy the segment recursively, with appropriate copying of references.\n \n         Optionally provide child segments which have already been dealt\n         with to avoid another copy operation.\n+\n+        NOTE: In the copy operation it's really important that we get\n+        a clean segregation so that we can't go backward and mutate the\n+        source object, but at the same time we should be mindful of what\n+        _needs_ to be copied to avoid a deep copy where one isn't required.\n         \"\"\"\n-        new_seg = copy(self)\n+        cls = self.__class__\n+        new_segment = cls.__new__(cls)\n         # Position markers are immutable, and it's important that we keep\n         # a reference to the same TemplatedFile, so keep the same position\n-        # marker.\n-        new_seg.pos_marker = self.pos_marker\n+        # marker. By updating from the source dict, we achieve that.\n+        # By using the __dict__ object we also transfer the _cache_ too\n+        # which is stored there by @cached_property.\n+        new_segment.__dict__.update(self.__dict__)\n+\n+        # Reset the parent if provided.\n+        if parent:\n+            new_segment.set_parent(parent)\n+\n+        # If the segment doesn't have a segments property, we're done.\n+        # NOTE: This is a proxy way of understanding whether it's a RawSegment\n+        # of not. Typically will _have_ a `segments` attribute, but it's an\n+        # empty tuple.\n+        if not self.__dict__.get(\"segments\", None):\n+            assert (\n+                not segments\n+            ), f\"Cannot provide `segments` argument to {cls.__name__} `.copy()`\\n\"\n         # If segments were provided, use them.\n-        if segments:\n-            new_seg.segments = segments\n-        # Otherwise copy them.\n-        elif self.segments:\n-            new_seg.segments = tuple(seg.copy() for seg in self.segments)\n-        return new_seg\n+        elif segments:\n+            new_segment.segments = segments\n+        # Otherwise we should handle recursive segment coping.\n+        # We use the native .copy() method (this method!) appropriately\n+        # so that the same logic is applied in recursion.\n+        # We set the parent for children directly on the copy method\n+        # to ensure those line up properly.\n+        else:\n+            new_segment.segments = tuple(\n+                seg.copy(parent=new_segment) for seg in self.segments\n+            )\n+\n+        return new_segment\n \n     def as_record(self, **kwargs: bool) -> Optional[RecordSerialisedSegment]:\n         \"\"\"Return the segment as a structurally simplified record.\n@@ -1400,7 +1429,6 @@ def apply_fixes(\n                                 # of a create_before/create_after pair, also add\n                                 # this segment before the edit.\n                                 seg_buffer.append(seg)\n-                                seg.set_parent(self)\n \n                             # We're doing a replacement (it could be a single\n                             # segment or an iterable)\n@@ -1408,7 +1436,6 @@ def apply_fixes(\n                             consumed_pos = False\n                             for s in f.edit:\n                                 seg_buffer.append(s)\n-                                s.set_parent(self)\n                                 # If one of them has the same raw representation\n                                 # then the first that matches gets to take the\n                                 # original position marker.\n@@ -1424,7 +1451,6 @@ def apply_fixes(\n                                 # in the case of a creation before, also add this\n                                 # segment on the end\n                                 seg_buffer.append(seg)\n-                                seg.set_parent(self)\n \n                         else:  # pragma: no cover\n                             raise ValueError(\n@@ -1434,7 +1460,7 @@ def apply_fixes(\n                             )\n                 else:\n                     seg_buffer.append(seg)\n-                    seg.set_parent(self)\n+\n             # Invalidate any caches\n             self.invalidate_caches()\n \n@@ -1495,6 +1521,7 @@ def apply_fixes(\n             # Pass through any additional kwargs\n             **{k: getattr(self, k) for k in self.additional_kwargs},\n         )\n+        new_seg.set_as_parent(recurse=False)\n         # Only validate if there's a match_grammar. Otherwise we may get\n         # strange results (for example with the BracketedSegment).\n         if requires_validate and (\ndiff --git a/src/sqlfluff/core/rules/base.py b/src/sqlfluff/core/rules/base.py\n--- a/src/sqlfluff/core/rules/base.py\n+++ b/src/sqlfluff/core/rules/base.py\n@@ -1143,7 +1143,7 @@ def _choose_anchor_segment(\n             if root_segment\n             else None\n         )\n-        assert path\n+        assert path, f\"No path found from {root_segment} to {segment}!\"\n         for seg in path[::-1]:\n             # If the segment allows non code ends, then no problem.\n             # We're done. This is usually the outer file segment.\n", "test_patch": "diff --git a/test/core/parser/segments/segments_base_test.py b/test/core/parser/segments/segments_base_test.py\n--- a/test/core/parser/segments/segments_base_test.py\n+++ b/test/core/parser/segments/segments_base_test.py\n@@ -302,3 +302,58 @@ def test__parser__base_segments_pickle_safe(raw_seg_list):\n     assert test_seg == result_seg\n     # Check specifically the treatment of the parent position.\n     assert result_seg.segments[0].get_parent() is result_seg\n+\n+\n+def test__parser__base_segments_copy_isolation(DummySegment, raw_seg_list):\n+    \"\"\"Test copy isolation in BaseSegment.\n+\n+    First on one of the raws and then on the dummy segment.\n+    \"\"\"\n+    # On a raw\n+    a_seg = raw_seg_list[0]\n+    a_copy = a_seg.copy()\n+    assert a_seg is not a_copy\n+    assert a_seg == a_copy\n+    assert a_seg.pos_marker is a_copy.pos_marker\n+    a_copy.pos_marker = None\n+    assert a_copy.pos_marker is None\n+    assert a_seg.pos_marker is not None\n+\n+    # On a base\n+    b_seg = DummySegment(segments=raw_seg_list)\n+    b_copy = b_seg.copy()\n+    assert b_seg is not b_copy\n+    assert b_seg == b_copy\n+    assert b_seg.pos_marker is b_copy.pos_marker\n+    b_copy.pos_marker = None\n+    assert b_copy.pos_marker is None\n+    assert b_seg.pos_marker is not None\n+\n+    # On addition to a lint Fix\n+    fix = LintFix(\"replace\", a_seg, [b_seg])\n+    for s in fix.edit:\n+        assert not s.pos_marker\n+    assert b_seg.pos_marker\n+\n+\n+def test__parser__base_segments_parent_ref(DummySegment, raw_seg_list):\n+    \"\"\"Test getting and setting parents on BaseSegment.\"\"\"\n+    # Check initially no parent (because not set)\n+    assert not raw_seg_list[0].get_parent()\n+    # Add it to a segment (still not set)\n+    seg = DummySegment(segments=raw_seg_list)\n+    assert not seg.segments[0].get_parent()\n+    # Set one parent on one of them (but not another)\n+    seg.segments[0].set_parent(seg)\n+    assert seg.segments[0].get_parent() is seg\n+    assert not seg.segments[1].get_parent()\n+    # Set parent on all of them\n+    seg.set_as_parent()\n+    assert seg.segments[0].get_parent() is seg\n+    assert seg.segments[1].get_parent() is seg\n+    # Remove segment from parent, but don't unset.\n+    # Should still check an return None.\n+    seg_0 = seg.segments[0]\n+    seg.segments = seg.segments[1:]\n+    assert seg_0 not in seg.segments\n+    assert not seg_0.get_parent()\ndiff --git a/test/rules/std_RF01_LT09_test.py b/test/rules/std_RF01_LT09_test.py\nnew file mode 100644\n--- /dev/null\n+++ b/test/rules/std_RF01_LT09_test.py\n@@ -0,0 +1,27 @@\n+\"\"\"Tests observed conflict between RF01 & LT09.\n+\n+Root cause was BaseSegment.copy().\n+\"\"\"\n+from sqlfluff.core import FluffConfig, Linter\n+\n+\n+def test__rules__std_RF01_LT09_copy() -> None:\n+    \"\"\"Tests observed conflict between RF01 & LT09.\n+\n+    https://github.com/sqlfluff/sqlfluff/issues/5203\n+    \"\"\"\n+    sql = \"\"\"\n+SELECT\n+    DISTINCT `FIELD`\n+FROM `TABLE`;\n+\"\"\"\n+    cfg = FluffConfig.from_kwargs(\n+        dialect=\"mysql\",\n+        rules=[\"RF01\", \"LT09\"],\n+    )\n+    result = Linter(config=cfg).lint_string(sql)\n+    for violation in result.violations:\n+        assert \"Unexpected exception\" not in violation.description\n+    assert len(result.violations) == 1\n+    only_violation = result.violations[0]\n+    assert only_violation.rule_code() == \"LT09\"\n", "problem_statement": "Exception thrown when SELECT DISTINCT not on the same line\n### Search before asking\r\n\r\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\r\n\r\n\r\n### What Happened\r\n\r\nCheck a file containing this request:\r\n\r\n```sql\r\nSELECT\r\n    DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\nIt fails this way:\r\n\r\n```log\r\nCRITICAL   [RF01] Applying rule RF01 to 'file.sql' threw an Exception:  \r\nTraceback (most recent call last):\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/rules/base.py\", line 864, in crawl\r\n    res = self._eval(context=context)\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 107, in _eval\r\n    self._analyze_table_references(\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 152, in _analyze_table_references\r\n    if not self._should_ignore_reference(r, selectable):\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 168, in _should_ignore_reference\r\n    ref_path = selectable.selectable.path_to(reference)\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/parser/segments/base.py\", line 1184, in path_to\r\n    elif not self.get_start_loc() <= midpoint.get_start_loc() <= self.get_end_loc():\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/parser/segments/base.py\", line 877, in get_start_loc\r\n    assert self.pos_marker\r\nAssertionError\r\n== [file.sql] FAIL\r\nL:   1 | P:   1 | LT09 | Select targets should be on a new line unless there is\r\n                       | only one select target. [layout.select_targets]\r\nL:   1 | P:   1 | LT10 | 'SELECT' modifiers (e.g. 'DISTINCT') must be on the same\r\n                       | line as 'SELECT'. [layout.select_modifiers]\r\nL:   1 | P:   1 | RF01 | Unexpected exception: ;\r\nCould you open an issue at\r\n                       | https://github.com/sqlfluff/sqlfluff/issues ?\r\nYou can\r\n                       | ignore this exception for now, by adding '-- noqa: RF01'\r\n                       | at the end\r\nof line 1\r\n [references.from]\r\nL:   2 | P:   1 | LT02 | Line should not be indented. [layout.indent]\r\nL:   3 | P:  13 | LT12 | Files must end with a single trailing newline.\r\n                       | [layout.end_of_file]\r\nAll Finished!\r\n```\r\n\r\nChecking the following request does not throw an exception (move `DISTINCT` on same line than `SELECT`):\r\n\r\n```sql\r\nSELECT DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\nAdditionally, I'd like to add that checking the first request on https://online.sqlfluff.com/fluffed leads to the same exception. But if you check this request:\r\n```sql\r\nSELECT \r\nDISTINCT\r\n`FIELD`\r\nFROM `TABLE`;\r\n```\r\nThen the website crashes.\r\n\r\n### Expected Behaviour\r\n\r\nI would expect not to have an exception.\r\n\r\n### Observed Behaviour\r\n\r\nAn exception was thrown whereas, I think, there is no reason to throw it.\r\n\r\n### How to reproduce\r\n\r\nCheck the following SQL:\r\n\r\n```sql\r\nSELECT\r\n    DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\n### Dialect\r\n\r\nMySQL\r\n\r\n### Version\r\n\r\n2.3.2\r\n\r\n### Configuration\r\n\r\n```\r\n[sqlfluff]\r\n# Supported dialects https://docs.sqlfluff.com/en/stable/dialects.html\r\ndialect = mysql\r\nencoding = utf-8\r\n# Exclude rule LT01/layout.spacing: it expects a space even after type of fields (i.e. \"INT (11)\")\r\n# Exclude rule ST05/structure.subquery: MySQL badly supports CTEs.\r\nexclude_rules = LT01, ST05\r\nignore = parsing\r\nmax_line_length = 120\r\n# Below controls SQLFluff output, see max_line_length for SQL output\r\noutput_line_length = 80\r\ntemplater = raw\r\nverbose = 0\r\n\r\n[sqlfluff:layout:type:binary_operator]\r\nline_position = leading\r\n\r\n[sqlfluff:layout:type:comma]\r\nline_position = trailing\r\nspacing_before = touch\r\n\r\n[sqlfluff:indentation]\r\n# See https://docs.sqlfluff.com/en/stable/indentation.html\r\nindent_unit = space\r\nindented_joins = True\r\nindented_using_on = True\r\ntab_space_size = 4\r\n\r\n# Some rules can be configured directly from the config common to other rules\r\n[sqlfluff:rules]\r\nallow_scalar = True\r\nquoted_identifiers_policy = none\r\nsingle_table_references = consistent\r\nunquoted_identifiers_policy = all\r\n\r\n[sqlfluff:rules:aliasing.column]\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:aliasing.table]\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:ambiguous.column_references]\r\ngroup_by_and_order_by_style = consistent\r\n\r\n[sqlfluff:rules:capitalisation.functions]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.identifiers]\r\nextended_capitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.keywords]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.literals]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.types]\r\nextended_capitalisation_policy = upper\r\n\r\n[sqlfluff:rules:convention.count_rows]\r\nprefer_count_0 = False\r\nprefer_count_1 = True\r\n\r\n[sqlfluff:rules:convention.select_trailing_comma]\r\nselect_clause_trailing_comma = forbid\r\n\r\n[sqlfluff:rules:convention.terminator]\r\nmultiline_newline = False\r\nrequire_final_semicolon = True\r\n\r\n[sqlfluff:rules:layout.long_lines]\r\nignore_comment_lines = True\r\n\r\n[sqlfluff:rules:references.keywords]\r\nignore_words = None\r\nquoted_identifiers_policy = none\r\nunquoted_identifiers_policy = all\r\n\r\n[sqlfluff:rules:convention.quoted_literals]\r\npreferred_quoted_literal_style = single_quotes\r\n\r\n[sqlfluff:rules:references.quoting]\r\nprefer_quoted_identifiers = True\r\n\r\n[sqlfluff:rules:references.special_chars]\r\nadditional_allowed_characters = \"\"\r\nallow_space_in_identifier = False\r\nquoted_identifiers_policy = all\r\n# Special characters in identifiers\r\nunquoted_identifiers_policy = all\r\n```\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [X] Yes I am willing to submit a PR!\r\n\r\n### Code of Conduct\r\n\r\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\r\n\nException thrown when SELECT DISTINCT not on the same line\n### Search before asking\r\n\r\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\r\n\r\n\r\n### What Happened\r\n\r\nCheck a file containing this request:\r\n\r\n```sql\r\nSELECT\r\n    DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\nIt fails this way:\r\n\r\n```log\r\nCRITICAL   [RF01] Applying rule RF01 to 'file.sql' threw an Exception:  \r\nTraceback (most recent call last):\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/rules/base.py\", line 864, in crawl\r\n    res = self._eval(context=context)\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 107, in _eval\r\n    self._analyze_table_references(\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 152, in _analyze_table_references\r\n    if not self._should_ignore_reference(r, selectable):\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/rules/references/RF01.py\", line 168, in _should_ignore_reference\r\n    ref_path = selectable.selectable.path_to(reference)\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/parser/segments/base.py\", line 1184, in path_to\r\n    elif not self.get_start_loc() <= midpoint.get_start_loc() <= self.get_end_loc():\r\n  File \"/app/.venv/lib/python3.9/site-packages/sqlfluff/core/parser/segments/base.py\", line 877, in get_start_loc\r\n    assert self.pos_marker\r\nAssertionError\r\n== [file.sql] FAIL\r\nL:   1 | P:   1 | LT09 | Select targets should be on a new line unless there is\r\n                       | only one select target. [layout.select_targets]\r\nL:   1 | P:   1 | LT10 | 'SELECT' modifiers (e.g. 'DISTINCT') must be on the same\r\n                       | line as 'SELECT'. [layout.select_modifiers]\r\nL:   1 | P:   1 | RF01 | Unexpected exception: ;\r\nCould you open an issue at\r\n                       | https://github.com/sqlfluff/sqlfluff/issues ?\r\nYou can\r\n                       | ignore this exception for now, by adding '-- noqa: RF01'\r\n                       | at the end\r\nof line 1\r\n [references.from]\r\nL:   2 | P:   1 | LT02 | Line should not be indented. [layout.indent]\r\nL:   3 | P:  13 | LT12 | Files must end with a single trailing newline.\r\n                       | [layout.end_of_file]\r\nAll Finished!\r\n```\r\n\r\nChecking the following request does not throw an exception (move `DISTINCT` on same line than `SELECT`):\r\n\r\n```sql\r\nSELECT DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\nAdditionally, I'd like to add that checking the first request on https://online.sqlfluff.com/fluffed leads to the same exception. But if you check this request:\r\n```sql\r\nSELECT \r\nDISTINCT\r\n`FIELD`\r\nFROM `TABLE`;\r\n```\r\nThen the website crashes.\r\n\r\n### Expected Behaviour\r\n\r\nI would expect not to have an exception.\r\n\r\n### Observed Behaviour\r\n\r\nAn exception was thrown whereas, I think, there is no reason to throw it.\r\n\r\n### How to reproduce\r\n\r\nCheck the following SQL:\r\n\r\n```sql\r\nSELECT\r\n    DISTINCT `FIELD`\r\nFROM `TABLE`;\r\n```\r\n\r\n### Dialect\r\n\r\nMySQL\r\n\r\n### Version\r\n\r\n2.3.2\r\n\r\n### Configuration\r\n\r\n```\r\n[sqlfluff]\r\n# Supported dialects https://docs.sqlfluff.com/en/stable/dialects.html\r\ndialect = mysql\r\nencoding = utf-8\r\n# Exclude rule LT01/layout.spacing: it expects a space even after type of fields (i.e. \"INT (11)\")\r\n# Exclude rule ST05/structure.subquery: MySQL badly supports CTEs.\r\nexclude_rules = LT01, ST05\r\nignore = parsing\r\nmax_line_length = 120\r\n# Below controls SQLFluff output, see max_line_length for SQL output\r\noutput_line_length = 80\r\ntemplater = raw\r\nverbose = 0\r\n\r\n[sqlfluff:layout:type:binary_operator]\r\nline_position = leading\r\n\r\n[sqlfluff:layout:type:comma]\r\nline_position = trailing\r\nspacing_before = touch\r\n\r\n[sqlfluff:indentation]\r\n# See https://docs.sqlfluff.com/en/stable/indentation.html\r\nindent_unit = space\r\nindented_joins = True\r\nindented_using_on = True\r\ntab_space_size = 4\r\n\r\n# Some rules can be configured directly from the config common to other rules\r\n[sqlfluff:rules]\r\nallow_scalar = True\r\nquoted_identifiers_policy = none\r\nsingle_table_references = consistent\r\nunquoted_identifiers_policy = all\r\n\r\n[sqlfluff:rules:aliasing.column]\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:aliasing.table]\r\naliasing = explicit\r\n\r\n[sqlfluff:rules:ambiguous.column_references]\r\ngroup_by_and_order_by_style = consistent\r\n\r\n[sqlfluff:rules:capitalisation.functions]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.identifiers]\r\nextended_capitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.keywords]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.literals]\r\ncapitalisation_policy = upper\r\nignore_words = None\r\n\r\n[sqlfluff:rules:capitalisation.types]\r\nextended_capitalisation_policy = upper\r\n\r\n[sqlfluff:rules:convention.count_rows]\r\nprefer_count_0 = False\r\nprefer_count_1 = True\r\n\r\n[sqlfluff:rules:convention.select_trailing_comma]\r\nselect_clause_trailing_comma = forbid\r\n\r\n[sqlfluff:rules:convention.terminator]\r\nmultiline_newline = False\r\nrequire_final_semicolon = True\r\n\r\n[sqlfluff:rules:layout.long_lines]\r\nignore_comment_lines = True\r\n\r\n[sqlfluff:rules:references.keywords]\r\nignore_words = None\r\nquoted_identifiers_policy = none\r\nunquoted_identifiers_policy = all\r\n\r\n[sqlfluff:rules:convention.quoted_literals]\r\npreferred_quoted_literal_style = single_quotes\r\n\r\n[sqlfluff:rules:references.quoting]\r\nprefer_quoted_identifiers = True\r\n\r\n[sqlfluff:rules:references.special_chars]\r\nadditional_allowed_characters = \"\"\r\nallow_space_in_identifier = False\r\nquoted_identifiers_policy = all\r\n# Special characters in identifiers\r\nunquoted_identifiers_policy = all\r\n```\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [X] Yes I am willing to submit a PR!\r\n\r\n### Code of Conduct\r\n\r\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\r\n\n", "hints_text": "\n", "created_at": "2023-09-13T18:57:24Z", "version": "2.2", "FAIL_TO_PASS": ["test/rules/std_RF01_LT09_test.py::test__rules__std_RF01_LT09_copy"], "PASS_TO_PASS": ["test/core/parser/segments/segments_base_test.py::test__parser__base_segments_type", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_class_types", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_descendant_type_set", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_direct_descendant_type_set", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_to_tuple_a", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_to_tuple_b", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_to_tuple_c", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_as_record_a", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_as_record_b", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_as_record_c", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_count_segments", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in0-None]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in1--1]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in2-0]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in3-0]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in4-0]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_find_start_or_end_non_code[list_in5-None]", "test/core/parser/segments/segments_base_test.py::test__parser_base_segments_compute_anchor_edit_info", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_path_to", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_stubs", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_raw", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_base", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_raw_compare", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_base_compare", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_pickle_safe", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_copy_isolation", "test/core/parser/segments/segments_base_test.py::test__parser__base_segments_parent_ref"], "environment_setup_commit": "3625934f16857ade527f5f7dfa84b874061ea739"}, {"repo": "marshmallow-code/marshmallow", "instance_id": "marshmallow-code__marshmallow-1524", "base_commit": "7015fc4333a2f32cd58c3465296e834acd4496ff", "patch": "diff --git a/src/marshmallow/validate.py b/src/marshmallow/validate.py\n--- a/src/marshmallow/validate.py\n+++ b/src/marshmallow/validate.py\n@@ -65,7 +65,7 @@ def _regex_generator(self, relative, require_tld):\n                 r'\\[?[A-F0-9]*:[A-F0-9:]+\\]?)',  # ...or ipv6\n                 r'(?::\\d+)?',  # optional port\n                 r')?' if relative else r'',  # host is optional, allow for relative URLs\n-                r'(?:/?|[/?]\\S+)$',\n+                r'(?:/?|[/?]\\S+)\\Z',\n             )), re.IGNORECASE)\n \n         def __call__(self, relative, require_tld):\n@@ -120,18 +120,18 @@ class Email(Validator):\n     \"\"\"\n \n     USER_REGEX = re.compile(\n-        r\"(^[-!#$%&'*+/=?^`{}|~\\w]+(\\.[-!#$%&'*+/=?^`{}|~\\w]+)*$\"  # dot-atom\n+        r\"(^[-!#$%&'*+/=?^`{}|~\\w]+(\\.[-!#$%&'*+/=?^`{}|~\\w]+)*\\Z\"  # dot-atom\n         # quoted-string\n         r'|^\"([\\001-\\010\\013\\014\\016-\\037!#-\\[\\]-\\177]'\n-        r'|\\\\[\\001-\\011\\013\\014\\016-\\177])*\"$)', re.IGNORECASE | re.UNICODE)\n+        r'|\\\\[\\001-\\011\\013\\014\\016-\\177])*\"\\Z)', re.IGNORECASE | re.UNICODE)\n \n     DOMAIN_REGEX = re.compile(\n         # domain\n         r'(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\\.)+'\n-        r'(?:[A-Z]{2,6}|[A-Z0-9-]{2,})$'\n+        r'(?:[A-Z]{2,6}|[A-Z0-9-]{2,})\\Z'\n         # literal form, ipv4 address (SMTP 4.1.3)\n         r'|^\\[(25[0-5]|2[0-4]\\d|[0-1]?\\d?\\d)'\n-        r'(\\.(25[0-5]|2[0-4]\\d|[0-1]?\\d?\\d)){3}\\]$', re.IGNORECASE | re.UNICODE)\n+        r'(\\.(25[0-5]|2[0-4]\\d|[0-1]?\\d?\\d)){3}\\]\\Z', re.IGNORECASE | re.UNICODE)\n \n     DOMAIN_WHITELIST = ('localhost',)\n \n", "test_patch": "diff --git a/tests/test_validate.py b/tests/test_validate.py\n--- a/tests/test_validate.py\n+++ b/tests/test_validate.py\n@@ -41,6 +41,7 @@ def test_url_absolute_valid(valid_url):\n     'http:/example.org',\n     'foo://example.org',\n     '../icons/logo.gif',\n+    'https://example.org\\n',\n     'abc',\n     '..',\n     '/',\n@@ -71,6 +72,7 @@ def test_url_relative_valid(valid_url):\n     'suppliers.html',\n     '../icons/logo.gif',\n     'icons/logo.gif',\n+    'http://example.org\\n',\n     '../.../g',\n     '...',\n     '\\\\',\n@@ -98,6 +100,7 @@ def test_url_dont_require_tld_valid(valid_url):\n \n @pytest.mark.parametrize('invalid_url', [\n     'http//example',\n+    'http://example\\n',\n     'http://.example.org',\n     'http:///foo/bar',\n     'http:// /foo/bar',\n@@ -165,6 +168,8 @@ def test_email_valid(valid_email):\n     assert validator(valid_email) == valid_email\n \n @pytest.mark.parametrize('invalid_email', [\n+    'niceandsimple\\n@example.com',\n+    'NiCeAnDsImPlE@eXaMpLe.CoM\\n',\n     'a\"b(c)d,e:f;g<h>i[j\\\\k]l@example.com',\n     'just\"not\"right@example.com',\n     'this is\"not\\allowed@example.com',\n", "problem_statement": "Incorrect Email Validation\nhttps://github.com/marshmallow-code/marshmallow/blob/fbe22eb47db5df64b2c4133f9a5cb6c6920e8dd2/src/marshmallow/validate.py#L136-L151\r\n\r\nThe email validation regex will match `email@domain.com\\n`, `email\\n@domain.com`, and `email\\n@domain.com\\n`.\r\n\r\nThe issue is that `$` is used to match until the end of a string. Instead, `\\Z` should be used. - https://stackoverflow.com/a/48730645\r\n\r\nIt is possible that other validators might suffer from the same bug, so it would be good if other regexes were also checked.\r\n\r\nIt is unclear, but this may lead to a security vulnerability in some projects that use marshmallow (depending on how the validator is used), so a quick fix here might be helpful. In my quick look around I didn't notice anything critical, however, so I figured it would be fine to open this issue.\n", "hints_text": "`^` behaves as expected, but `$` also matches a \"string-ending newline\". This appears to be a holdover from POSIX that most languages have long abandoned as default behavior.\r\n\r\nThanks for reporting this @nbanmp. If you are interesting in contributing a MR, I would be happy to review it, otherwise I can pick this up as soon as I get a chance.\r\n\r\nAlso, can you provide any examples of using an email address in a way that would make this issue a security vulnerability? The best I can come up with is that it will terminate the email headers early and cause an email to be malformed or undeliverable, which would be a standard bug. Since the newline can only be at the very end of the string, this would not allow injecting additional headers into an email. If it does have security implications I can submit a CVE and flag the vulnerable versions on TideLift.\nI don't know if I would call this a vulnerability in marshmallow, but it might - in rare cases - allow for exploitation of certain vulnerabilities in other tools.\r\n\r\nThe most likely place this might result in a vulnerability is when verifying an email for authentication, and allowing multiple accounts with the same, but different emails.\r\n\r\nA minor form of this would be allowing the same email to be used multiple times in sites that want to prevent that.\r\n\r\nA more serious, but unlikely form of this would be something like:\r\n\r\nDepending on how emails are stored / used, whitespace around them might be stripped or not between sending emails, and checking emails. It's not a vulnerability in marshmallow itself, but it might result in inconsistencies in other apps lead to actual vulnerabilities.\r\n\r\n```python\r\nvalidateemail(email)\r\n\r\nif email in database:\r\n    error()\r\nelse:\r\n    saveaccount(email.strip(), password) # This might overwrite an account with the correct email\r\n```\r\n\r\nThis example is actually what I was thinking of, because I was looking through the code for CTFd, which had a similar vulnerability occurring with the username, and, if not for dumb luck (a random lack of .strip() / it being in a different place), the same vulnerability would have also been possible using the email, even though the email was invalid.\r\n", "created_at": "2020-02-26T20:55:30Z", "version": "2.20", "FAIL_TO_PASS": ["tests/test_validate.py::test_url_absolute_invalid[https://example.org\\n]", "tests/test_validate.py::test_url_relative_invalid[http://example.org\\n]", "tests/test_validate.py::test_url_dont_require_tld_invalid[http://example\\n]", "tests/test_validate.py::test_email_invalid[niceandsimple\\n@example.com]", "tests/test_validate.py::test_email_invalid[NiCeAnDsImPlE@eXaMpLe.CoM\\n]"], "PASS_TO_PASS": ["tests/test_validate.py::test_url_absolute_valid[http://example.org]", "tests/test_validate.py::test_url_absolute_valid[https://example.org]", "tests/test_validate.py::test_url_absolute_valid[ftp://example.org]", "tests/test_validate.py::test_url_absolute_valid[ftps://example.org]", "tests/test_validate.py::test_url_absolute_valid[http://example.co.jp]", "tests/test_validate.py::test_url_absolute_valid[http://www.example.com/a%C2%B1b]", "tests/test_validate.py::test_url_absolute_valid[http://www.example.com/~username/]", "tests/test_validate.py::test_url_absolute_valid[http://info.example.com/?fred]", "tests/test_validate.py::test_url_absolute_valid[http://xn--mgbh0fb.xn--kgbechtv/]", "tests/test_validate.py::test_url_absolute_valid[http://example.com/blue/red%3Fand+green]", "tests/test_validate.py::test_url_absolute_valid[http://www.example.com/?array%5Bkey%5D=value]", "tests/test_validate.py::test_url_absolute_valid[http://xn--rsum-bpad.example.org/]", "tests/test_validate.py::test_url_absolute_valid[http://123.45.67.8/]", "tests/test_validate.py::test_url_absolute_valid[http://2001:db8::ff00:42:8329]", "tests/test_validate.py::test_url_absolute_valid[http://www.example.com:8000/foo]", "tests/test_validate.py::test_url_absolute_valid[http://user@example.com]", "tests/test_validate.py::test_url_absolute_valid[http://user:pass@example.com]", "tests/test_validate.py::test_url_absolute_invalid[http:///example.com/]", "tests/test_validate.py::test_url_absolute_invalid[https:///example.com/]", "tests/test_validate.py::test_url_absolute_invalid[https://example.org\\\\]", "tests/test_validate.py::test_url_absolute_invalid[ftp:///example.com/]", "tests/test_validate.py::test_url_absolute_invalid[ftps:///example.com/]", "tests/test_validate.py::test_url_absolute_invalid[http//example.org]", "tests/test_validate.py::test_url_absolute_invalid[http:///]", "tests/test_validate.py::test_url_absolute_invalid[http:/example.org]", "tests/test_validate.py::test_url_absolute_invalid[foo://example.org]", "tests/test_validate.py::test_url_absolute_invalid[../icons/logo.gif]", "tests/test_validate.py::test_url_absolute_invalid[abc]", "tests/test_validate.py::test_url_absolute_invalid[..]", "tests/test_validate.py::test_url_absolute_invalid[/]", "tests/test_validate.py::test_url_absolute_invalid[", "tests/test_validate.py::test_url_absolute_invalid[]", "tests/test_validate.py::test_url_absolute_invalid[None]", "tests/test_validate.py::test_url_relative_valid[http://example.org]", "tests/test_validate.py::test_url_relative_valid[http://123.45.67.8/]", "tests/test_validate.py::test_url_relative_valid[http://example.com/foo/bar/../baz]", "tests/test_validate.py::test_url_relative_valid[https://example.com/../icons/logo.gif]", "tests/test_validate.py::test_url_relative_valid[http://example.com/./icons/logo.gif]", "tests/test_validate.py::test_url_relative_valid[ftp://example.com/../../../../g]", "tests/test_validate.py::test_url_relative_valid[http://example.com/g?y/./x]", "tests/test_validate.py::test_url_relative_invalid[http//example.org]", "tests/test_validate.py::test_url_relative_invalid[suppliers.html]", "tests/test_validate.py::test_url_relative_invalid[../icons/logo.gif]", "tests/test_validate.py::test_url_relative_invalid[icons/logo.gif]", "tests/test_validate.py::test_url_relative_invalid[../.../g]", "tests/test_validate.py::test_url_relative_invalid[...]", "tests/test_validate.py::test_url_relative_invalid[\\\\]", "tests/test_validate.py::test_url_relative_invalid[", "tests/test_validate.py::test_url_relative_invalid[]", "tests/test_validate.py::test_url_relative_invalid[None]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://example.org]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://123.45.67.8/]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://example]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://example.]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://example:80]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://user.name:pass.word@example]", "tests/test_validate.py::test_url_dont_require_tld_valid[http://example/foo/bar]", "tests/test_validate.py::test_url_dont_require_tld_invalid[http//example]", "tests/test_validate.py::test_url_dont_require_tld_invalid[http://.example.org]", "tests/test_validate.py::test_url_dont_require_tld_invalid[http:///foo/bar]", "tests/test_validate.py::test_url_dont_require_tld_invalid[http://", "tests/test_validate.py::test_url_dont_require_tld_invalid[]", "tests/test_validate.py::test_url_dont_require_tld_invalid[None]", "tests/test_validate.py::test_url_custom_scheme", "tests/test_validate.py::test_url_relative_and_custom_schemes", "tests/test_validate.py::test_url_custom_message", "tests/test_validate.py::test_url_repr", "tests/test_validate.py::test_email_valid[niceandsimple@example.com]", "tests/test_validate.py::test_email_valid[NiCeAnDsImPlE@eXaMpLe.CoM]", "tests/test_validate.py::test_email_valid[very.common@example.com]", "tests/test_validate.py::test_email_valid[a.little.lengthy.but.fine@a.iana-servers.net]", "tests/test_validate.py::test_email_valid[disposable.style.email.with+symbol@example.com]", "tests/test_validate.py::test_email_valid[\"very.unusual.@.unusual.com\"@example.com]", "tests/test_validate.py::test_email_valid[!#$%&'*+-/=?^_`{}|~@example.org]", "tests/test_validate.py::test_email_valid[niceandsimple@[64.233.160.0]]", "tests/test_validate.py::test_email_valid[niceandsimple@localhost]", "tests/test_validate.py::test_email_valid[jos\\xe9@blah.com]", "tests/test_validate.py::test_email_valid[\\u03b4\\u03bf\\u03ba.\\u03b9\\u03bc\\u03ae@\\u03c0\\u03b1\\u03c1\\u03ac\\u03b4\\u03b5\\u03b9\\u03b3\\u03bc\\u03b1.\\u03b4\\u03bf\\u03ba\\u03b9\\u03bc\\u03ae]", "tests/test_validate.py::test_email_invalid[a\"b(c)d,e:f;g<h>i[j\\\\k]l@example.com]", "tests/test_validate.py::test_email_invalid[just\"not\"right@example.com]", "tests/test_validate.py::test_email_invalid[this", "tests/test_validate.py::test_email_invalid[this\\\\", "tests/test_validate.py::test_email_invalid[\"much.more", "tests/test_validate.py::test_email_invalid[\"very.(),:;<>[]\".VERY.\"very@\\\\", "tests/test_validate.py::test_email_invalid[\"", "tests/test_validate.py::test_email_invalid[user@example]", "tests/test_validate.py::test_email_invalid[@nouser.com]", "tests/test_validate.py::test_email_invalid[example.com]", "tests/test_validate.py::test_email_invalid[user]", "tests/test_validate.py::test_email_invalid[]", "tests/test_validate.py::test_email_invalid[None]", "tests/test_validate.py::test_email_custom_message", "tests/test_validate.py::test_email_repr", "tests/test_validate.py::test_range_min", "tests/test_validate.py::test_range_max", "tests/test_validate.py::test_range_custom_message", "tests/test_validate.py::test_range_repr", "tests/test_validate.py::test_length_min", "tests/test_validate.py::test_length_max", "tests/test_validate.py::test_length_equal", "tests/test_validate.py::test_length_custom_message", "tests/test_validate.py::test_length_repr", "tests/test_validate.py::test_equal", "tests/test_validate.py::test_equal_custom_message", "tests/test_validate.py::test_equal_repr", "tests/test_validate.py::test_regexp_str", "tests/test_validate.py::test_regexp_compile", "tests/test_validate.py::test_regexp_custom_message", "tests/test_validate.py::test_regexp_repr", "tests/test_validate.py::test_predicate", "tests/test_validate.py::test_predicate_custom_message", "tests/test_validate.py::test_predicate_repr", "tests/test_validate.py::test_noneof", "tests/test_validate.py::test_noneof_custom_message", "tests/test_validate.py::test_noneof_repr", "tests/test_validate.py::test_oneof", "tests/test_validate.py::test_oneof_options", "tests/test_validate.py::test_oneof_text", "tests/test_validate.py::test_oneof_custom_message", "tests/test_validate.py::test_oneof_repr", "tests/test_validate.py::test_containsonly_in_list", "tests/test_validate.py::test_contains_only_unhashable_types", "tests/test_validate.py::test_containsonly_in_tuple", "tests/test_validate.py::test_contains_only_in_string", "tests/test_validate.py::test_contains_only_invalid", "tests/test_validate.py::test_containsonly_custom_message", "tests/test_validate.py::test_containsonly_repr"], "environment_setup_commit": "7015fc4333a2f32cd58c3465296e834acd4496ff"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-2998", "base_commit": "47c8bb29104761474e455ef2e6fdaa7a8cc20a56", "patch": "diff --git a/src/sqlfluff/rules/L027.py b/src/sqlfluff/rules/L027.py\n--- a/src/sqlfluff/rules/L027.py\n+++ b/src/sqlfluff/rules/L027.py\n@@ -73,4 +73,21 @@ def _lint_references_and_aliases(\n                     )\n                 )\n \n+            all_table_aliases = [t.ref_str for t in table_aliases] + standalone_aliases\n+\n+            # For qualified references, we want to check that the alias is actually\n+            # valid\n+            if (\n+                this_ref_type == \"qualified\"\n+                and list(r.iter_raw_references())[0].part not in all_table_aliases\n+            ):\n+                violation_buff.append(\n+                    LintResult(\n+                        anchor=r,\n+                        description=f\"Qualified reference {r.raw!r} not found in \"\n+                        f\"available tables/view aliases {all_table_aliases} in select \"\n+                        \"with more than one referenced table/view.\",\n+                    )\n+                )\n+\n         return violation_buff or None\n", "test_patch": "diff --git a/test/fixtures/rules/std_rule_cases/L027.yml b/test/fixtures/rules/std_rule_cases/L027.yml\n--- a/test/fixtures/rules/std_rule_cases/L027.yml\n+++ b/test/fixtures/rules/std_rule_cases/L027.yml\n@@ -220,3 +220,40 @@ test_pass_rowtype_with_join:\n   configs:\n     core:\n       dialect: hive\n+\n+test_fail_column_name_not_found_in_table_aliases_bigquery:\n+  # qualified reference should actually exists in table aliases\n+  fail_str: |\n+    SELECT\n+        a.bar,\n+        b.foo,\n+        this_is.some_struct.id\n+    FROM\n+        a LEFT JOIN b ON TRUE\n+  configs:\n+    core:\n+      dialect: bigquery\n+\n+test_pass_column_name_is_a_struct_bigquery:\n+  # check structs work as expected\n+  pass_str: |\n+    SELECT\n+        a.bar,\n+        b.this_is.some_struct.id\n+    FROM\n+        a LEFT JOIN b ON TRUE\n+  configs:\n+    core:\n+      dialect: bigquery\n+\n+test_pass_column_name_from_unnest_bigquery:\n+  # Check that we allow an table alias come from UNNEST statement\n+  pass_str: |\n+    SELECT\n+        a.bar,\n+        e.foo\n+    FROM\n+        a LEFT JOIN UNEST(a.events) AS e\n+  configs:\n+    core:\n+      dialect: bigquery\ndiff --git a/test/rules/std_test.py b/test/rules/std_test.py\n--- a/test/rules/std_test.py\n+++ b/test/rules/std_test.py\n@@ -68,7 +68,7 @@\n         ),\n         (\"L016\", \"block_comment_errors_2.sql\", [(1, 85), (2, 86)]),\n         # Column references\n-        (\"L027\", \"column_references.sql\", [(1, 8)]),\n+        (\"L027\", \"column_references.sql\", [(1, 8), (1, 11)]),\n         (\"L027\", \"column_references_bare_function.sql\", []),\n         (\"L026\", \"column_references.sql\", [(1, 11)]),\n         (\"L025\", \"column_references.sql\", [(2, 11)]),\n", "problem_statement": "BigQuery: Accessing `STRUCT` elements evades triggering L027\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\nAccessing unreferenced `STRUCT` elements using BigQuery dot notation in a multi table query does not trigger L027.\n\n### Expected Behaviour\n\nL027 gets triggered.\n\n### Observed Behaviour\n\nL027 does not get triggered.\n\n### How to reproduce\n\n```sql\r\nSELECT\r\n    t1.col1,\r\n    t2.col2,\r\n    events.id\r\nFROM t_table1 AS t1\r\nLEFT JOIN t_table2 AS t2\r\n    ON TRUE\r\n```\n\n### Dialect\n\nBigQUery\n\n### Version\n\n`0.11.2` using online.sqlfluff.com\n\n### Configuration\n\nN/A\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 tricky.\r\n\r\nBasicaly L026 works to make sure qualified columns only use tables in the from clause. This doesn\u2019t really work for `STRUCT`s as impossible to know if it\u2019s a qualified column or a `STRUCT`, so is off by default for languages that support them - like BigQuery.\r\n\r\nL027 works to make sure columns are qualified for multi-table joins (i.e. have at least one dot). But it doesn\u2019t check the qualifiers are valid - that\u2019s L026\u2019s job, which as I say is off by default for BigQuery.", "created_at": "2022-04-04T20:29:42Z", "version": "0.11", "FAIL_TO_PASS": ["test/rules/std_test.py::test__rules__std_file[L027-column_references.sql-violations16]"], "PASS_TO_PASS": ["test/rules/std_test.py::test__rules__std_file[L001-indentation_errors.sql-violations0]", "test/rules/std_test.py::test__rules__std_file[L002-indentation_errors.sql-violations1]", "test/rules/std_test.py::test__rules__std_file[L003-indentation_errors.sql-violations2]", "test/rules/std_test.py::test__rules__std_file[L004-indentation_errors.sql-violations3]", "test/rules/std_test.py::test__rules__std_file[L005-whitespace_errors.sql-violations4]", "test/rules/std_test.py::test__rules__std_file[L019-whitespace_errors.sql-violations5]", "test/rules/std_test.py::test__rules__std_file[L008-whitespace_errors.sql-violations6]", "test/rules/std_test.py::test__rules__std_file[L006-operator_errors.sql-violations7]", "test/rules/std_test.py::test__rules__std_file[L039-operator_errors.sql-violations8]", "test/rules/std_test.py::test__rules__std_file[L007-operator_errors.sql-violations9]", "test/rules/std_test.py::test__rules__std_file[L006-operator_errors_negative.sql-violations10]", "test/rules/std_test.py::test__rules__std_file[L039-operator_errors_negative.sql-violations11]", "test/rules/std_test.py::test__rules__std_file[L003-indentation_error_hard.sql-violations12]", "test/rules/std_test.py::test__rules__std_file[L003-indentation_error_contained.sql-violations13]", "test/rules/std_test.py::test__rules__std_file[L016-block_comment_errors.sql-violations14]", "test/rules/std_test.py::test__rules__std_file[L016-block_comment_errors_2.sql-violations15]", "test/rules/std_test.py::test__rules__std_file[L027-column_references_bare_function.sql-violations17]", "test/rules/std_test.py::test__rules__std_file[L026-column_references.sql-violations18]", "test/rules/std_test.py::test__rules__std_file[L025-column_references.sql-violations19]", "test/rules/std_test.py::test__rules__std_file[L021-select_distinct_group_by.sql-violations20]", "test/rules/std_test.py::test__rules__std_file[L006-operator_errors_ignore.sql-violations21]", "test/rules/std_test.py::test__rules__std_file[L031-aliases_in_join_error.sql-violations22]", "test/rules/std_test.py::test__rules__std_file[L046-heavy_templating.sql-violations23]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict0]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict1]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict2]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict3]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict4]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict5]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict6]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict7]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict8]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict9]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict10]", "test/rules/std_test.py::test_improper_configs_are_rejected[rule_config_dict11]"], "environment_setup_commit": "2bdeb9354d33e3fb4dfd6782e1e1921939ecb55a"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1268", "base_commit": "ce5cbce5ba11cdc2f8139ade66feea1e181a7944", "patch": "diff --git a/astroid/nodes/as_string.py b/astroid/nodes/as_string.py\n--- a/astroid/nodes/as_string.py\n+++ b/astroid/nodes/as_string.py\n@@ -36,6 +36,7 @@\n         MatchSingleton,\n         MatchStar,\n         MatchValue,\n+        Unknown,\n     )\n \n # pylint: disable=unused-argument\n@@ -643,6 +644,9 @@ def visit_property(self, node):\n     def visit_evaluatedobject(self, node):\n         return node.original.accept(self)\n \n+    def visit_unknown(self, node: \"Unknown\") -> str:\n+        return str(node)\n+\n \n def _import_string(names):\n     \"\"\"return a list of (name, asname) formatted as a string\"\"\"\n", "test_patch": "diff --git a/tests/unittest_nodes.py b/tests/unittest_nodes.py\n--- a/tests/unittest_nodes.py\n+++ b/tests/unittest_nodes.py\n@@ -306,6 +306,11 @@ def test_f_strings(self):\n         ast = abuilder.string_build(code)\n         self.assertEqual(ast.as_string().strip(), code.strip())\n \n+    @staticmethod\n+    def test_as_string_unknown() -> None:\n+        assert nodes.Unknown().as_string() == \"Unknown.Unknown()\"\n+        assert nodes.Unknown(lineno=1, col_offset=0).as_string() == \"Unknown.Unknown()\"\n+\n \n class _NodeTest(unittest.TestCase):\n     \"\"\"test transformation of If Node\"\"\"\n", "problem_statement": "'AsStringVisitor' object has no attribute 'visit_unknown'\n```python\r\n>>> import astroid\r\n>>> astroid.nodes.Unknown().as_string()\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv/lib/python3.9/site-packages/astroid/nodes/node_ng.py\", line 609, in as_string\r\n    return AsStringVisitor()(self)\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv/lib/python3.9/site-packages/astroid/nodes/as_string.py\", line 56, in __call__\r\n    return node.accept(self).replace(DOC_NEWLINE, \"\\n\")\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv/lib/python3.9/site-packages/astroid/nodes/node_ng.py\", line 220, in accept\r\n    func = getattr(visitor, \"visit_\" + self.__class__.__name__.lower())\r\nAttributeError: 'AsStringVisitor' object has no attribute 'visit_unknown'\r\n>>> \r\n```\r\n### `python -c \"from astroid import __pkginfo__; print(__pkginfo__.version)\"` output\r\n\r\n2.8.6-dev0\n", "hints_text": "Thank you for opening the issue.\nI don't believe `Unknown().as_string()` is ever called regularly. AFAIK it's only used during inference. What should the string representation of an `Unknown` node be? So not sure this needs to be addressed.\nProbably just `'Unknown'`.\nIt's mostly only a problem when we do something like this:\n\n```python\ninferred = infer(node)\nif inferred is not Uninferable:\n    if inferred.as_string().contains(some_value):\n        ...\n```\nSo for the most part, as long as it doesn't crash we're good.", "created_at": "2021-11-21T16:15:23Z", "version": "2.9", "FAIL_TO_PASS": ["tests/unittest_nodes.py::AsStringTest::test_as_string_unknown"], "PASS_TO_PASS": ["tests/unittest_nodes.py::AsStringTest::test_3k_annotations_and_metaclass", "tests/unittest_nodes.py::AsStringTest::test_3k_as_string", "tests/unittest_nodes.py::AsStringTest::test_as_string", "tests/unittest_nodes.py::AsStringTest::test_as_string_for_list_containing_uninferable", "tests/unittest_nodes.py::AsStringTest::test_class_def", "tests/unittest_nodes.py::AsStringTest::test_ellipsis", "tests/unittest_nodes.py::AsStringTest::test_f_strings", "tests/unittest_nodes.py::AsStringTest::test_frozenset_as_string", "tests/unittest_nodes.py::AsStringTest::test_func_signature_issue_185", "tests/unittest_nodes.py::AsStringTest::test_int_attribute", "tests/unittest_nodes.py::AsStringTest::test_module2_as_string", "tests/unittest_nodes.py::AsStringTest::test_module_as_string", "tests/unittest_nodes.py::AsStringTest::test_operator_precedence", "tests/unittest_nodes.py::AsStringTest::test_slice_and_subscripts", "tests/unittest_nodes.py::AsStringTest::test_slices", "tests/unittest_nodes.py::AsStringTest::test_tuple_as_string", "tests/unittest_nodes.py::AsStringTest::test_varargs_kwargs_as_string", "tests/unittest_nodes.py::IfNodeTest::test_block_range", "tests/unittest_nodes.py::IfNodeTest::test_if_elif_else_node", "tests/unittest_nodes.py::IfNodeTest::test_if_sys_guard", "tests/unittest_nodes.py::IfNodeTest::test_if_typing_guard", "tests/unittest_nodes.py::TryExceptNodeTest::test_block_range", "tests/unittest_nodes.py::TryFinallyNodeTest::test_block_range", "tests/unittest_nodes.py::TryExceptFinallyNodeTest::test_block_range", "tests/unittest_nodes.py::ImportNodeTest::test_absolute_import", "tests/unittest_nodes.py::ImportNodeTest::test_as_string", "tests/unittest_nodes.py::ImportNodeTest::test_bad_import_inference", "tests/unittest_nodes.py::ImportNodeTest::test_conditional", "tests/unittest_nodes.py::ImportNodeTest::test_conditional_import", "tests/unittest_nodes.py::ImportNodeTest::test_from_self_resolve", "tests/unittest_nodes.py::ImportNodeTest::test_import_self_resolve", "tests/unittest_nodes.py::ImportNodeTest::test_more_absolute_import", "tests/unittest_nodes.py::ImportNodeTest::test_real_name", "tests/unittest_nodes.py::CmpNodeTest::test_as_string", "tests/unittest_nodes.py::ConstNodeTest::test_bool", "tests/unittest_nodes.py::ConstNodeTest::test_complex", "tests/unittest_nodes.py::ConstNodeTest::test_copy", "tests/unittest_nodes.py::ConstNodeTest::test_float", "tests/unittest_nodes.py::ConstNodeTest::test_int", "tests/unittest_nodes.py::ConstNodeTest::test_none", "tests/unittest_nodes.py::ConstNodeTest::test_str", "tests/unittest_nodes.py::ConstNodeTest::test_str_kind", "tests/unittest_nodes.py::ConstNodeTest::test_unicode", "tests/unittest_nodes.py::NameNodeTest::test_assign_to_true", "tests/unittest_nodes.py::TestNamedExprNode::test_frame", "tests/unittest_nodes.py::TestNamedExprNode::test_scope", "tests/unittest_nodes.py::AnnAssignNodeTest::test_as_string", "tests/unittest_nodes.py::AnnAssignNodeTest::test_complex", "tests/unittest_nodes.py::AnnAssignNodeTest::test_primitive", "tests/unittest_nodes.py::AnnAssignNodeTest::test_primitive_without_initial_value", "tests/unittest_nodes.py::ArgumentsNodeTC::test_kwoargs", "tests/unittest_nodes.py::ArgumentsNodeTC::test_positional_only", "tests/unittest_nodes.py::UnboundMethodNodeTest::test_no_super_getattr", "tests/unittest_nodes.py::BoundMethodNodeTest::test_is_property", "tests/unittest_nodes.py::AliasesTest::test_aliases", "tests/unittest_nodes.py::Python35AsyncTest::test_async_await_keywords", "tests/unittest_nodes.py::Python35AsyncTest::test_asyncfor_as_string", "tests/unittest_nodes.py::Python35AsyncTest::test_asyncwith_as_string", "tests/unittest_nodes.py::Python35AsyncTest::test_await_as_string", "tests/unittest_nodes.py::Python35AsyncTest::test_decorated_async_def_as_string", "tests/unittest_nodes.py::ContextTest::test_list_del", "tests/unittest_nodes.py::ContextTest::test_list_load", "tests/unittest_nodes.py::ContextTest::test_list_store", "tests/unittest_nodes.py::ContextTest::test_starred_load", "tests/unittest_nodes.py::ContextTest::test_starred_store", "tests/unittest_nodes.py::ContextTest::test_subscript_del", "tests/unittest_nodes.py::ContextTest::test_subscript_load", "tests/unittest_nodes.py::ContextTest::test_subscript_store", "tests/unittest_nodes.py::ContextTest::test_tuple_load", "tests/unittest_nodes.py::ContextTest::test_tuple_store", "tests/unittest_nodes.py::test_unknown", "tests/unittest_nodes.py::test_type_comments_with", "tests/unittest_nodes.py::test_type_comments_for", "tests/unittest_nodes.py::test_type_coments_assign", "tests/unittest_nodes.py::test_type_comments_invalid_expression", "tests/unittest_nodes.py::test_type_comments_invalid_function_comments", "tests/unittest_nodes.py::test_type_comments_function", "tests/unittest_nodes.py::test_type_comments_arguments", "tests/unittest_nodes.py::test_type_comments_posonly_arguments", "tests/unittest_nodes.py::test_correct_function_type_comment_parent", "tests/unittest_nodes.py::test_is_generator_for_yield_assignments", "tests/unittest_nodes.py::test_f_string_correct_line_numbering", "tests/unittest_nodes.py::test_assignment_expression", "tests/unittest_nodes.py::test_assignment_expression_in_functiondef", "tests/unittest_nodes.py::test_get_doc", "tests/unittest_nodes.py::test_parse_fstring_debug_mode", "tests/unittest_nodes.py::test_parse_type_comments_with_proper_parent", "tests/unittest_nodes.py::test_const_itered", "tests/unittest_nodes.py::test_is_generator_for_yield_in_while", "tests/unittest_nodes.py::test_is_generator_for_yield_in_if", "tests/unittest_nodes.py::test_is_generator_for_yield_in_aug_assign"], "environment_setup_commit": "0d1211558670cfefd95b39984b8d5f7f34837f32"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1031", "base_commit": "64f5b8daaa798836579c56912244b7732ab073be", "patch": "diff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -76,7 +76,7 @@ def empty_value_for_VR(VR, raw=False):\n         if it is empty.\n     \"\"\"\n     if VR == 'SQ':\n-        return []\n+        return b'' if raw else []\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_filewriter.py b/pydicom/tests/test_filewriter.py\n--- a/pydicom/tests/test_filewriter.py\n+++ b/pydicom/tests/test_filewriter.py\n@@ -15,11 +15,11 @@\n \n from pydicom._storage_sopclass_uids import CTImageStorage\n from pydicom import config, __version_info__, uid\n-from pydicom.data import get_testdata_files, get_charset_files\n+from pydicom.data import get_testdata_file, get_charset_files\n from pydicom.dataset import Dataset, FileDataset\n from pydicom.dataelem import DataElement, RawDataElement\n from pydicom.filebase import DicomBytesIO\n-from pydicom.filereader import dcmread, read_dataset\n+from pydicom.filereader import dcmread, read_dataset, read_file\n from pydicom.filewriter import (write_data_element, write_dataset,\n                                 correct_ambiguous_vr, write_file_meta_info,\n                                 correct_ambiguous_vr_element, write_numbers,\n@@ -34,16 +34,16 @@\n from pydicom.values import convert_text\n from ._write_stds import impl_LE_deflen_std_hex\n \n-rtplan_name = get_testdata_files(\"rtplan.dcm\")[0]\n-rtdose_name = get_testdata_files(\"rtdose.dcm\")[0]\n-ct_name = get_testdata_files(\"CT_small.dcm\")[0]\n-mr_name = get_testdata_files(\"MR_small.dcm\")[0]\n-mr_implicit_name = get_testdata_files(\"MR_small_implicit.dcm\")[0]\n-mr_bigendian_name = get_testdata_files(\"MR_small_bigendian.dcm\")[0]\n-jpeg_name = get_testdata_files(\"JPEG2000.dcm\")[0]\n-no_ts = get_testdata_files(\"meta_missing_tsyntax.dcm\")[0]\n-color_pl_name = get_testdata_files(\"color-pl.dcm\")[0]\n-sc_rgb_name = get_testdata_files(\"SC_rgb.dcm\")[0]\n+rtplan_name = get_testdata_file(\"rtplan.dcm\")\n+rtdose_name = get_testdata_file(\"rtdose.dcm\")\n+ct_name = get_testdata_file(\"CT_small.dcm\")\n+mr_name = get_testdata_file(\"MR_small.dcm\")\n+mr_implicit_name = get_testdata_file(\"MR_small_implicit.dcm\")\n+mr_bigendian_name = get_testdata_file(\"MR_small_bigendian.dcm\")\n+jpeg_name = get_testdata_file(\"JPEG2000.dcm\")\n+no_ts = get_testdata_file(\"meta_missing_tsyntax.dcm\")\n+color_pl_name = get_testdata_file(\"color-pl.dcm\")\n+sc_rgb_name = get_testdata_file(\"SC_rgb.dcm\")\n datetime_name = mr_name\n \n unicode_name = get_charset_files(\"chrH31.dcm\")[0]\n@@ -204,6 +204,15 @@ def test_write_removes_grouplength(self):\n         # group length has been removed\n         assert 0x00080000 not in ds\n \n+    def test_write_empty_sequence(self):\n+        \"\"\"Make sure that empty sequence is correctly written.\"\"\"\n+        # regression test for #1030\n+        ds = read_file(get_testdata_file('test-SR.dcm'))\n+        ds.save_as(self.file_out)\n+        self.file_out.seek(0)\n+        ds = read_file(self.file_out)\n+        assert ds.PerformedProcedureCodeSequence == []\n+\n \n class TestScratchWriteDateTime(TestWriteFile):\n     \"\"\"Write and reread simple or multi-value DA/DT/TM data elements\"\"\"\n", "problem_statement": "Crash writing DICOM with 1.4.0\npydicom 1.4.0\r\nWindows-10-10.0.18362-SP0\r\nPython  3.7.4 (tags/v3.7.4:e09359112e, Jul  8 2019, 20:34:20) [MSC v.1916 64 bit (AMD64)]\r\nGDCM 3.0.2\r\nPillow 7.0.0\r\n\r\nType error raises when writing file with pydicom 1.4.0, works in 1.3.0.\r\n\r\n```\r\nds = pydicom.read_file('fail2404.anon.dcm')\r\n#print(ds.get((0x0040, 0x0275)))\r\nds.save_as('bort.dcm')\r\n```\r\n\r\nInterestingly, the crash goes away if the offending tag is accessed (uncomment the print and then the `save_as` works fine).\r\n\r\n```\r\nTraceback (most recent call last):\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\tag.py\", line 30, in tag_in_exception\r\n    yield\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 555, in write_dataset\r\n    write_data_element(fp, dataset.get_item(tag), dataset_encoding)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 463, in write_data_element\r\n    buffer.write(data_element.value)\r\nTypeError: a bytes-like object is required, not 'list'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"./pydcmbug.py\", line 7, in <module>\r\n    ds.save_as('bort.dcm')\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\dataset.py\", line 1810, in save_as\r\n    pydicom.dcmwrite(filename, self, write_like_original)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 946, in dcmwrite\r\n    write_dataset(fp, get_item(dataset, slice(0x00010000, None)))\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 555, in write_dataset\r\n    write_data_element(fp, dataset.get_item(tag), dataset_encoding)\r\n  File \"C:\\Program Files\\Python37\\lib\\contextlib.py\", line 130, in __exit__\r\n    self.gen.throw(type, value, traceback)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\tag.py\", line 37, in tag_in_exception\r\n    raise type(ex)(msg)\r\nTypeError: With tag (0040, 0275) got exception: a bytes-like object is required, not 'list'\r\nTraceback (most recent call last):\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\tag.py\", line 30, in tag_in_exception\r\n    yield\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 555, in write_dataset\r\n    write_data_element(fp, dataset.get_item(tag), dataset_encoding)\r\n  File \"C:\\Program Files\\Python37\\lib\\site-packages\\pydicom\\filewriter.py\", line 463, in write_data_element\r\n    buffer.write(data_element.value)\r\nTypeError: a bytes-like object is required, not 'list'\r\n```\r\n\r\n[fail.zip](https://github.com/pydicom/pydicom/files/4072693/fail.zip)\r\n\n", "hints_text": "Ok, found the problem. This has been introduced by by PR #965 (by myself). Will provide a fix ASAP...\r\n", "created_at": "2020-01-16T18:53:50Z", "version": "1.4", "FAIL_TO_PASS": ["pydicom/tests/test_filewriter.py::TestWriteFile::test_write_empty_sequence", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_empty_sequence"], "PASS_TO_PASS": ["pydicom/tests/test_filewriter.py::TestWriteFile::testRTPlan", "pydicom/tests/test_filewriter.py::TestWriteFile::testRTDose", "pydicom/tests/test_filewriter.py::TestWriteFile::testCT", "pydicom/tests/test_filewriter.py::TestWriteFile::testMR", "pydicom/tests/test_filewriter.py::TestWriteFile::testUnicode", "pydicom/tests/test_filewriter.py::TestWriteFile::testMultiPN", "pydicom/tests/test_filewriter.py::TestWriteFile::testJPEG2000", "pydicom/tests/test_filewriter.py::TestWriteFile::testListItemWriteBack", "pydicom/tests/test_filewriter.py::TestWriteFile::testwrite_short_uid", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_no_ts", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_double_filemeta", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_ffff_ffff", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_removes_grouplength", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testRTPlan", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testRTDose", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testCT", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testMR", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testUnicode", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testMultiPN", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testJPEG2000", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testListItemWriteBack", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testwrite_short_uid", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_no_ts", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_double_filemeta", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_ffff_ffff", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_removes_grouplength", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_multivalue_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_empty_AT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_empty_LO", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_TM", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_TM", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_DT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_DT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_ascii_vr_with_padding", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OD_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OD_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OL_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OL_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UC_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UC_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UR_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UR_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UN_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_unknown_vr_raises", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_representation_vm_one", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_representation_vm_three", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_data", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_waveform_bits_allocated", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_lut_descriptor", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_overlay", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_sequence", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_write_new_ambiguous", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_explicit_using_attribute", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_explicit_using_index", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_implicit_using_attribute", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_implicit_using_index", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_not_ambiguous", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_not_ambiguous_raw_data_element", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_correct_ambiguous_data_element", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_correct_ambiguous_raw_data_element", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_raises", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_little_endian", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_big_endian", "pydicom/tests/test_filewriter.py::TestScratchWrite::testImpl_LE_deflen_write", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_preamble_default", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_preamble_custom", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_no_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_none_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_bad_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_prefix", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_prefix_none", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_ds_changed", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raw_elements_preserved_implicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raw_elements_preserved_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_implicit_to_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_dataset", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_dataset_with_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_implicit_to_explicit_vr_using_destination", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_explicit_to_implicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_big_to_little_endian", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_little_to_big_endian", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_changed_character_set", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_private_tag_vr_from_implicit_data", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_rgb_from_implicit_to_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_not_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_raises", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_media_storage_sop_class_uid_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_no_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raise_no_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_add_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_standard", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_commandset_no_written", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_bad_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_missing_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_group_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_group_length_updated", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_version", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_implementation_version_name_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_implementation_class_uid_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_filelike_position", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_default", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_custom", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_no_preamble", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_ds_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_file_meta_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_filemeta", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_filemeta", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_read_write_identical", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_transfer_syntax_not_added", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_bad_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_missing_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_group_length_updated", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_filelike_position", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_meta_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_empty_value", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_list", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_singleton", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_exception", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_big_endian", "pydicom/tests/test_filewriter.py::TestWritePN::test_no_encoding", "pydicom/tests/test_filewriter.py::TestWritePN::test_single_byte_multi_charset_groups", "pydicom/tests/test_filewriter.py::TestWritePN::test_single_byte_multi_charset_values", "pydicom/tests/test_filewriter.py::TestWriteText::test_no_encoding", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_byte_multi_charset_text", "pydicom/tests/test_filewriter.py::TestWriteText::test_encode_mixed_charsets_text", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_byte_multi_charset_text_multivalue", "pydicom/tests/test_filewriter.py::TestWriteText::test_invalid_encoding", "pydicom/tests/test_filewriter.py::TestWriteText::test_invalid_encoding_enforce_standard", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_value_with_delimiters", "pydicom/tests/test_filewriter.py::TestWriteDT::test_format_dt", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_little_endian_correct_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_big_endian_correct_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_little_endian_incorrect_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_big_endian_incorrect_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_writing_to_gzip", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_writing_too_big_data_in_explicit_encoding"], "environment_setup_commit": "5098c9147fadcb3e5918487036867931435adeb8"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1959", "base_commit": "49691cc04f2d38b174787280f7ed38f818c828bd", "patch": "diff --git a/astroid/raw_building.py b/astroid/raw_building.py\n--- a/astroid/raw_building.py\n+++ b/astroid/raw_building.py\n@@ -324,6 +324,17 @@ def _build_from_function(\n         object_build_function(node, member, name)\n \n \n+def _safe_has_attribute(obj, member: str) -> bool:\n+    \"\"\"Required because unexpected RunTimeError can be raised.\n+\n+    See https://github.com/PyCQA/astroid/issues/1958\n+    \"\"\"\n+    try:\n+        return hasattr(obj, member)\n+    except Exception:  # pylint: disable=broad-except\n+        return False\n+\n+\n class InspectBuilder:\n     \"\"\"class for building nodes from living object\n \n@@ -419,7 +430,7 @@ def object_build(\n                 # This should be called for Jython, where some builtin\n                 # methods aren't caught by isbuiltin branch.\n                 _build_from_function(node, name, member, self._module)\n-            elif hasattr(member, \"__all__\"):\n+            elif _safe_has_attribute(member, \"__all__\"):\n                 module = build_module(name)\n                 _attach_local_node(node, module, name)\n                 # recursion\n", "test_patch": "diff --git a/tests/testdata/python3/data/fake_module_with_broken_getattr.py b/tests/testdata/python3/data/fake_module_with_broken_getattr.py\nnew file mode 100644\n--- /dev/null\n+++ b/tests/testdata/python3/data/fake_module_with_broken_getattr.py\n@@ -0,0 +1,7 @@\n+class Broken:\n+\n+    def __getattr__(self, name):\n+        raise Exception(\"boom\")\n+\n+\n+broken = Broken()\ndiff --git a/tests/unittest_raw_building.py b/tests/unittest_raw_building.py\n--- a/tests/unittest_raw_building.py\n+++ b/tests/unittest_raw_building.py\n@@ -1,3 +1,9 @@\n+\"\"\"\n+'tests.testdata.python3.data.fake_module_with_warnings' and\n+'tests.testdata.python3.data.fake_module_with_warnings' are fake modules\n+to simulate issues in unittest below\n+\"\"\"\n+\n # Licensed under the LGPL: https://www.gnu.org/licenses/old-licenses/lgpl-2.1.en.html\n # For details: https://github.com/PyCQA/astroid/blob/main/LICENSE\n # Copyright (c) https://github.com/PyCQA/astroid/blob/main/CONTRIBUTORS.txt\n@@ -8,7 +14,7 @@\n import _io\n import pytest\n \n-# A fake module to simulate pandas in unittest below\n+import tests.testdata.python3.data.fake_module_with_broken_getattr as fm_getattr\n import tests.testdata.python3.data.fake_module_with_warnings as fm\n from astroid.builder import AstroidBuilder\n from astroid.const import IS_PYPY\n@@ -102,6 +108,14 @@ def test_build_function_deepinspect_deprecation(self) -> None:\n         # This should not raise an exception\n         AstroidBuilder().module_build(m, \"test\")\n \n+    def test_module_object_with_broken_getattr(self) -> None:\n+        # Tests https://github.com/PyCQA/astroid/issues/1958\n+        # When astroid deep inspection of modules raises\n+        # errors when using hasattr().\n+\n+        # This should not raise an exception\n+        AstroidBuilder().inspect_build(fm_getattr, \"test\")\n+\n \n if __name__ == \"__main__\":\n     unittest.main()\n", "problem_statement": "v2.13.x regression: Crash when inspecting `PyQt5.QtWidgets` due to `RuntimeError` during `hasattr`\n### Steps to reproduce\r\n\r\nInstall PyQt5, run `pylint --extension-pkg-whitelist=PyQt5 x.py` over a file containing `from PyQt5 import QtWidgets`\r\n\r\n### Current behavior\r\n\r\nWith astroid 2.12.13 and pylint 2.15.10, this works fine. With astroid 2.13.2, this happens:\r\n\r\n```pytb\r\nException on node <ImportFrom l.1 at 0x7fc5a3c47d00> in file '/home/florian/tmp/pylintbug/x.py'\r\nTraceback (most recent call last):\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/utils/ast_walker.py\", line 90, in walk\r\n    callback(astroid)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/checkers/variables.py\", line 1726, in visit_importfrom\r\n    self._check_module_attrs(node, module, name.split(\".\"))\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/checkers/variables.py\", line 2701, in _check_module_attrs\r\n    module = next(module.getattr(name)[0].infer())\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 412, in getattr\r\n    result = [self.import_module(name, relative_only=True)]\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 527, in import_module\r\n    return AstroidManager().ast_from_module_name(\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/manager.py\", line 205, in ast_from_module_name\r\n    return self.ast_from_module(named_module, modname)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/manager.py\", line 312, in ast_from_module\r\n    return AstroidBuilder(self).module_build(module, modname)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/builder.py\", line 101, in module_build\r\n    node = self.inspect_build(module, modname=modname, path=path)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/raw_building.py\", line 366, in inspect_build\r\n    self.object_build(node, module)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/raw_building.py\", line 422, in object_build\r\n    elif hasattr(member, \"__all__\"):\r\nRuntimeError: wrapped C/C++ object of type QApplication has been deleted\r\nx.py:1:0: F0002: x.py: Fatal error while checking 'x.py'. Please open an issue in our bug tracker so we address this. There is a pre-filled template that you can use in '/home/florian/.cache/pylint/pylint-crash-2023-01-10-11-06-17.txt'. (astroid-error)\r\n```\r\n\r\nIt looks like it happens when `member` is `QtWidgets.qApp`, which is a kind of \"magic\" object referring to the QApplication singleton. Since none exists, it looks like PyQt doesn't like trying to access an attribute on that.\r\n\r\nBisected to:\r\n\r\n- #1885 \r\n\r\nIt looks like 974f26f75eb3eccb4bcd8ea143901baf60a685ff is the exact culprit.\r\n\r\ncc @nickdrozd \r\n\r\n(took the freedom to add appropriate labels already, hope that's fine)\r\n\n", "hints_text": "Thank you for the investigation @The-Compiler ! I'm going to revert the offending commit seeing it's only a cleanup. Let's do something more elegant later on if required.\nThe commit you linked is from a fork, so I tried to reproduce locally to bisect on pylint main repo but I cannot\r\n```\r\npip3 install PyQt5\r\nCollecting PyQt5\r\n Collecting PyQt5-sip<13,>=12.11\r\nCollecting PyQt5-Qt5>=5.15.0\r\nInstalling collected packages: PyQt5-Qt5, PyQt5-sip, PyQt5\r\nSuccessfully installed PyQt5-5.15.7 PyQt5-Qt5-5.15.2 PyQt5-sip-12.11.0\r\n(venv) fix-crash-regression-2.13.2: pylint --extension-pkg-whitelist=PyQt5 x.py \r\n************* Module x\r\nx.py:1:0: W0611: Unused QtWidgets imported from PyQt5 (unused-import)\r\n---------------------------------------------------------------------\r\n\r\nYour code has been rated at 0.00/10 (previous run: -50.00/10, +50.00)\r\n```\r\nCould you provide your other dependencies maybe ?\r\n\n> The commit you linked is from a fork, so I tried to reproduce locally to bisect on pylint main repo but I cannot\r\n\r\nThe main astroid repo bisects to the merge of #1885 (f26dbe419ac15a87ed65e9b55ed15d3d8100b608) - that was a squash merge, 974f26f75eb3eccb4bcd8ea143901baf60a685ff is my guess at what in the PR is the problem.\r\n\r\n> Could you provide your other dependencies maybe ?\r\n\r\nNothing else really:\r\n\r\n```\r\n\u2500[florian@aragog]\u2500\u2500[~/tmp/pylintbug]\u2500\u2500[23-01-10]\u2500\u2500[12:35]\u2500\u2500\u2500\u2500\u2504\r\n$ python3 --version\r\nPython 3.10.8\r\n\r\n\u2500[florian@aragog]\u2500\u2500[~/tmp/pylintbug]\u2500\u2500[23-01-10]\u2500\u2500[12:35]\u2500\u2500\u2500\u2500\u2504\r\n$ python3 -m venv .venv\r\npython3 -m venv .venv  2.87s user 0.26s system 98% cpu 3.170 total\r\n\r\n\u2500[florian@aragog]\u2500\u2500[~/tmp/pylintbug]\u2500\u2500[23-01-10]\u2500\u2500[12:35]\u2500\u2500\u2500\u2500\u2504\r\n$ .venv/bin/pip install pylint PyQt5\r\nCollecting pylint\r\n  Using cached pylint-2.15.10-py3-none-any.whl (509 kB)\r\nCollecting PyQt5\r\n  Using cached PyQt5-5.15.7-cp37-abi3-manylinux1_x86_64.whl (8.4 MB)\r\nCollecting tomli>=1.1.0\r\n  Using cached tomli-2.0.1-py3-none-any.whl (12 kB)\r\nCollecting tomlkit>=0.10.1\r\n  Using cached tomlkit-0.11.6-py3-none-any.whl (35 kB)\r\nCollecting platformdirs>=2.2.0\r\n  Using cached platformdirs-2.6.2-py3-none-any.whl (14 kB)\r\nCollecting mccabe<0.8,>=0.6\r\n  Using cached mccabe-0.7.0-py2.py3-none-any.whl (7.3 kB)\r\nCollecting dill>=0.2\r\n  Using cached dill-0.3.6-py3-none-any.whl (110 kB)\r\nCollecting astroid<=2.14.0-dev0,>=2.12.13\r\n  Using cached astroid-2.13.2-py3-none-any.whl (272 kB)\r\nCollecting isort<6,>=4.2.5\r\n  Using cached isort-5.11.4-py3-none-any.whl (104 kB)\r\nCollecting PyQt5-Qt5>=5.15.0\r\n  Using cached PyQt5_Qt5-5.15.2-py3-none-manylinux2014_x86_64.whl (59.9 MB)\r\nCollecting PyQt5-sip<13,>=12.11\r\n  Using cached PyQt5_sip-12.11.0-cp310-cp310-manylinux1_x86_64.whl (359 kB)\r\nCollecting lazy-object-proxy>=1.4.0\r\n  Using cached lazy_object_proxy-1.9.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl (63 kB)\r\nCollecting wrapt<2,>=1.11\r\n  Using cached wrapt-1.14.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl (77 kB)\r\nCollecting typing-extensions>=4.0.0\r\n  Using cached typing_extensions-4.4.0-py3-none-any.whl (26 kB)\r\nInstalling collected packages: PyQt5-Qt5, wrapt, typing-extensions, tomlkit, tomli, PyQt5-sip, platformdirs, mccabe, lazy-object-proxy, isort, dill, PyQt5, astroid, pylint\r\nSuccessfully installed PyQt5-5.15.7 PyQt5-Qt5-5.15.2 PyQt5-sip-12.11.0 astroid-2.13.2 dill-0.3.6 isort-5.11.4 lazy-object-proxy-1.9.0 mccabe-0.7.0 platformdirs-2.6.2 pylint-2.15.10 tomli-2.0.1 tomlkit-0.11.6 typing-extensions-4.4.0 wrapt-1.14.1\r\n\r\n[notice] A new release of pip available: 22.2.2 -> 22.3.1\r\n[notice] To update, run: python3 -m pip install --upgrade pip\r\n.venv/bin/pip install pylint PyQt5  3.18s user 0.57s system 74% cpu 5.004 total\r\n\r\n\u2500[florian@aragog]\u2500\u2500[~/tmp/pylintbug]\u2500\u2500[23-01-10]\u2500\u2500[12:35]\u2500\u2500\u2500\u2500\u2504\r\n$ cat x.py\r\nfrom PyQt5 import QtWidgets\r\n\r\n\u2500[florian@aragog]\u2500\u2500[~/tmp/pylintbug]\u2500\u2500[23-01-10]\u2500\u2500[12:35]\u2500\u2500\u2500\u2500\u2504\r\n$ .venv/bin/pylint --extension-pkg-whitelist=PyQt5 x.py \r\n************* Module x\r\nx.py:1:0: C0114: Missing module docstring (missing-module-docstring)\r\nException on node <ImportFrom l.1 at 0x7fa1f6e37d90> in file '/home/florian/tmp/pylintbug/x.py'\r\nTraceback (most recent call last):\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/utils/ast_walker.py\", line 90, in walk\r\n    callback(astroid)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/checkers/variables.py\", line 1726, in visit_importfrom\r\n    self._check_module_attrs(node, module, name.split(\".\"))\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/pylint/checkers/variables.py\", line 2701, in _check_module_attrs\r\n    module = next(module.getattr(name)[0].infer())\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 412, in getattr\r\n    result = [self.import_module(name, relative_only=True)]\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 527, in import_module\r\n    return AstroidManager().ast_from_module_name(\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/manager.py\", line 205, in ast_from_module_name\r\n    return self.ast_from_module(named_module, modname)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/manager.py\", line 312, in ast_from_module\r\n    return AstroidBuilder(self).module_build(module, modname)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/builder.py\", line 101, in module_build\r\n    node = self.inspect_build(module, modname=modname, path=path)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/raw_building.py\", line 366, in inspect_build\r\n    self.object_build(node, module)\r\n  File \"/home/florian/tmp/pylintbug/.venv/lib/python3.10/site-packages/astroid/raw_building.py\", line 422, in object_build\r\n    elif hasattr(member, \"__all__\"):\r\nRuntimeError: wrapped C/C++ object of type QApplication has been deleted\r\nx.py:1:0: F0002: x.py: Fatal error while checking 'x.py'. Please open an issue in our bug tracker so we address this. There is a pre-filled template that you can use in '/home/florian/.cache/pylint/pylint-crash-2023-01-10-12-36-04.txt'. (astroid-error)\r\n\r\n------------------------------------------------------------------\r\nYour code has been rated at 0.00/10 (previous run: 0.00/10, +0.00)\r\n```\r\n", "created_at": "2023-01-10T12:07:01Z", "version": "2.14", "FAIL_TO_PASS": ["tests/unittest_raw_building.py::RawBuildingTC::test_module_object_with_broken_getattr"], "PASS_TO_PASS": ["tests/unittest_raw_building.py::RawBuildingTC::test_attach_dummy_node", "tests/unittest_raw_building.py::RawBuildingTC::test_build_class", "tests/unittest_raw_building.py::RawBuildingTC::test_build_from_import", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_args", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_deepinspect_deprecation", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_defaults", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_kwonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_posonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_module", "tests/unittest_raw_building.py::RawBuildingTC::test_io_is__io"], "environment_setup_commit": "0c9ab0fe56703fa83c73e514a1020d398d23fa7f"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1349", "base_commit": "aba071f707f9025882e57f3e55cc9e3e90e869b2", "patch": "diff --git a/pvlib/spectrum/spectrl2.py b/pvlib/spectrum/spectrl2.py\n--- a/pvlib/spectrum/spectrl2.py\n+++ b/pvlib/spectrum/spectrl2.py\n@@ -260,6 +260,11 @@ def spectrl2(apparent_zenith, aoi, surface_tilt, ground_albedo,\n     2-5                 kasten1966 kasten1966 kastenyoung1989\n     =================== ========== ========== ===============\n \n+    This implementation also deviates from the reference by including a\n+    check for angles of incidence greater than 90 degrees; without this,\n+    the model might return negative spectral irradiance values when the\n+    sun is behind the plane of array.\n+\n     References\n     ----------\n     .. [1] Bird, R, and Riordan, C., 1984, \"Simple solar spectral model for\n@@ -357,10 +362,16 @@ def spectrl2(apparent_zenith, aoi, surface_tilt, ground_albedo,\n     Is = (Ir + Ia + Ig) * Cs  # Eq 3-1\n \n     # calculate spectral irradiance on a tilted surface, Eq 3-18\n-    Ibeam = Id * cosd(aoi)\n-\n-    # don't need surface_azimuth if we provide projection_ratio\n-    projection_ratio = cosd(aoi) / cosZ\n+    # Note: clipping cosd(aoi) to >=0 is not in the reference, but is necessary\n+    # to prevent nonsense values when the sun is behind the plane of array.\n+    # The same constraint is applied in irradiance.haydavies when not\n+    # supplying `projection_ratio`.\n+    aoi_projection_nn = np.maximum(cosd(aoi), 0)  # GH 1348\n+    Ibeam = Id * aoi_projection_nn\n+\n+    # don't need surface_azimuth if we provide projection_ratio.\n+    # Also constrain cos zenith to avoid blowup, as in irradiance.haydavies\n+    projection_ratio = aoi_projection_nn / np.maximum(cosZ, 0.01745)\n     Isky = pvlib.irradiance.haydavies(surface_tilt=surface_tilt,\n                                       surface_azimuth=None,\n                                       dhi=Is,\n", "test_patch": "diff --git a/pvlib/tests/test_spectrum.py b/pvlib/tests/test_spectrum.py\n--- a/pvlib/tests/test_spectrum.py\n+++ b/pvlib/tests/test_spectrum.py\n@@ -92,3 +92,17 @@ def test_dayofyear_missing(spectrl2_data):\n     kwargs.pop('dayofyear')\n     with pytest.raises(ValueError, match='dayofyear must be specified'):\n         _ = spectrum.spectrl2(**kwargs)\n+\n+\n+def test_aoi_gt_90(spectrl2_data):\n+    # test that returned irradiance values are non-negative when aoi > 90\n+    # see GH #1348\n+    kwargs, _ = spectrl2_data\n+    kwargs['apparent_zenith'] = 70\n+    kwargs['aoi'] = 130\n+    kwargs['surface_tilt'] = 60\n+\n+    spectra = spectrum.spectrl2(**kwargs)\n+    for key in ['poa_direct', 'poa_global']:\n+        message = f'{key} contains negative values for aoi>90'\n+        assert np.all(spectra[key] >= 0), message\n", "problem_statement": "`spectrum.spectrl2` calculates negative irradiance for angle of incidence outside +/- 90\u00b0\nWhen using pvlib (but also the spectrl2 implementation provided by NREL), I obtain negative Irradiance for a north-facing panel.\r\nFrom @kevinsa5 's [reply on StackOverflow](https://stackoverflow.com/questions/70172766/pvlib-bird1984-north-facing-element-shows-negative-irradiance/70174010#70174010) I take that this is in fact not intended.\r\n\r\nIn the example code below, the angle of incidence is calculated as values around 115\u00b0, so exceeding a possible (implicitly assumed) +/- 90\u00b0 bound (sun behind panel).\r\n\r\nThis seems to be left open in the original report ([Bird & Riordan, 1984](https://www.nrel.gov/docs/legosti/old/2436.pdf)).\r\n\r\nThe direct irradiance `I_d` (*of a horizontal panel*, Eq 2-1) is obtained by multiplying by cosine of the sun zenith angle. I'd guess that setting that value strictly to zero for angles when cosZ is negative would not be too much of a stretch.\r\n\r\nThen, the direct irradiance `I_d` goes into (Eq 3-18):\r\n\r\n```\r\nI_T(t) = I_d*cos(aoi) + I_s * ( (I_d*cos(aoi) / (H_0*D*cos(Z)) ) + 0.5*(1+cos(t)) * (1 - I_d/(H_0*D)) + 0.5 * I_T0 * r_g * (1-cos(t))\r\n```\r\n\r\nAs such, when you view the angle of incidence `aoi` as the analogue of the sun zenith angle in the prior example, the two first terms of the diffuse irradiation (Eq 3-18) would become zero, which - again - for the direct irradiance would kind of make sense. What remains of (Eq 3-18) would be\r\n\r\n```\r\nI_T(t) = 0 + 0 + 0.5*(1+cos(t))*(1 - 0) + 0.5*I_T0*r_g*(1-cos(t))\r\n```\r\n\r\nI'm not from the field, so I'm very, very wary about the implications of such a work-around suggestion. Can anyone with a proper background comment on this? (Maybe it's the future of air conditioning :-D)\r\n\r\n\r\n**MWE based on the tutorial below**\r\n\r\n```python\r\n## Using PV Lib\r\n\r\nfrom pvlib import spectrum, solarposition, irradiance, atmosphere\r\nimport pandas as pd\r\nimport matplotlib.pyplot as plt\r\n\r\n# assumptions from the technical report:\r\nlat = 49.88\r\nlon = 8.63\r\ntilt = 45\r\nazimuth = 0 # North = 0\r\npressure = 101300  # sea level, roughly\r\nwater_vapor_content = 0.5  # cm\r\ntau500 = 0.1\r\nozone = 0.31  # atm-cm\r\nalbedo = 0.2\r\n\r\ntimes = pd.date_range('2021-11-30 8:00', freq='h', periods=6, tz=\"Europe/Berlin\") # , tz='Etc/GMT+9'\r\nsolpos = solarposition.get_solarposition(times, lat, lon)\r\naoi = irradiance.aoi(tilt, azimuth, solpos.apparent_zenith, solpos.azimuth)\r\n\r\n# The technical report uses the 'kasten1966' airmass model, but later\r\n# versions of SPECTRL2 use 'kastenyoung1989'.  Here we use 'kasten1966'\r\n# for consistency with the technical report.\r\nrelative_airmass = atmosphere.get_relative_airmass(solpos.apparent_zenith,\r\n                                                   model='kasten1966')\r\n\r\nspectra = spectrum.spectrl2(\r\n    apparent_zenith=solpos.apparent_zenith,\r\n    aoi=aoi,\r\n    surface_tilt=tilt,\r\n    ground_albedo=albedo,\r\n    surface_pressure=pressure,\r\n    relative_airmass=relative_airmass,\r\n    precipitable_water=water_vapor_content,\r\n    ozone=ozone,\r\n    aerosol_turbidity_500nm=tau500,\r\n)\r\n\r\nplt.figure()\r\nplt.plot(spectra['wavelength'], spectra['poa_global'])\r\nplt.xlim(200, 2700)\r\n# plt.ylim(0, 1.8)\r\nplt.title(r\"2021-11-30, Darmstadt, $\\tau=0.1$, Wv=0.5 cm\")\r\nplt.ylabel(r\"Irradiance ($W m^{-2} nm^{-1}$)\")\r\nplt.xlabel(r\"Wavelength ($nm$)\")\r\ntime_labels = times.strftime(\"%H:%M %p\")\r\nlabels = [\r\n    \"AM {:0.02f}, Z{:0.02f}, {}\".format(*vals)\r\n    for vals in zip(relative_airmass, solpos.apparent_zenith, time_labels)\r\n]\r\nplt.legend(labels)\r\nplt.show()\r\n```\r\n\r\n![Figure_ne](https://user-images.githubusercontent.com/15192310/144224709-dea899e4-435e-4ff2-a3de-9e9524b28eb8.png)\r\n\r\n\n", "hints_text": "Thanks @cweickhmann!  I want to take a closer look at the technical report to be sure, but on a first glance I think the problem here is the same one solved by the line marked with `# GH 526` in `irradiance.haydavies`:\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/aba071f707f9025882e57f3e55cc9e3e90e869b2/pvlib/irradiance.py#L811-L816\r\n\r\nNote that, even though `spectrum.spectrl2` uses `irradiance.haydavies` under the hood, the above branch is not hit because `spectrl2` passes in a pre-calculated `projection_ratio`.  So I think clipping the projection to be non-negative before passing it to `haydavies` would solve the problem.  The `# GH 432` line might be desirable as well, though I don't think it's relevant for this issue.  \r\n\r\nDoes anyone have qualms about us deviating from the reference by implementing that fix and making a note about it in the docstring?  `aoi > 90` is hardly an uncommon occurrence, even for arrays that aren't high-latitude and facing north. \n> deviating from the reference by implementing that fix and making a note about it \r\n\r\nI support that.", "created_at": "2021-12-04T20:54:05Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_spectrum.py::test_aoi_gt_90"], "PASS_TO_PASS": ["pvlib/tests/test_spectrum.py::test_spectrl2", "pvlib/tests/test_spectrum.py::test_spectrl2_array", "pvlib/tests/test_spectrum.py::test_spectrl2_series", "pvlib/tests/test_spectrum.py::test_dayofyear_missing"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4834", "base_commit": "e884df2d00473a6ab166cb92a68d0b500a89d159", "patch": "diff --git a/src/sqlfluff/rules/layout/LT12.py b/src/sqlfluff/rules/layout/LT12.py\n--- a/src/sqlfluff/rules/layout/LT12.py\n+++ b/src/sqlfluff/rules/layout/LT12.py\n@@ -124,6 +124,10 @@ def _eval(self, context: RuleContext) -> Optional[LintResult]:\n         # We only care about the final segment of the parse tree.\n         parent_stack, segment = get_last_segment(FunctionalContext(context).segment)\n         self.logger.debug(\"Found last segment as: %s\", segment)\n+        if not segment:\n+            # NOTE: Edge case. If the file is totally empty, we won't find a final\n+            # segment. In this case return without error.\n+            return None\n \n         trailing_newlines = Segments(*get_trailing_newlines(context.segment))\n         trailing_literal_newlines = trailing_newlines\ndiff --git a/src/sqlfluff/utils/reflow/sequence.py b/src/sqlfluff/utils/reflow/sequence.py\n--- a/src/sqlfluff/utils/reflow/sequence.py\n+++ b/src/sqlfluff/utils/reflow/sequence.py\n@@ -104,7 +104,10 @@ def get_raw(self) -> str:\n \n     @staticmethod\n     def _validate_reflow_sequence(elements: ReflowSequenceType):\n-        assert elements, \"ReflowSequence has empty elements.\"\n+        # An empty set of elements _is_ allowed as an edge case.\n+        if not elements:\n+            # Return early if so\n+            return None\n         # Check odds and evens\n         OddType = elements[0].__class__\n         EvenType = ReflowPoint if OddType is ReflowBlock else ReflowBlock\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@@ -251,6 +251,14 @@ 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_lint_empty_stdin():\n+    \"\"\"Check linting an empty file raises no exceptions.\n+\n+    https://github.com/sqlfluff/sqlfluff/issues/4807\n+    \"\"\"\n+    invoke_assert_code(args=[lint, (\"-d\", \"ansi\", \"-\")], cli_input=\"\")\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", "problem_statement": "Running `lint` on an empty file fails with critical Exception\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\nThis is a bit of an odd one. When running `sqlfluff lint` on an empty file it fails with (Python) exception.\r\n\r\nWhile trying to lint empty file is probably not the main use-case for SQLFluff I still consider this somewhat relevant, when applying SQLFluff in a dynamic code base. \n\n### Expected Behaviour\n\nI'm not entirely sure what the correct result is. Feasible option are\r\n\r\n- Passing\r\n- Raise some kind of lint error (but not a critical exception)\r\n\r\nMy personal take is that lint should pass, which (I think) is similar behaviour to other linters.\n\n### Observed Behaviour\n\n`LT01` and `LT12` with an critical Exception\r\n\r\n```\r\nCRITICAL   [LT01] Applying rule LT01 to 'stdin' threw an Exception: ReflowSequence has empty elements.\r\nCRITICAL   [LT12] Applying rule LT12 to 'stdin' threw an Exception: tuple index out of range\r\n```\r\n\n\n### How to reproduce\n\n```sh\r\ncat /dev/null | sqlfluff lint --dialect ansi -\r\n```\n\n### Dialect\n\nansi\n\n### Version\n\nlatest main branch\r\n\r\n```\r\ngit rev-parse HEAD\r\nd19de0ecd16d298f9e3bfb91da122734c40c01e5\r\n```\n\n### Configuration\n\ndefault\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\nRunning `lint` on an empty file fails with critical Exception\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\nThis is a bit of an odd one. When running `sqlfluff lint` on an empty file it fails with (Python) exception.\r\n\r\nWhile trying to lint empty file is probably not the main use-case for SQLFluff I still consider this somewhat relevant, when applying SQLFluff in a dynamic code base. \n\n### Expected Behaviour\n\nI'm not entirely sure what the correct result is. Feasible option are\r\n\r\n- Passing\r\n- Raise some kind of lint error (but not a critical exception)\r\n\r\nMy personal take is that lint should pass, which (I think) is similar behaviour to other linters.\n\n### Observed Behaviour\n\n`LT01` and `LT12` with an critical Exception\r\n\r\n```\r\nCRITICAL   [LT01] Applying rule LT01 to 'stdin' threw an Exception: ReflowSequence has empty elements.\r\nCRITICAL   [LT12] Applying rule LT12 to 'stdin' threw an Exception: tuple index out of range\r\n```\r\n\n\n### How to reproduce\n\n```sh\r\ncat /dev/null | sqlfluff lint --dialect ansi -\r\n```\n\n### Dialect\n\nansi\n\n### Version\n\nlatest main branch\r\n\r\n```\r\ngit rev-parse HEAD\r\nd19de0ecd16d298f9e3bfb91da122734c40c01e5\r\n```\n\n### Configuration\n\ndefault\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'll pick up this one - I've been in the \"empty\" file code recently and might have influenced this outcome.\nI'll pick up this one - I've been in the \"empty\" file code recently and might have influenced this outcome.", "created_at": "2023-05-02T06:58:53Z", "version": "2.0", "FAIL_TO_PASS": ["test/cli/commands_test.py::test__cli__command_lint_empty_stdin"], "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[command26]", "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-0]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command4-0]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command5-2]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command6-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command7-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command8-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command9-2]", "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[LT01-test/fixtures/linter/indentation_errors.sql0]", "test/cli/commands_test.py::test__cli__command__fix[LT01-test/fixtures/linter/whitespace_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[LT01-test/fixtures/linter/indentation_errors.sql1]", "test/cli/commands_test.py::test__cli__command__fix[LT02-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_format_stdin[select", "test/cli/commands_test.py::test__cli__command_format_stdin[", "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[LT01-test/fixtures/linter/indentation_errors.sql-y-0-0]", "test/cli/commands_test.py::test__cli__command__fix_no_force[LT01-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[None-none]", "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_multiple_files[outfile-none]", "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::TestProgressBars::test_cli_fix_disabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_fix_disabled_progress_bar_deprecated_option", "test/cli/commands_test.py::test__cli__fix_multiple_errors_no_show_errors", "test/cli/commands_test.py::test__cli__fix_multiple_errors_quiet_force", "test/cli/commands_test.py::test__cli__fix_multiple_errors_quiet_no_force", "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": "3629c3e702939c07264cc5ea903566ddc9ea2bb0"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1653", "base_commit": "faf27fee1fcbae3b7056bd02f9e98b7d6e5cb42d", "patch": "diff --git a/pvlib/snow.py b/pvlib/snow.py\n--- a/pvlib/snow.py\n+++ b/pvlib/snow.py\n@@ -219,7 +219,7 @@ def _townsend_effective_snow(snow_total, snow_events):\n \n def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n                   temp_air, poa_global, slant_height, lower_edge_height,\n-                  angle_of_repose=40):\n+                  string_factor=1.0, angle_of_repose=40):\n     '''\n     Calculates monthly snow loss based on the Townsend monthly snow loss\n     model [1]_.\n@@ -230,7 +230,8 @@ def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n         Snow received each month. Referred to as S in [1]_. [cm]\n \n     snow_events : array-like\n-        Number of snowfall events each month. Referred to as N in [1]_. [-]\n+        Number of snowfall events each month. May be int or float type for\n+        the average events in a typical month. Referred to as N in [1]_.\n \n     surface_tilt : float\n         Tilt angle of the array. [deg]\n@@ -250,6 +251,11 @@ def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n     lower_edge_height : float\n         Distance from array lower edge to the ground. [m]\n \n+    string_factor : float, default 1.0\n+        Multiplier applied to monthly loss fraction. Use 1.0 if the DC array\n+        has only one string of modules in the slant direction, use 0.75\n+        otherwise. [-]\n+\n     angle_of_repose : float, default 40\n         Piled snow angle, assumed to stabilize at 40\u00b0, the midpoint of\n         25\u00b0-55\u00b0 avalanching slope angles. [deg]\n@@ -263,7 +269,12 @@ def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n     -----\n     This model has not been validated for tracking arrays; however, for\n     tracking arrays [1]_ suggests using the maximum rotation angle in place\n-    of ``surface_tilt``.\n+    of ``surface_tilt``. The author of [1]_ recommends using one-half the\n+    table width for ``slant_height``, i.e., the distance from the tracker\n+    axis to the module edge.\n+\n+    The parameter `string_factor` is an enhancement added to the model after\n+    publication of [1]_ per private communication with the model's author.\n \n     References\n     ----------\n@@ -273,13 +284,22 @@ def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n        :doi:`10.1109/PVSC.2011.6186627`\n     '''\n \n+    # unit conversions from cm and m to in, from C to K, and from % to fraction\n+    # doing this early to facilitate comparison of this code with [1]\n+    snow_total_inches = snow_total / 2.54  # to inches\n+    relative_humidity_fraction = relative_humidity / 100.\n+    poa_global_kWh = poa_global / 1000.\n+    slant_height_inches = slant_height * 39.37\n+    lower_edge_height_inches = lower_edge_height * 39.37\n+    temp_air_kelvin = temp_air + 273.15\n+\n     C1 = 5.7e04\n     C2 = 0.51\n \n-    snow_total_prev = np.roll(snow_total, 1)\n+    snow_total_prev = np.roll(snow_total_inches, 1)\n     snow_events_prev = np.roll(snow_events, 1)\n \n-    effective_snow = _townsend_effective_snow(snow_total, snow_events)\n+    effective_snow = _townsend_effective_snow(snow_total_inches, snow_events)\n     effective_snow_prev = _townsend_effective_snow(\n         snow_total_prev,\n         snow_events_prev\n@@ -288,37 +308,38 @@ def loss_townsend(snow_total, snow_events, surface_tilt, relative_humidity,\n         1 / 3 * effective_snow_prev\n         + 2 / 3 * effective_snow\n     )\n-    effective_snow_weighted_m = effective_snow_weighted / 100\n \n-    lower_edge_height_clipped = np.maximum(lower_edge_height, 0.01)\n+    # the lower limit of 0.1 in^2 is per private communication with the model's\n+    # author. CWH 1/30/2023\n+    lower_edge_distance = np.clip(\n+        lower_edge_height_inches**2 - effective_snow_weighted**2, a_min=0.1,\n+        a_max=None)\n     gamma = (\n-        slant_height\n-        * effective_snow_weighted_m\n+        slant_height_inches\n+        * effective_snow_weighted\n         * cosd(surface_tilt)\n-        / (lower_edge_height_clipped**2 - effective_snow_weighted_m**2)\n+        / lower_edge_distance\n         * 2\n         * tand(angle_of_repose)\n     )\n \n     ground_interference_term = 1 - C2 * np.exp(-gamma)\n-    relative_humidity_fraction = relative_humidity / 100\n-    temp_air_kelvin = temp_air + 273.15\n-    effective_snow_weighted_in = effective_snow_weighted / 2.54\n-    poa_global_kWh = poa_global / 1000\n \n     # Calculate Eqn. 3 in the reference.\n     # Although the reference says Eqn. 3 calculates percentage loss, the y-axis\n     # of Figure 7 indicates Eqn. 3 calculates fractional loss. Since the slope\n     # of the line in Figure 7 is the same as C1 in Eqn. 3, it is assumed that\n     # Eqn. 3 calculates fractional loss.\n+\n     loss_fraction = (\n         C1\n-        * effective_snow_weighted_in\n+        * effective_snow_weighted\n         * cosd(surface_tilt)**2\n         * ground_interference_term\n         * relative_humidity_fraction\n         / temp_air_kelvin**2\n         / poa_global_kWh**0.67\n+        * string_factor\n     )\n \n     return np.clip(loss_fraction, 0, 1)\n", "test_patch": "diff --git a/pvlib/tests/test_snow.py b/pvlib/tests/test_snow.py\n--- a/pvlib/tests/test_snow.py\n+++ b/pvlib/tests/test_snow.py\n@@ -6,6 +6,8 @@\n from pvlib import snow\n from pvlib.tools import sind\n \n+import pytest\n+\n \n def test_fully_covered_nrel():\n     dt = pd.date_range(start=\"2019-1-1 12:00:00\", end=\"2019-1-1 18:00:00\",\n@@ -108,6 +110,7 @@ def test__townsend_effective_snow():\n \n \n def test_loss_townsend():\n+    # hand-calculated solution\n     snow_total = np.array([25.4, 25.4, 12.7, 2.54, 0, 0, 0, 0, 0, 0, 12.7,\n                            25.4])\n     snow_events = np.array([2, 2, 1, 0, 0, 0, 0, 0, 0, 0, 2, 3])\n@@ -118,6 +121,7 @@ def test_loss_townsend():\n     poa_global = np.array([350000, 350000, 350000, 350000, 350000, 350000,\n                            350000, 350000, 350000, 350000, 350000, 350000])\n     angle_of_repose = 40\n+    string_factor = 1.0\n     slant_height = 2.54\n     lower_edge_height = 0.254\n     expected = np.array([0.07696253, 0.07992262, 0.06216201, 0.01715392, 0, 0,\n@@ -125,5 +129,84 @@ def test_loss_townsend():\n     actual = snow.loss_townsend(snow_total, snow_events, surface_tilt,\n                                 relative_humidity, temp_air,\n                                 poa_global, slant_height,\n-                                lower_edge_height, angle_of_repose)\n+                                lower_edge_height, string_factor,\n+                                angle_of_repose)\n     np.testing.assert_allclose(expected, actual, rtol=1e-05)\n+\n+\n+@pytest.mark.parametrize(\n+    'poa_global,surface_tilt,slant_height,lower_edge_height,string_factor,expected',  # noQA: E501\n+    [\n+        (np.asarray(\n+            [60., 80., 100., 125., 175., 225., 225., 210., 175., 125., 90.,\n+             60.], dtype=float) * 1000.,\n+         2.,\n+         79. / 39.37,\n+         3. / 39.37,\n+         1.0,\n+         np.asarray(\n+            [44, 34, 20, 9, 3, 1, 0, 0, 0, 2, 6, 25], dtype=float)\n+         ),\n+        (np.asarray(\n+            [60., 80., 100., 125., 175., 225., 225., 210., 175., 125., 90.,\n+             60.], dtype=float) * 1000.,\n+         5.,\n+         316 / 39.37,\n+         120. / 39.37,\n+         0.75,\n+         np.asarray(\n+            [22, 16, 9, 4, 1, 0, 0, 0, 0, 1, 2, 12], dtype=float)\n+         ),\n+        (np.asarray(\n+            [60., 80., 100., 125., 175., 225., 225., 210., 175., 125., 90.,\n+             60.], dtype=float) * 1000.,\n+         23.,\n+         158 / 39.27,\n+         12 / 39.37,\n+         0.75,\n+         np.asarray(\n+            [28, 21, 13, 6, 2, 0, 0, 0, 0, 1, 4, 16], dtype=float)\n+         ),\n+        (np.asarray(\n+            [80., 100., 125., 150., 225., 300., 300., 275., 225., 150., 115.,\n+             80.], dtype=float) * 1000.,\n+         52.,\n+         39.5 / 39.37,\n+         34. / 39.37,\n+         0.75,\n+         np.asarray(\n+             [7, 5, 3, 1, 0, 0, 0, 0, 0, 0, 1, 4], dtype=float)\n+         ),\n+        (np.asarray(\n+            [80., 100., 125., 150., 225., 300., 300., 275., 225., 150., 115.,\n+             80.], dtype=float) * 1000.,\n+         60.,\n+         39.5 / 39.37,\n+         25. / 39.37,\n+         1.,\n+         np.asarray(\n+             [7, 5, 3, 1, 0, 0, 0, 0, 0, 0, 1, 3], dtype=float)\n+         )\n+    ]\n+)\n+def test_loss_townsend_cases(poa_global, surface_tilt, slant_height,\n+                             lower_edge_height, string_factor, expected):\n+    # test cases from Townsend, 1/27/2023, addeed by cwh\n+    # snow_total in inches, convert to cm for pvlib\n+    snow_total = np.asarray(\n+        [20, 15, 10, 4, 1.5, 0, 0, 0, 0, 1.5, 4, 15], dtype=float) * 2.54\n+    # snow events are an average for each month\n+    snow_events = np.asarray(\n+        [5, 4.2, 2.8, 1.3, 0.8, 0, 0, 0, 0, 0.5, 1.5, 4.5], dtype=float)\n+    # air temperature in C\n+    temp_air = np.asarray(\n+        [-6., -2., 1., 4., 7., 10., 13., 16., 14., 12., 7., -3.], dtype=float)\n+    # relative humidity in %\n+    relative_humidity = np.asarray(\n+        [78., 80., 75., 65., 60., 55., 55., 55., 50., 55., 60., 70.],\n+        dtype=float)\n+    actual = snow.loss_townsend(\n+        snow_total, snow_events, surface_tilt, relative_humidity, temp_air,\n+        poa_global, slant_height, lower_edge_height, string_factor)\n+    actual = np.around(actual * 100)\n+    assert np.allclose(expected, actual)\n", "problem_statement": "Corrections to Townsend snow model\nPrivate communications with the model's author have turned up some issues with the pvlib implementation. Chief among the issues is  this part of the calculation:\r\n\r\n```\r\n    lower_edge_height_clipped = np.maximum(lower_edge_height, 0.01)\r\n    gamma = (\r\n        slant_height\r\n        * effective_snow_weighted_m\r\n        * cosd(surface_tilt)\r\n        / (lower_edge_height_clipped**2 - effective_snow_weighted_m**2)\r\n        * 2\r\n        * tand(angle_of_repose)\r\n    )\r\n\r\n    ground_interference_term = 1 - C2 * np.exp(-gamma)\r\n```\r\n\r\nWhen `lower_edge_height_clipped` < `effective_snow_weighted_m`, `gamma` < 0 and the `ground_interference_term` can become negative. In contrast, the author's intent is that C2 < `ground_interference_terms` < 1. The author recommends clipping the squared difference (lower bound being worked out but will be something like 0.01.).\r\n\r\nOther issues appear to arise from the unit conversions. The published model uses inches for distance and snow depth. The pvlib code uses cm for snow depth (convenience for working with external snow data) and m for distances (for consistency with the rest of pvlib). After several steps, including the `ground_interference_term` calculation, the code converts from cm or m to inches to apply the final formula for loss (since the formula involves some coefficients determined by a regression). It would be easier to trace the pvlib code back to the paper if the internal unit conversions (from cm / m to inches) were done earlier.\r\n\n", "hints_text": "Interestingly, clamping the difference of squares at 0.01 was part of the original PR but got lost along the way: https://github.com/pvlib/pvlib-python/pull/1251#discussion_r830258000\r\n\r\nIt would be great if the communication with the author results in improved tests as well as improved code.\nAfter communications with the author, the pvlib code is missing two items:\r\n\r\n- a lower bound on `lower_edge_height_clipped**2 - effective_snow_weighted_m**2` which the author specifies should be 0.1 in^2.\r\n- a factor that multiplies the monthly loss fraction, to represent the potential for a string of modules at the top of the slanted array to generate power while strings at lower positions are still affected by snow.  This factor was brought up in #1625 \r\n\r\nNeither is documented in the 2011 paper but should be added to 1) prevent unreasonably low loss values (item 1) and to better represent the loss for systems with multiple, horizontally-oriented strings.\r\n\r\nAlso, the author recommends advising users to enter 1/2 the total module width as the slant_height for single-axis tracked systems, which makes sense to me, as snow could slide off either surface depending on its rotation. ", "created_at": "2023-01-28T00:30:44Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_snow.py::test_loss_townsend", "pvlib/tests/test_snow.py::test_loss_townsend_cases[poa_global0-2.0-2.0066040132080265-0.07620015240030481-1.0-expected0]", "pvlib/tests/test_snow.py::test_loss_townsend_cases[poa_global1-5.0-8.026416052832106-3.0480060960121924-0.75-expected1]", "pvlib/tests/test_snow.py::test_loss_townsend_cases[poa_global2-23.0-4.023427552839317-0.30480060960121924-0.75-expected2]", "pvlib/tests/test_snow.py::test_loss_townsend_cases[poa_global4-60.0-1.0033020066040133-0.63500127000254-1.0-expected4]"], "PASS_TO_PASS": ["pvlib/tests/test_snow.py::test_fully_covered_nrel", "pvlib/tests/test_snow.py::test_coverage_nrel_hourly", "pvlib/tests/test_snow.py::test_coverage_nrel_subhourly", "pvlib/tests/test_snow.py::test_fully_covered_nrel_irregular", "pvlib/tests/test_snow.py::test_coverage_nrel_initial", "pvlib/tests/test_snow.py::test_dc_loss_nrel", "pvlib/tests/test_snow.py::test__townsend_effective_snow", "pvlib/tests/test_snow.py::test_loss_townsend_cases[poa_global3-52.0-1.0033020066040133-0.8636017272034545-0.75-expected3]"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1236", "base_commit": "c2c6145d679adc97924d6c8a761a50b8e2819e3f", "patch": "diff --git a/pydicom/pixel_data_handlers/__init__.py b/pydicom/pixel_data_handlers/__init__.py\n--- a/pydicom/pixel_data_handlers/__init__.py\n+++ b/pydicom/pixel_data_handlers/__init__.py\n@@ -1,7 +1,7 @@\n \n from pydicom.pixel_data_handlers.util import (\n     apply_color_lut, apply_modality_lut, apply_voi_lut, convert_color_space,\n+    apply_voi, apply_windowing\n )\n \n apply_rescale = apply_modality_lut\n-apply_windowing = apply_voi_lut\ndiff --git a/pydicom/pixel_data_handlers/util.py b/pydicom/pixel_data_handlers/util.py\n--- a/pydicom/pixel_data_handlers/util.py\n+++ b/pydicom/pixel_data_handlers/util.py\n@@ -3,7 +3,7 @@\n \n from struct import unpack\n from sys import byteorder\n-from typing import Dict\n+from typing import Dict, Optional, Union, List, Tuple, TYPE_CHECKING, cast\n import warnings\n \n try:\n@@ -15,8 +15,15 @@\n from pydicom.data import get_palette_files\n from pydicom.uid import UID\n \n+if TYPE_CHECKING:\n+    from pydicom.dataset import Dataset\n \n-def apply_color_lut(arr, ds=None, palette=None):\n+\n+def apply_color_lut(\n+    arr: \"np.ndarray\",\n+    ds: Optional[\"Dataset\"] = None,\n+    palette: Optional[Union[str, UID]] = None\n+) -> \"np.ndarray\":\n     \"\"\"Apply a color palette lookup table to `arr`.\n \n     .. versionadded:: 1.4\n@@ -187,7 +194,7 @@ def apply_color_lut(arr, ds=None, palette=None):\n     return out\n \n \n-def apply_modality_lut(arr, ds):\n+def apply_modality_lut(arr: \"np.ndarray\", ds: \"Dataset\") -> \"np.ndarray\":\n     \"\"\"Apply a modality lookup table or rescale operation to `arr`.\n \n     .. versionadded:: 1.4\n@@ -261,11 +268,20 @@ def apply_modality_lut(arr, ds):\n     return arr\n \n \n-def apply_voi_lut(arr, ds, index=0):\n+def apply_voi_lut(\n+    arr: \"np.ndarray\",\n+    ds: \"Dataset\",\n+    index: int = 0,\n+    prefer_lut: bool = True\n+) -> \"np.ndarray\":\n     \"\"\"Apply a VOI lookup table or windowing operation to `arr`.\n \n     .. versionadded:: 1.4\n \n+    .. versionchanged:: 2.1\n+\n+        Added the `prefer_lut` keyword parameter\n+\n     Parameters\n     ----------\n     arr : numpy.ndarray\n@@ -280,8 +296,12 @@ def apply_voi_lut(arr, ds, index=0):\n         ``np.float64``. If neither are present then `arr` will be returned\n         unchanged.\n     index : int, optional\n-        Where the VOI LUT Module contains multiple possible views, this is\n+        When the VOI LUT Module contains multiple alternative views, this is\n         the index of the view to return (default ``0``).\n+    prefer_lut : bool\n+        When the VOI LUT Module contains both *Window Width*/*Window Center*\n+        and *VOI LUT Sequence*, if ``True`` (default) then apply the VOI LUT,\n+        otherwise apply the windowing operation.\n \n     Returns\n     -------\n@@ -297,6 +317,8 @@ def apply_voi_lut(arr, ds, index=0):\n     See Also\n     --------\n     :func:`~pydicom.pixel_data_handlers.util.apply_modality_lut`\n+    :func:`~pydicom.pixel_data_handlers.util.apply_voi`\n+    :func:`~pydicom.pixel_data_handlers.util.apply_windowing`\n \n     References\n     ----------\n@@ -307,134 +329,253 @@ def apply_voi_lut(arr, ds, index=0):\n     * DICOM Standard, Part 4, :dcm:`Annex N.2.1.1\n       <part04/sect_N.2.html#sect_N.2.1.1>`\n     \"\"\"\n+    valid_voi = False\n     if 'VOILUTSequence' in ds:\n-        if not np.issubdtype(arr.dtype, np.integer):\n-            warnings.warn(\n-                \"Applying a VOI LUT on a float input array may give \"\n-                \"incorrect results\"\n-            )\n+        valid_voi = None not in [\n+            ds.VOILUTSequence[0].get('LUTDescriptor', None),\n+            ds.VOILUTSequence[0].get('LUTData', None)\n+        ]\n+    valid_windowing = None not in [\n+        ds.get('WindowCenter', None),\n+        ds.get('WindowWidth', None)\n+    ]\n \n-        # VOI LUT Sequence contains one or more items\n-        item = ds.VOILUTSequence[index]\n-        nr_entries = item.LUTDescriptor[0] or 2**16\n-        first_map = item.LUTDescriptor[1]\n+    if valid_voi and valid_windowing:\n+        if prefer_lut:\n+            return apply_voi(arr, ds, index)\n \n-        # PS3.3 C.8.11.3.1.5: may be 8, 10-16\n-        nominal_depth = item.LUTDescriptor[2]\n-        if nominal_depth in list(range(10, 17)):\n-            dtype = 'uint16'\n-        elif nominal_depth == 8:\n-            dtype = 'uint8'\n-        else:\n-            raise NotImplementedError(\n-                \"'{}' bits per LUT entry is not supported\"\n-                .format(nominal_depth)\n-            )\n+        return apply_windowing(arr, ds, index)\n \n-        # Ambiguous VR, US or OW\n-        if item['LUTData'].VR == 'OW':\n-            endianness = '<' if ds.is_little_endian else '>'\n-            unpack_fmt = '{}{}H'.format(endianness, nr_entries)\n-            lut_data = unpack(unpack_fmt, item.LUTData)\n-        else:\n-            lut_data = item.LUTData\n-        lut_data = np.asarray(lut_data, dtype=dtype)\n+    if valid_voi:\n+        return apply_voi(arr, ds, index)\n \n-        # IVs < `first_map` get set to first LUT entry (i.e. index 0)\n-        clipped_iv = np.zeros(arr.shape, dtype=dtype)\n-        # IVs >= `first_map` are mapped by the VOI LUT\n-        # `first_map` may be negative, positive or 0\n-        mapped_pixels = arr >= first_map\n-        clipped_iv[mapped_pixels] = arr[mapped_pixels] - first_map\n-        # IVs > number of entries get set to last entry\n-        np.clip(clipped_iv, 0, nr_entries - 1, out=clipped_iv)\n+    if valid_windowing:\n+        return apply_windowing(arr, ds, index)\n \n-        return lut_data[clipped_iv]\n-    elif 'WindowCenter' in ds and 'WindowWidth' in ds:\n-        if ds.PhotometricInterpretation not in ['MONOCHROME1', 'MONOCHROME2']:\n-            raise ValueError(\n-                \"When performing a windowing operation only 'MONOCHROME1' and \"\n-                \"'MONOCHROME2' are allowed for (0028,0004) Photometric \"\n-                \"Interpretation\"\n-            )\n+    return arr\n \n-        # May be LINEAR (default), LINEAR_EXACT, SIGMOID or not present, VM 1\n-        voi_func = getattr(ds, 'VOILUTFunction', 'LINEAR').upper()\n-        # VR DS, VM 1-n\n-        elem = ds['WindowCenter']\n-        center = elem.value[index] if elem.VM > 1 else elem.value\n-        elem = ds['WindowWidth']\n-        width = elem.value[index] if elem.VM > 1 else elem.value\n-\n-        # The output range depends on whether or not a modality LUT or rescale\n-        #   operation has been applied\n-        if 'ModalityLUTSequence' in ds:\n-            # Unsigned - see PS3.3 C.11.1.1.1\n-            y_min = 0\n-            bit_depth = ds.ModalityLUTSequence[0].LUTDescriptor[2]\n-            y_max = 2**bit_depth - 1\n-        elif ds.PixelRepresentation == 0:\n-            # Unsigned\n-            y_min = 0\n-            y_max = 2**ds.BitsStored - 1\n-        else:\n-            # Signed\n-            y_min = -2**(ds.BitsStored - 1)\n-            y_max = 2**(ds.BitsStored - 1) - 1\n-\n-        if 'RescaleSlope' in ds and 'RescaleIntercept' in ds:\n-            # Otherwise its the actual data range\n-            y_min = y_min * ds.RescaleSlope + ds.RescaleIntercept\n-            y_max = y_max * ds.RescaleSlope + ds.RescaleIntercept\n-\n-        y_range = y_max - y_min\n-        arr = arr.astype('float64')\n-\n-        if voi_func in ['LINEAR', 'LINEAR_EXACT']:\n-            # PS3.3 C.11.2.1.2.1 and C.11.2.1.3.2\n-            if voi_func == 'LINEAR':\n-                if width < 1:\n-                    raise ValueError(\n-                        \"The (0028,1051) Window Width must be greater than or \"\n-                        \"equal to 1 for a 'LINEAR' windowing operation\"\n-                    )\n-                center -= 0.5\n-                width -= 1\n-            elif width <= 0:\n-                raise ValueError(\n-                    \"The (0028,1051) Window Width must be greater than 0 \"\n-                    \"for a 'LINEAR_EXACT' windowing operation\"\n-                )\n \n-            below = arr <= (center - width / 2)\n-            above = arr > (center + width / 2)\n-            between = np.logical_and(~below, ~above)\n+def apply_voi(\n+    arr: \"np.ndarray\", ds: \"Dataset\", index: int = 0\n+) -> \"np.ndarray\":\n+    \"\"\"Apply a VOI lookup table to `arr`.\n \n-            arr[below] = y_min\n-            arr[above] = y_max\n-            if between.any():\n-                arr[between] = (\n-                    ((arr[between] - center) / width + 0.5) * y_range + y_min\n-                )\n-        elif voi_func == 'SIGMOID':\n-            # PS3.3 C.11.2.1.3.1\n-            if width <= 0:\n+    .. versionadded:: 2.1\n+\n+    Parameters\n+    ----------\n+    arr : numpy.ndarray\n+        The :class:`~numpy.ndarray` to apply the VOI LUT to.\n+    ds : dataset.Dataset\n+        A dataset containing a :dcm:`VOI LUT Module<part03/sect_C.11.2.html>`.\n+        If (0028,3010) *VOI LUT Sequence* is present then returns an array\n+        of ``np.uint8`` or ``np.uint16``, depending on the 3rd value of\n+        (0028,3002) *LUT Descriptor*, otherwise `arr` will be returned\n+        unchanged.\n+    index : int, optional\n+        When the VOI LUT Module contains multiple alternative views, this is\n+        the index of the view to return (default ``0``).\n+\n+    Returns\n+    -------\n+    numpy.ndarray\n+        An array with applied VOI LUT.\n+\n+    See Also\n+    --------\n+    :func:`~pydicom.pixel_data_handlers.util.apply_modality_lut`\n+    :func:`~pydicom.pixel_data_handlers.util.apply_windowing`\n+\n+    References\n+    ----------\n+    * DICOM Standard, Part 3, :dcm:`Annex C.11.2\n+      <part03/sect_C.11.html#sect_C.11.2>`\n+    * DICOM Standard, Part 3, :dcm:`Annex C.8.11.3.1.5\n+      <part03/sect_C.8.11.3.html#sect_C.8.11.3.1.5>`\n+    * DICOM Standard, Part 4, :dcm:`Annex N.2.1.1\n+      <part04/sect_N.2.html#sect_N.2.1.1>`\n+    \"\"\"\n+    if \"VOILUTSequence\" not in ds:\n+        return arr\n+\n+    if not np.issubdtype(arr.dtype, np.integer):\n+        warnings.warn(\n+            \"Applying a VOI LUT on a float input array may give \"\n+            \"incorrect results\"\n+        )\n+\n+    # VOI LUT Sequence contains one or more items\n+    item = ds.VOILUTSequence[index]\n+    nr_entries = item.LUTDescriptor[0] or 2**16\n+    first_map = item.LUTDescriptor[1]\n+\n+    # PS3.3 C.8.11.3.1.5: may be 8, 10-16\n+    nominal_depth = item.LUTDescriptor[2]\n+    if nominal_depth in list(range(10, 17)):\n+        dtype = 'uint16'\n+    elif nominal_depth == 8:\n+        dtype = 'uint8'\n+    else:\n+        raise NotImplementedError(\n+            f\"'{nominal_depth}' bits per LUT entry is not supported\"\n+        )\n+\n+    # Ambiguous VR, US or OW\n+    if item['LUTData'].VR == 'OW':\n+        endianness = '<' if ds.is_little_endian else '>'\n+        unpack_fmt = f'{endianness}{nr_entries}H'\n+        lut_data = unpack(unpack_fmt, item.LUTData)\n+    else:\n+        lut_data = item.LUTData\n+    lut_data = np.asarray(lut_data, dtype=dtype)\n+\n+    # IVs < `first_map` get set to first LUT entry (i.e. index 0)\n+    clipped_iv = np.zeros(arr.shape, dtype=dtype)\n+    # IVs >= `first_map` are mapped by the VOI LUT\n+    # `first_map` may be negative, positive or 0\n+    mapped_pixels = arr >= first_map\n+    clipped_iv[mapped_pixels] = arr[mapped_pixels] - first_map\n+    # IVs > number of entries get set to last entry\n+    np.clip(clipped_iv, 0, nr_entries - 1, out=clipped_iv)\n+\n+    return lut_data[clipped_iv]\n+\n+\n+def apply_windowing(\n+    arr: \"np.ndarray\", ds: \"Dataset\", index: int = 0\n+) -> \"np.ndarray\":\n+    \"\"\"Apply a windowing operation to `arr`.\n+\n+    .. versionadded:: 2.1\n+\n+    Parameters\n+    ----------\n+    arr : numpy.ndarray\n+        The :class:`~numpy.ndarray` to apply the windowing operation to.\n+    ds : dataset.Dataset\n+        A dataset containing a :dcm:`VOI LUT Module<part03/sect_C.11.2.html>`.\n+        If (0028,1050) *Window Center* and (0028,1051) *Window Width* are\n+        present then returns an array of ``np.float64``, otherwise `arr` will\n+        be returned unchanged.\n+    index : int, optional\n+        When the VOI LUT Module contains multiple alternative views, this is\n+        the index of the view to return (default ``0``).\n+\n+    Returns\n+    -------\n+    numpy.ndarray\n+        An array with applied windowing operation.\n+\n+    Notes\n+    -----\n+    When the dataset requires a modality LUT or rescale operation as part of\n+    the Modality LUT module then that must be applied before any windowing\n+    operation.\n+\n+    See Also\n+    --------\n+    :func:`~pydicom.pixel_data_handlers.util.apply_modality_lut`\n+    :func:`~pydicom.pixel_data_handlers.util.apply_voi`\n+\n+    References\n+    ----------\n+    * DICOM Standard, Part 3, :dcm:`Annex C.11.2\n+      <part03/sect_C.11.html#sect_C.11.2>`\n+    * DICOM Standard, Part 3, :dcm:`Annex C.8.11.3.1.5\n+      <part03/sect_C.8.11.3.html#sect_C.8.11.3.1.5>`\n+    * DICOM Standard, Part 4, :dcm:`Annex N.2.1.1\n+      <part04/sect_N.2.html#sect_N.2.1.1>`\n+    \"\"\"\n+    if \"WindowWidth\" not in ds and \"WindowCenter\" not in ds:\n+        return arr\n+\n+    if ds.PhotometricInterpretation not in ['MONOCHROME1', 'MONOCHROME2']:\n+        raise ValueError(\n+            \"When performing a windowing operation only 'MONOCHROME1' and \"\n+            \"'MONOCHROME2' are allowed for (0028,0004) Photometric \"\n+            \"Interpretation\"\n+        )\n+\n+    # May be LINEAR (default), LINEAR_EXACT, SIGMOID or not present, VM 1\n+    voi_func = cast(str, getattr(ds, 'VOILUTFunction', 'LINEAR')).upper()\n+    # VR DS, VM 1-n\n+    elem = ds['WindowCenter']\n+    center = elem.value[index] if elem.VM > 1 else elem.value\n+    elem = ds['WindowWidth']\n+    width = elem.value[index] if elem.VM > 1 else elem.value\n+\n+    # The output range depends on whether or not a modality LUT or rescale\n+    #   operation has been applied\n+    if 'ModalityLUTSequence' in ds:\n+        # Unsigned - see PS3.3 C.11.1.1.1\n+        y_min = 0\n+        bit_depth = ds.ModalityLUTSequence[0].LUTDescriptor[2]\n+        y_max = 2**bit_depth - 1\n+    elif ds.PixelRepresentation == 0:\n+        # Unsigned\n+        y_min = 0\n+        y_max = 2**ds.BitsStored - 1\n+    else:\n+        # Signed\n+        y_min = -2**(ds.BitsStored - 1)\n+        y_max = 2**(ds.BitsStored - 1) - 1\n+\n+    slope = ds.get('RescaleSlope', None)\n+    intercept = ds.get('RescaleIntercept', None)\n+    if slope is not None and intercept is not None:\n+        # Otherwise its the actual data range\n+        y_min = y_min * ds.RescaleSlope + ds.RescaleIntercept\n+        y_max = y_max * ds.RescaleSlope + ds.RescaleIntercept\n+\n+    y_range = y_max - y_min\n+    arr = arr.astype('float64')\n+\n+    if voi_func in ['LINEAR', 'LINEAR_EXACT']:\n+        # PS3.3 C.11.2.1.2.1 and C.11.2.1.3.2\n+        if voi_func == 'LINEAR':\n+            if width < 1:\n                 raise ValueError(\n-                    \"The (0028,1051) Window Width must be greater than 0 \"\n-                    \"for a 'SIGMOID' windowing operation\"\n+                    \"The (0028,1051) Window Width must be greater than or \"\n+                    \"equal to 1 for a 'LINEAR' windowing operation\"\n                 )\n+            center -= 0.5\n+            width -= 1\n+        elif width <= 0:\n+            raise ValueError(\n+                \"The (0028,1051) Window Width must be greater than 0 \"\n+                \"for a 'LINEAR_EXACT' windowing operation\"\n+            )\n \n-            arr = y_range / (1 + np.exp(-4 * (arr - center) / width)) + y_min\n-        else:\n+        below = arr <= (center - width / 2)\n+        above = arr > (center + width / 2)\n+        between = np.logical_and(~below, ~above)\n+\n+        arr[below] = y_min\n+        arr[above] = y_max\n+        if between.any():\n+            arr[between] = (\n+                ((arr[between] - center) / width + 0.5) * y_range + y_min\n+            )\n+    elif voi_func == 'SIGMOID':\n+        # PS3.3 C.11.2.1.3.1\n+        if width <= 0:\n             raise ValueError(\n-                \"Unsupported (0028,1056) VOI LUT Function value '{}'\"\n-                .format(voi_func)\n+                \"The (0028,1051) Window Width must be greater than 0 \"\n+                \"for a 'SIGMOID' windowing operation\"\n             )\n \n+        arr = y_range / (1 + np.exp(-4 * (arr - center) / width)) + y_min\n+    else:\n+        raise ValueError(\n+            f\"Unsupported (0028,1056) VOI LUT Function value '{voi_func}'\"\n+        )\n+\n     return arr\n \n \n-def convert_color_space(arr, current, desired):\n+def convert_color_space(\n+    arr: \"np.ndarray\", current: str, desired: str\n+) -> \"np.ndarray\":\n     \"\"\"Convert the image(s) in `arr` from one color space to another.\n \n     .. versionchanged:: 1.4\n@@ -501,7 +642,7 @@ def _no_change(arr):\n     return converter(arr)\n \n \n-def _convert_RGB_to_YBR_FULL(arr):\n+def _convert_RGB_to_YBR_FULL(arr: \"np.ndarray\") -> \"np.ndarray\":\n     \"\"\"Return an ndarray converted from RGB to YBR_FULL color space.\n \n     Parameters\n@@ -543,7 +684,7 @@ def _convert_RGB_to_YBR_FULL(arr):\n     return arr.astype(orig_dtype)\n \n \n-def _convert_YBR_FULL_to_RGB(arr):\n+def _convert_YBR_FULL_to_RGB(arr: \"np.ndarray\") -> \"np.ndarray\":\n     \"\"\"Return an ndarray converted from YBR_FULL to RGB color space.\n \n     Parameters\n@@ -585,7 +726,9 @@ def _convert_YBR_FULL_to_RGB(arr):\n     return arr.astype(orig_dtype)\n \n \n-def dtype_corrected_for_endianness(is_little_endian, numpy_dtype):\n+def dtype_corrected_for_endianness(\n+    is_little_endian: bool, numpy_dtype: \"np.dtype\"\n+) -> \"np.dtype\":\n     \"\"\"Return a :class:`numpy.dtype` corrected for system and :class:`Dataset`\n     endianness.\n \n@@ -618,7 +761,12 @@ def dtype_corrected_for_endianness(is_little_endian, numpy_dtype):\n     return numpy_dtype\n \n \n-def _expand_segmented_lut(data, fmt, nr_segments=None, last_value=None):\n+def _expand_segmented_lut(\n+    data: Tuple[int, ...],\n+    fmt: str,\n+    nr_segments: Optional[int] = None,\n+    last_value: Optional[int] = None\n+) -> List[int]:\n     \"\"\"Return a list containing the expanded lookup table data.\n \n     Parameters\n@@ -652,7 +800,7 @@ def _expand_segmented_lut(data, fmt, nr_segments=None, last_value=None):\n     # Little endian: e.g. 0x0302 0x0100, big endian, e.g. 0x0203 0x0001\n     indirect_ii = [3, 2, 1, 0] if '<' in fmt else [2, 3, 0, 1]\n \n-    lut = []\n+    lut: List[int] = []\n     offset = 0\n     segments_read = 0\n     # Use `offset + 1` to account for possible trailing null\n@@ -722,7 +870,7 @@ def _expand_segmented_lut(data, fmt, nr_segments=None, last_value=None):\n     return lut\n \n \n-def get_expected_length(ds, unit='bytes'):\n+def get_expected_length(ds: \"Dataset\", unit: str = 'bytes') -> int:\n     \"\"\"Return the expected length (in terms of bytes or pixels) of the *Pixel\n     Data*.\n \n@@ -765,14 +913,14 @@ def get_expected_length(ds, unit='bytes'):\n         The expected length of the *Pixel Data* in either whole bytes or\n         pixels, excluding the NULL trailing padding byte for odd length data.\n     \"\"\"\n-    length = ds.Rows * ds.Columns * ds.SamplesPerPixel\n+    length: int = ds.Rows * ds.Columns * ds.SamplesPerPixel\n     length *= get_nr_frames(ds)\n \n     if unit == 'pixels':\n         return length\n \n     # Correct for the number of bytes per pixel\n-    bits_allocated = ds.BitsAllocated\n+    bits_allocated = cast(int, ds.BitsAllocated)\n     if bits_allocated == 1:\n         # Determine the nearest whole number of bytes needed to contain\n         #   1-bit pixel data. e.g. 10 x 10 1-bit pixels is 100 bits, which\n@@ -788,7 +936,7 @@ def get_expected_length(ds, unit='bytes'):\n     return length\n \n \n-def get_image_pixel_ids(ds) -> Dict[str, int]:\n+def get_image_pixel_ids(ds: \"Dataset\") -> Dict[str, int]:\n     \"\"\"Return a dict of the pixel data affecting element's :func:`id` values.\n \n     .. versionadded:: 1.4\n@@ -845,7 +993,7 @@ def get_image_pixel_ids(ds) -> Dict[str, int]:\n     return {kw: id(getattr(ds, kw, None)) for kw in keywords}\n \n \n-def get_j2k_parameters(codestream):\n+def get_j2k_parameters(codestream: bytes) -> Dict[str, object]:\n     \"\"\"Return a dict containing JPEG 2000 component parameters.\n \n     .. versionadded:: 2.1\n@@ -883,7 +1031,7 @@ def get_j2k_parameters(codestream):\n     return {}\n \n \n-def get_nr_frames(ds):\n+def get_nr_frames(ds: \"Dataset\") -> int:\n     \"\"\"Return NumberOfFrames or 1 if NumberOfFrames is None.\n \n     Parameters\n@@ -897,7 +1045,7 @@ def get_nr_frames(ds):\n     int\n         An integer for the NumberOfFrames or 1 if NumberOfFrames is None\n     \"\"\"\n-    nr_frames = getattr(ds, 'NumberOfFrames', 1)\n+    nr_frames: Optional[int] = getattr(ds, 'NumberOfFrames', 1)\n     # 'NumberOfFrames' may exist in the DICOM file but have value equal to None\n     if nr_frames is None:\n         warnings.warn(\"A value of None for (0028,0008) 'Number of Frames' is \"\n@@ -908,7 +1056,7 @@ def get_nr_frames(ds):\n     return nr_frames\n \n \n-def pixel_dtype(ds, as_float=False):\n+def pixel_dtype(ds: \"Dataset\", as_float: bool = False) -> \"np.dtype\":\n     \"\"\"Return a :class:`numpy.dtype` for the pixel data in `ds`.\n \n     Suitable for use with IODs containing the Image Pixel module (with\n@@ -1009,7 +1157,7 @@ def pixel_dtype(ds, as_float=False):\n     return dtype\n \n \n-def reshape_pixel_array(ds, arr):\n+def reshape_pixel_array(ds: \"Dataset\", arr: \"np.ndarray\") -> \"np.ndarray\":\n     \"\"\"Return a reshaped :class:`numpy.ndarray` `arr`.\n \n     +------------------------------------------+-----------+----------+\n", "test_patch": "diff --git a/pydicom/tests/test_handler_util.py b/pydicom/tests/test_handler_util.py\n--- a/pydicom/tests/test_handler_util.py\n+++ b/pydicom/tests/test_handler_util.py\n@@ -28,7 +28,9 @@\n     apply_modality_lut,\n     apply_voi_lut,\n     get_j2k_parameters,\n-    get_nr_frames\n+    get_nr_frames,\n+    apply_voi,\n+    apply_windowing\n )\n from pydicom.uid import (ExplicitVRLittleEndian, ImplicitVRLittleEndian,\n                          UncompressedPixelTransferSyntaxes)\n@@ -1412,191 +1414,8 @@ def test_unknown_opcode_raises(self):\n \n \n @pytest.mark.skipif(not HAVE_NP, reason=\"Numpy is not available\")\n-class TestNumpy_VOILUT:\n-    \"\"\"Tests for util.apply_voi_lut().\"\"\"\n-    def test_voi_single_view(self):\n-        \"\"\"Test VOI LUT with a single view.\"\"\"\n-        ds = dcmread(VOI_08_1F)\n-        assert 8 == ds.BitsAllocated\n-        assert 8 == ds.BitsStored\n-        assert 0 == ds.PixelRepresentation\n-        item = ds.VOILUTSequence[0]\n-        assert [256, 0, 16] == item.LUTDescriptor\n-        lut = item.LUTData\n-        assert 0 == lut[0]\n-        assert 19532 == lut[76]\n-        assert 45746 == lut[178]\n-        assert 65535 == lut[255]\n-\n-        arr = ds.pixel_array\n-        assert 0 == arr[387, 448]\n-        assert 76 == arr[178, 126]\n-        assert 178 == arr[186, 389]\n-        assert 255 == arr[129, 79]\n-\n-        out = apply_voi_lut(arr, ds)\n-        assert 0 == out[387, 448]\n-        assert 19532 == out[178, 126]\n-        assert 45746 == out[186, 389]\n-        assert 65535 == out[129, 79]\n-\n-    def test_voi_multi_view(self):\n-        \"\"\"Test VOI LUT with multiple views.\"\"\"\n-        ds = dcmread(VOI_08_1F)\n-        assert 8 == ds.BitsAllocated\n-        assert 8 == ds.BitsStored\n-        assert 0 == ds.PixelRepresentation\n-        item0 = ds.VOILUTSequence[0]\n-        # Add another view thats the inverse\n-        ds.VOILUTSequence.append(Dataset())\n-        item1 = ds.VOILUTSequence[1]\n-        item1.LUTDescriptor = [256, 0, 16]\n-        item1.LUTData = item0.LUTData[::-1]\n-\n-        arr = ds.pixel_array\n-        assert 0 == arr[387, 448]\n-        assert 76 == arr[178, 126]\n-        assert 178 == arr[186, 389]\n-        assert 255 == arr[129, 79]\n-\n-        out0 = apply_voi_lut(arr, ds)\n-        assert 0 == out0[387, 448]\n-        assert 19532 == out0[178, 126]\n-        assert 45746 == out0[186, 389]\n-        assert 65535 == out0[129, 79]\n-\n-        out1 = apply_voi_lut(arr, ds, index=1)\n-        assert 65535 == out1[387, 448]\n-        assert 46003 == out1[178, 126]\n-        assert 19789 == out1[186, 389]\n-        assert 0 == out1[129, 79]\n-\n-    def test_voi_multi_frame(self):\n-        \"\"\"Test VOI with a multiple frames.\"\"\"\n-        ds = dcmread(VOI_08_1F)\n-        assert 8 == ds.BitsAllocated\n-        assert 8 == ds.BitsStored\n-        assert 0 == ds.PixelRepresentation\n-\n-        arr = ds.pixel_array\n-        arr = np.stack([arr, 255 - arr])\n-        assert (2, 512, 512) == arr.shape\n-\n-        out = apply_voi_lut(arr, ds)\n-        assert 0 == out[0, 387, 448]\n-        assert 19532 == out[0, 178, 126]\n-        assert 45746 == out[0, 186, 389]\n-        assert 65535 == out[0, 129, 79]\n-        assert 65535 == out[1, 387, 448]\n-        assert 46003 == out[1, 178, 126]\n-        assert 19789 == out[1, 186, 389]\n-        assert 0 == out[1, 129, 79]\n-\n-    def test_voi_zero_entries(self):\n-        \"\"\"Test that 0 entries is interpreted correctly.\"\"\"\n-        ds = dcmread(VOI_08_1F)\n-        seq = ds.VOILUTSequence[0]\n-        seq.LUTDescriptor = [0, 0, 16]\n-        assert 256 == len(seq.LUTData)\n-        arr = np.asarray([0, 255, 256, 65535])\n-        msg = r\"index 256 is out of bounds\"\n-        with pytest.raises(IndexError, match=msg):\n-            apply_voi_lut(arr, ds)\n-\n-        # LUTData with 65536 entries\n-        seq.LUTData = [0] * 65535 + [1]\n-        out = apply_voi_lut(arr, ds)\n-        assert [0, 0, 0, 1] == list(out)\n-\n-    def test_voi_uint8(self):\n-        \"\"\"Test uint VOI LUT with an 8-bit LUT.\"\"\"\n-        ds = Dataset()\n-        ds.PixelRepresentation = 0\n-        ds.BitsStored = 8\n-        ds.VOILUTSequence = [Dataset()]\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor = [4, 0, 8]\n-        item.LUTData = [0, 127, 128, 255]\n-        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n-        out = apply_voi_lut(arr, ds)\n-        assert 'uint8' == out.dtype\n-        assert [0, 127, 255, 255, 255] == out.tolist()\n-\n-    def test_voi_uint16(self):\n-        \"\"\"Test uint VOI LUT with an 16-bit LUT.\"\"\"\n-        ds = Dataset()\n-        ds.PixelRepresentation = 0\n-        ds.BitsStored = 16\n-        ds.VOILUTSequence = [Dataset()]\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor = [4, 0, 16]\n-        item.LUTData = [0, 127, 32768, 65535]\n-        arr = np.asarray([0, 1, 2, 3, 255], dtype='uint16')\n-        out = apply_voi_lut(arr, ds)\n-        assert 'uint16' == out.dtype\n-        assert [0, 127, 32768, 65535, 65535] == out.tolist()\n-\n-    def test_voi_int8(self):\n-        \"\"\"Test int VOI LUT with an 8-bit LUT.\"\"\"\n-        ds = Dataset()\n-        ds.PixelRepresentation = 1\n-        ds.BitsStored = 8\n-        ds.VOILUTSequence = [Dataset()]\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor = [4, 0, 8]\n-        item.LUTData = [0, 127, 128, 255]\n-        arr = np.asarray([0, -1, 2, -128, 127], dtype='int8')\n-        out = apply_voi_lut(arr, ds)\n-        assert 'uint8' == out.dtype\n-        assert [0, 0, 128, 0, 255] == out.tolist()\n-\n-    def test_voi_int16(self):\n-        \"\"\"Test int VOI LUT with an 16-bit LUT.\"\"\"\n-        ds = Dataset()\n-        ds.PixelRepresentation = 0\n-        ds.BitsStored = 16\n-        ds.VOILUTSequence = [Dataset()]\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor = [4, 0, 16]\n-        item.LUTData = [0, 127, 32768, 65535]\n-        arr = np.asarray([0, -1, 2, -128, 255], dtype='int16')\n-        out = apply_voi_lut(arr, ds)\n-        assert 'uint16' == out.dtype\n-        assert [0, 0, 32768, 0, 65535] == out.tolist()\n-\n-    def test_voi_bad_depth(self):\n-        \"\"\"Test bad LUT depth raises exception.\"\"\"\n-        ds = dcmread(VOI_08_1F)\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor[2] = 7\n-        msg = r\"'7' bits per LUT entry is not supported\"\n-        with pytest.raises(NotImplementedError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n-\n-        item.LUTDescriptor[2] = 17\n-        msg = r\"'17' bits per LUT entry is not supported\"\n-        with pytest.raises(NotImplementedError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n-\n-    def test_voi_uint16_array_float(self):\n-        \"\"\"Test warning when array is float and VOI LUT with an 16-bit LUT\"\"\"\n-        ds = Dataset()\n-        ds.PixelRepresentation = 0\n-        ds.BitsStored = 16\n-        ds.VOILUTSequence = [Dataset()]\n-        item = ds.VOILUTSequence[0]\n-        item.LUTDescriptor = [4, 0, 16]\n-        item.LUTData = [0, 127, 32768, 65535]\n-        arr = np.asarray([0, 1, 2, 3, 255], dtype='float64')\n-        msg = (\n-            r\"Applying a VOI LUT on a float input array may give \"\n-            r\"incorrect results\"\n-        )\n-\n-        with pytest.warns(UserWarning, match=msg):\n-            out = apply_voi_lut(arr, ds)\n-            assert [0, 127, 32768, 65535, 65535] == out.tolist()\n-\n+class TestNumpy_ApplyWindowing:\n+    \"\"\"Tests for util.apply_windowing().\"\"\"\n     def test_window_single_view(self):\n         \"\"\"Test windowing with a single view.\"\"\"\n         # 12-bit unsigned\n@@ -1611,7 +1430,7 @@ def test_window_single_view(self):\n \n         arr = ds.pixel_array\n         assert 642 == arr[326, 130]\n-        out = apply_voi_lut(arr, ds)\n+        out = apply_windowing(arr, ds)\n         assert 3046.6 == pytest.approx(out[326, 130], abs=0.1)\n \n     def test_window_multi_view(self):\n@@ -1631,9 +1450,9 @@ def test_window_multi_view(self):\n \n         arr = ds.pixel_array\n         assert 642 == arr[326, 130]\n-        out = apply_voi_lut(arr, ds)\n+        out = apply_windowing(arr, ds)\n         assert 3046.6 == pytest.approx(out[326, 130], abs=0.1)\n-        out = apply_voi_lut(arr, ds, index=1)\n+        out = apply_windowing(arr, ds, index=1)\n         assert 4095.0 == pytest.approx(out[326, 130], abs=0.1)\n \n     def test_window_uint8(self):\n@@ -1647,24 +1466,24 @@ def test_window_uint8(self):\n         # Linear\n         ds.WindowWidth = 1\n         ds.WindowCenter = 0\n-        assert [255, 255, 255, 255, 255] == apply_voi_lut(arr, ds).tolist()\n+        assert [255, 255, 255, 255, 255] == apply_windowing(arr, ds).tolist()\n \n         ds.WindowWidth = 128\n         ds.WindowCenter = 254\n         assert [0, 0, 0, 128.5, 130.5] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         # Linear exact\n         ds.VOILUTFunction = 'LINEAR_EXACT'\n         assert [0, 0, 0, 127.5, 129.5] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         # Sigmoid\n         ds.VOILUTFunction = 'SIGMOID'\n         assert [0.1, 0.1, 4.9, 127.5, 129.5] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n     def test_window_uint16(self):\n@@ -1677,22 +1496,22 @@ def test_window_uint16(self):\n \n         ds.WindowWidth = 1\n         ds.WindowCenter = 0\n-        assert [65535] * 5 == apply_voi_lut(arr, ds).tolist()\n+        assert [65535] * 5 == apply_windowing(arr, ds).tolist()\n \n         ds.WindowWidth = 32768\n         ds.WindowCenter = 254\n         assert [32260.5, 32262.5, 65535, 65535, 65535] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         ds.VOILUTFunction = 'LINEAR_EXACT'\n         assert [32259.5, 32261.5, 65535, 65535, 65535] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         ds.VOILUTFunction = 'SIGMOID'\n         assert [32259.5, 32261.5, 64319.8, 65512.3, 65512.3] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n     def test_window_uint32(self):\n@@ -1706,14 +1525,14 @@ def test_window_uint32(self):\n \n         ds.WindowWidth = 1\n         ds.WindowCenter = 0\n-        assert [y_max] * 5 == apply_voi_lut(arr, ds).tolist()\n+        assert [y_max] * 5 == apply_windowing(arr, ds).tolist()\n \n         ds.WindowWidth = 342423423423\n         ds.WindowCenter = 757336\n         assert (\n             [2147474148.4, 2147474148.4,\n              2174409724, 2201345299.7, 2201345299.7] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1721,7 +1540,7 @@ def test_window_uint32(self):\n         assert (\n             [2147474148.3, 2147474148.4,\n              2174409724, 2201345299.7, 2201345299.7] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1729,7 +1548,7 @@ def test_window_uint32(self):\n         assert (\n             [2147474148.3, 2147474148.4,\n              2174408313.1, 2201334008.2, 2201334008.3] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1745,25 +1564,25 @@ def test_window_int8(self):\n         ds.WindowWidth = 1\n         ds.WindowCenter = 0\n         assert [-128, -128, -128, 127, 127, 127, 127] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist()\n+            apply_windowing(arr, ds).tolist()\n         )\n \n         ds.WindowWidth = 128\n         ds.WindowCenter = -5\n         assert [-128, -128, 8.5, 10.5, 12.6, 127, 127] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         # Linear exact\n         ds.VOILUTFunction = 'LINEAR_EXACT'\n         assert [-128, -128, 7.5, 9.5, 11.5, 127, 127] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n         # Sigmoid\n         ds.VOILUTFunction = 'SIGMOID'\n         assert [-122.7, -122.5, 7.5, 9.4, 11.4, 122.8, 122.9] == pytest.approx(\n-            apply_voi_lut(arr, ds).tolist(), abs=0.1\n+            apply_windowing(arr, ds).tolist(), abs=0.1\n         )\n \n     def test_window_int16(self):\n@@ -1780,7 +1599,7 @@ def test_window_int16(self):\n         assert (\n             [-32768, -32768, -32768,\n              32767, 32767, 32767, 32767] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1789,7 +1608,7 @@ def test_window_int16(self):\n         assert (\n             [-32768, -32768, 2321.6,\n              2837.6, 3353.7, 32767, 32767] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1798,7 +1617,7 @@ def test_window_int16(self):\n         assert (\n             [-32768, -32768, 2047.5,\n              2559.5, 3071.5, 32767, 32767] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1807,7 +1626,7 @@ def test_window_int16(self):\n         assert (\n             [-31394.1, -31351.4, 2044.8,\n              2554.3, 3062.5, 31692, 31724.6] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1825,7 +1644,7 @@ def test_window_int32(self):\n         assert (\n             [-2**31, -2**31, -2**31,\n              2**31 - 1, 2**31 - 1, 2**31 - 1, 2**31 - 1] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1834,7 +1653,7 @@ def test_window_int32(self):\n         assert (\n             [-2147483648, -2147483648, 152183880, 186002520.1,\n              219821160.3, 2147483647, 2147483647] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1843,7 +1662,7 @@ def test_window_int32(self):\n         assert (\n             [-2147483648, -2147483648, 134217727.5, 167772159.5,\n              201326591.5, 2147483647, 2147483647] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1852,7 +1671,7 @@ def test_window_int32(self):\n         assert (\n             [-2057442919.3, -2054646500.7, 134043237.4, 167431657.4,\n              200738833.7, 2077033158.8, 2079166214.8] == pytest.approx(\n-                apply_voi_lut(arr, ds).tolist(), abs=0.1\n+                apply_windowing(arr, ds).tolist(), abs=0.1\n             )\n         )\n \n@@ -1872,7 +1691,7 @@ def test_window_multi_frame(self):\n         assert (2, 484, 484) == arr.shape\n         assert 642 == arr[0, 326, 130]\n         assert 3453 == arr[1, 326, 130]\n-        out = apply_voi_lut(arr, ds)\n+        out = apply_windowing(arr, ds)\n         assert 3046.6 == pytest.approx(out[0, 326, 130], abs=0.1)\n         assert 4095.0 == pytest.approx(out[1, 326, 130], abs=0.1)\n \n@@ -1902,7 +1721,7 @@ def test_window_rescale(self):\n         assert 770.4 == hu[326, 130]\n         assert 1347.6 == hu[316, 481]\n         # With rescale -> output range is 0 to 4914\n-        out = apply_voi_lut(hu, ds)\n+        out = apply_windowing(hu, ds)\n         assert 0 == pytest.approx(out[16, 60], abs=0.1)\n         assert 4455.6 == pytest.approx(out[326, 130], abs=0.1)\n         assert 4914.0 == pytest.approx(out[316, 481], abs=0.1)\n@@ -1930,7 +1749,7 @@ def test_window_modality_lut(self):\n         hu = apply_modality_lut(arr, ds)\n         assert 65535 == hu[16, 60]\n         assert 49147 == hu[0, 1]\n-        out = apply_voi_lut(hu, ds)\n+        out = apply_windowing(hu, ds)\n         assert 65535.0 == pytest.approx(out[16, 60], abs=0.1)\n         assert 32809.0 == pytest.approx(out[0, 1], abs=0.1)\n         # Output range must be 0 to 2**16 - 1\n@@ -1943,7 +1762,7 @@ def test_window_bad_photometric_interp(self):\n         ds.PhotometricInterpretation = 'RGB'\n         msg = r\"only 'MONOCHROME1' and 'MONOCHROME2' are allowed\"\n         with pytest.raises(ValueError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n+            apply_windowing(ds.pixel_array, ds)\n \n     def test_window_bad_parameters(self):\n         \"\"\"Test bad windowing parameters raise exceptions.\"\"\"\n@@ -1952,22 +1771,22 @@ def test_window_bad_parameters(self):\n         ds.VOILUTFunction = 'LINEAR'\n         msg = r\"Width must be greater than or equal to 1\"\n         with pytest.raises(ValueError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n+            apply_windowing(ds.pixel_array, ds)\n \n         ds.VOILUTFunction = 'LINEAR_EXACT'\n         msg = r\"Width must be greater than 0\"\n         with pytest.raises(ValueError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n+            apply_windowing(ds.pixel_array, ds)\n \n         ds.VOILUTFunction = 'SIGMOID'\n         msg = r\"Width must be greater than 0\"\n         with pytest.raises(ValueError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n+            apply_windowing(ds.pixel_array, ds)\n \n         ds.VOILUTFunction = 'UNKNOWN'\n         msg = r\"Unsupported \\(0028,1056\\) VOI LUT Function value 'UNKNOWN'\"\n         with pytest.raises(ValueError, match=msg):\n-            apply_voi_lut(ds.pixel_array, ds)\n+            apply_windowing(ds.pixel_array, ds)\n \n     def test_window_bad_index(self, no_numpy_use):\n         \"\"\"Test windowing with a bad view index.\"\"\"\n@@ -1975,7 +1794,7 @@ def test_window_bad_index(self, no_numpy_use):\n         assert 2 == len(ds.WindowWidth)\n         arr = ds.pixel_array\n         with pytest.raises(IndexError, match=r\"list index out of range\"):\n-            apply_voi_lut(arr, ds, index=2)\n+            apply_windowing(arr, ds, index=2)\n \n     def test_unchanged(self):\n         \"\"\"Test input array is unchanged if no VOI LUT\"\"\"\n@@ -1984,7 +1803,219 @@ def test_unchanged(self):\n         ds.PixelRepresentation = 1\n         ds.BitsStored = 8\n         arr = np.asarray([-128, -127, -1, 0, 1, 126, 127], dtype='int8')\n-        out = apply_voi_lut(arr, ds)\n+        out = apply_windowing(arr, ds)\n+        assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n+\n+    def test_rescale_empty(self):\n+        \"\"\"Test RescaleSlope and RescaleIntercept being empty.\"\"\"\n+        ds = dcmread(WIN_12_1F)\n+        ds.RescaleSlope = None\n+        ds.RescaleIntercept = None\n+\n+        arr = ds.pixel_array\n+        assert 0 == arr[16, 60]\n+        assert 642 == arr[326, 130]\n+        assert 1123 == arr[316, 481]\n+        out = apply_windowing(arr, ds)\n+        assert 0 == pytest.approx(out[16, 60], abs=0.1)\n+        assert 3046.6 == pytest.approx(out[326, 130], abs=0.1)\n+        assert 4095.0 == pytest.approx(out[316, 481], abs=0.1)\n+\n+\n+@pytest.mark.skipif(not HAVE_NP, reason=\"Numpy is not available\")\n+class TestNumpy_ApplyVOI:\n+    \"\"\"Tests for util.apply_voi().\"\"\"\n+    def test_voi_single_view(self):\n+        \"\"\"Test VOI LUT with a single view.\"\"\"\n+        ds = dcmread(VOI_08_1F)\n+        assert 8 == ds.BitsAllocated\n+        assert 8 == ds.BitsStored\n+        assert 0 == ds.PixelRepresentation\n+        item = ds.VOILUTSequence[0]\n+        assert [256, 0, 16] == item.LUTDescriptor\n+        lut = item.LUTData\n+        assert 0 == lut[0]\n+        assert 19532 == lut[76]\n+        assert 45746 == lut[178]\n+        assert 65535 == lut[255]\n+\n+        arr = ds.pixel_array\n+        assert 0 == arr[387, 448]\n+        assert 76 == arr[178, 126]\n+        assert 178 == arr[186, 389]\n+        assert 255 == arr[129, 79]\n+\n+        out = apply_voi(arr, ds)\n+        assert 0 == out[387, 448]\n+        assert 19532 == out[178, 126]\n+        assert 45746 == out[186, 389]\n+        assert 65535 == out[129, 79]\n+\n+    def test_voi_multi_view(self):\n+        \"\"\"Test VOI LUT with multiple views.\"\"\"\n+        ds = dcmread(VOI_08_1F)\n+        assert 8 == ds.BitsAllocated\n+        assert 8 == ds.BitsStored\n+        assert 0 == ds.PixelRepresentation\n+        item0 = ds.VOILUTSequence[0]\n+        # Add another view thats the inverse\n+        ds.VOILUTSequence.append(Dataset())\n+        item1 = ds.VOILUTSequence[1]\n+        item1.LUTDescriptor = [256, 0, 16]\n+        item1.LUTData = item0.LUTData[::-1]\n+\n+        arr = ds.pixel_array\n+        assert 0 == arr[387, 448]\n+        assert 76 == arr[178, 126]\n+        assert 178 == arr[186, 389]\n+        assert 255 == arr[129, 79]\n+\n+        out0 = apply_voi(arr, ds)\n+        assert 0 == out0[387, 448]\n+        assert 19532 == out0[178, 126]\n+        assert 45746 == out0[186, 389]\n+        assert 65535 == out0[129, 79]\n+\n+        out1 = apply_voi(arr, ds, index=1)\n+        assert 65535 == out1[387, 448]\n+        assert 46003 == out1[178, 126]\n+        assert 19789 == out1[186, 389]\n+        assert 0 == out1[129, 79]\n+\n+    def test_voi_multi_frame(self):\n+        \"\"\"Test VOI with a multiple frames.\"\"\"\n+        ds = dcmread(VOI_08_1F)\n+        assert 8 == ds.BitsAllocated\n+        assert 8 == ds.BitsStored\n+        assert 0 == ds.PixelRepresentation\n+\n+        arr = ds.pixel_array\n+        arr = np.stack([arr, 255 - arr])\n+        assert (2, 512, 512) == arr.shape\n+\n+        out = apply_voi(arr, ds)\n+        assert 0 == out[0, 387, 448]\n+        assert 19532 == out[0, 178, 126]\n+        assert 45746 == out[0, 186, 389]\n+        assert 65535 == out[0, 129, 79]\n+        assert 65535 == out[1, 387, 448]\n+        assert 46003 == out[1, 178, 126]\n+        assert 19789 == out[1, 186, 389]\n+        assert 0 == out[1, 129, 79]\n+\n+    def test_voi_zero_entries(self):\n+        \"\"\"Test that 0 entries is interpreted correctly.\"\"\"\n+        ds = dcmread(VOI_08_1F)\n+        seq = ds.VOILUTSequence[0]\n+        seq.LUTDescriptor = [0, 0, 16]\n+        assert 256 == len(seq.LUTData)\n+        arr = np.asarray([0, 255, 256, 65535])\n+        msg = r\"index 256 is out of bounds\"\n+        with pytest.raises(IndexError, match=msg):\n+            apply_voi(arr, ds)\n+\n+        # LUTData with 65536 entries\n+        seq.LUTData = [0] * 65535 + [1]\n+        out = apply_voi(arr, ds)\n+        assert [0, 0, 0, 1] == list(out)\n+\n+    def test_voi_uint8(self):\n+        \"\"\"Test uint VOI LUT with an 8-bit LUT.\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 8\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 8]\n+        item.LUTData = [0, 127, 128, 255]\n+        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n+        out = apply_voi(arr, ds)\n+        assert 'uint8' == out.dtype\n+        assert [0, 127, 255, 255, 255] == out.tolist()\n+\n+    def test_voi_uint16(self):\n+        \"\"\"Test uint VOI LUT with an 16-bit LUT.\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 16\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 16]\n+        item.LUTData = [0, 127, 32768, 65535]\n+        arr = np.asarray([0, 1, 2, 3, 255], dtype='uint16')\n+        out = apply_voi(arr, ds)\n+        assert 'uint16' == out.dtype\n+        assert [0, 127, 32768, 65535, 65535] == out.tolist()\n+\n+    def test_voi_int8(self):\n+        \"\"\"Test int VOI LUT with an 8-bit LUT.\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 1\n+        ds.BitsStored = 8\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 8]\n+        item.LUTData = [0, 127, 128, 255]\n+        arr = np.asarray([0, -1, 2, -128, 127], dtype='int8')\n+        out = apply_voi(arr, ds)\n+        assert 'uint8' == out.dtype\n+        assert [0, 0, 128, 0, 255] == out.tolist()\n+\n+    def test_voi_int16(self):\n+        \"\"\"Test int VOI LUT with an 16-bit LUT.\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 16\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 16]\n+        item.LUTData = [0, 127, 32768, 65535]\n+        arr = np.asarray([0, -1, 2, -128, 255], dtype='int16')\n+        out = apply_voi(arr, ds)\n+        assert 'uint16' == out.dtype\n+        assert [0, 0, 32768, 0, 65535] == out.tolist()\n+\n+    def test_voi_bad_depth(self):\n+        \"\"\"Test bad LUT depth raises exception.\"\"\"\n+        ds = dcmread(VOI_08_1F)\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor[2] = 7\n+        msg = r\"'7' bits per LUT entry is not supported\"\n+        with pytest.raises(NotImplementedError, match=msg):\n+            apply_voi(ds.pixel_array, ds)\n+\n+        item.LUTDescriptor[2] = 17\n+        msg = r\"'17' bits per LUT entry is not supported\"\n+        with pytest.raises(NotImplementedError, match=msg):\n+            apply_voi(ds.pixel_array, ds)\n+\n+    def test_voi_uint16_array_float(self):\n+        \"\"\"Test warning when array is float and VOI LUT with an 16-bit LUT\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 16\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 16]\n+        item.LUTData = [0, 127, 32768, 65535]\n+        arr = np.asarray([0, 1, 2, 3, 255], dtype='float64')\n+        msg = (\n+            r\"Applying a VOI LUT on a float input array may give \"\n+            r\"incorrect results\"\n+        )\n+\n+        with pytest.warns(UserWarning, match=msg):\n+            out = apply_voi(arr, ds)\n+            assert [0, 127, 32768, 65535, 65535] == out.tolist()\n+\n+    def test_unchanged(self):\n+        \"\"\"Test input array is unchanged if no VOI LUT\"\"\"\n+        ds = Dataset()\n+        ds.PhotometricInterpretation = 'MONOCHROME1'\n+        ds.PixelRepresentation = 1\n+        ds.BitsStored = 8\n+        arr = np.asarray([-128, -127, -1, 0, 1, 126, 127], dtype='int8')\n+        out = apply_voi(arr, ds)\n         assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n \n     def test_voi_lutdata_ow(self):\n@@ -2001,11 +2032,94 @@ def test_voi_lutdata_ow(self):\n         item.LUTData = pack('<4H', *item.LUTData)\n         item['LUTData'].VR = 'OW'\n         arr = np.asarray([0, 1, 2, 3, 255], dtype='uint16')\n-        out = apply_voi_lut(arr, ds)\n+        out = apply_voi(arr, ds)\n         assert 'uint16' == out.dtype\n         assert [0, 127, 32768, 65535, 65535] == out.tolist()\n \n \n+@pytest.mark.skipif(not HAVE_NP, reason=\"Numpy is not available\")\n+class TestNumpy_ApplyVOILUT:\n+    def test_unchanged(self):\n+        \"\"\"Test input array is unchanged if no VOI LUT\"\"\"\n+        ds = Dataset()\n+        ds.PhotometricInterpretation = 'MONOCHROME1'\n+        ds.PixelRepresentation = 1\n+        ds.BitsStored = 8\n+        arr = np.asarray([-128, -127, -1, 0, 1, 126, 127], dtype='int8')\n+        out = apply_voi_lut(arr, ds)\n+        assert [-128, -127, -1, 0, 1, 126, 127] == out.tolist()\n+\n+    def test_only_windowing(self):\n+        \"\"\"Test only windowing operation elements present.\"\"\"\n+        ds = Dataset()\n+        ds.PhotometricInterpretation = 'MONOCHROME1'\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 8\n+        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n+\n+        ds.WindowWidth = 1\n+        ds.WindowCenter = 0\n+        assert [255, 255, 255, 255, 255] == apply_voi_lut(arr, ds).tolist()\n+\n+    def test_only_voi(self):\n+        \"\"\"Test only LUT operation elements present.\"\"\"\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 8\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 8]\n+        item.LUTData = [0, 127, 128, 255]\n+        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n+        out = apply_voi_lut(arr, ds)\n+        assert 'uint8' == out.dtype\n+        assert [0, 127, 255, 255, 255] == out.tolist()\n+\n+    def test_voi_windowing(self):\n+        \"\"\"Test both LUT and windowing operation elements present.\"\"\"\n+        ds = Dataset()\n+        ds.PhotometricInterpretation = 'MONOCHROME1'\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 8\n+        ds.WindowWidth = 1\n+        ds.WindowCenter = 0\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 8]\n+        item.LUTData = [0, 127, 128, 255]\n+        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n+\n+        # Defaults to LUT\n+        out = apply_voi_lut(arr, ds)\n+        assert [0, 127, 255, 255, 255] == out.tolist()\n+\n+        out = apply_voi_lut(arr, ds, prefer_lut=False)\n+        assert [255, 255, 255, 255, 255] == out.tolist()\n+\n+    def test_voi_windowing_empty(self):\n+        \"\"\"Test empty VOI elements.\"\"\"\n+        ds = Dataset()\n+        ds.PhotometricInterpretation = 'MONOCHROME1'\n+        ds.PixelRepresentation = 0\n+        ds.BitsStored = 8\n+        ds.WindowWidth = 1\n+        ds.WindowCenter = 0\n+        ds.VOILUTSequence = [Dataset()]\n+        item = ds.VOILUTSequence[0]\n+        item.LUTDescriptor = [4, 0, 8]\n+        item.LUTData = [0, 127, 128, 255]\n+        arr = np.asarray([0, 1, 128, 254, 255], dtype='uint8')\n+\n+        # Test empty VOI elements\n+        item.LUTData = None\n+        out = apply_voi_lut(arr, ds)\n+        assert [255, 255, 255, 255, 255] == out.tolist()\n+\n+        # Test empty windowing elements\n+        ds.WindowWidth = None\n+        out = apply_voi_lut(arr, ds)\n+        assert [0, 1, 128, 254, 255] == out.tolist()\n+\n class TestGetJ2KParameters:\n     \"\"\"Tests for get_j2k_parameters.\"\"\"\n     def test_precision(self):\n", "problem_statement": "apply_voi_lut - unclear what it does if both WL/VOILUTFunction _and_ VOILUTSequence are present\nhttps://pydicom.github.io/pydicom/dev/reference/generated/pydicom.pixel_data_handlers.util.html#pydicom.pixel_data_handlers.util.apply_voi_lut\r\n\r\nHi all,\r\n\r\nI'm working with some mammo image (digital) that have both \r\n- window/level (0028,1050 0028,1051) plus VOILUTFunction (0028,1056) (set to SIGMOID) (set of 3 WL values)\r\n- VOILUT sequences (0028, 3010)\r\n\r\nspecified.\r\n\r\nProblem\r\n---\r\n\r\nIt's unclear from the documentation when both a VOILUT (0028,3010) _and_ WL (0028,1051...) are present which is applied - the lut or the wl.\r\n\r\nIt just says if a LUT's present, it will apply that, and if a WL set is present it will apply that.\r\n\r\nQuestions\r\n---\r\n\r\n- If both LUT and WL are supplied, by the dicom standard, which should be applied?\r\n- Separately to the above question about which is applied, if _both_ LUT and WL sequences are supplied, is there a way in `apply_voi_lut` to specify applying one or the other?  (ie force application of the WL instead of LUT etc)\r\n\r\n- Also, if an image has a sequence of WL values rather than being single valued (so 0028,1050 & 0028,1051 are sequences), does the `index` parameter to `apply_voi_lut` apply to specify which in the sequence you want to use?\r\n\r\nThanks!\r\n\napply_voi_lut can't handle missing DICOM meta info\nI have encountered two real life examples where `apply_voi_lut` does not handle corruption in DICOM meta fields\r\n\r\ncase 1:\r\n```\r\n(0028, 1050) Window Center                       DS: \"128.0\"\r\n(0028, 1051) Window Width                        DS: \"256.0\"\r\n(0028, 1052) Rescale Intercept                   DS: None\r\n(0028, 1053) Rescale Slope                       DS: None\r\n```\r\nthrows an exception\r\n\r\n```\r\n  File \"python3.7/site-packages/pydicom/pixel_data_handlers/util.py\", line 380, in apply_voi_lut\r\n    y_min = y_min * ds.RescaleSlope + ds.RescaleIntercept\r\nTypeError: unsupported operand type(s) for *: 'int' and 'NoneType' \r\n```\r\n\r\n\r\ncase 2:\r\n\r\n```\r\n(0028, 1050) Window Center                       DS: \"2607.0\"\r\n(0028, 1051) Window Width                        DS: \"2785.0\"\r\n(0028, 1052) Rescale Intercept                   DS: \"0.0\"\r\n(0028, 1053) Rescale Slope                       DS: \"1.0\"\r\n(0028, 1054) Rescale Type                        LO: 'US'\r\n(0028, 2110) Lossy Image Compression             CS: '00'\r\n(0028, 3010)  VOI LUT Sequence   1 item(s) ---- \r\n   (0028, 3002) LUT Descriptor                      SS: None\r\n   (0028, 3003) LUT Explanation                     LO: 'Noramal'\r\n   (0028, 3006) LUT Data                            OW: None\r\n```\r\n\r\nthrows an exception\r\n\r\n```\r\n  File \"python3.7/site-packages/pydicom/pixel_data_handlers/util.py\", line 312, in apply_voi_lut\r\n    nr_entries = item.LUTDescriptor[0] or 2**16\r\nTypeError: 'NoneType' object is not subscriptable\r\n```\r\n\r\n\r\nSo far I have handled this with:\r\n\r\n```\r\n    def _lut_convert(self):\r\n        return apply_voi_lut(self.input_dicom.pixel_array, self.input_dicom)\r\n\r\n    def _get_raw_data(self):\r\n\r\n        # convert to presentation LUT\r\n        try:\r\n            data = self._lut_convert()\r\n        # many things can be corrupted in the VOILUTSequence attribute,\r\n        # fall back to default WC/WW conversion\r\n        except Exception as e:\r\n            try:\r\n                if \"VOILUTSequence\" in self.input_dicom:\r\n                    del self.input_dicom[\"VOILUTSequence\"]\r\n                    data = self._lut_convert()\r\n            except Exception as e:\r\n                raise InvalidImage(f\"Could not convert to presentation LUT due to: {e}\")\r\n```\r\n\r\nWhile the case 1 could be seen as an expected behavior (?), I imagine case 2 should be handled by WC/WW transformations if followed DICOM standard?\r\n\r\n\r\n\r\n\r\n\r\n\r\n\r\n\r\n\n", "hints_text": "For some reason when I wrote it, I assumed it was a case of either/or for *VOI LUT Sequence*/*Window Center*, but now that I look at the Standard again I see it possible that both can be present (although only one can be applied). I'll probably add a flag to allow selecting which one is used when both are present.\n> If both LUT and WL are supplied, by the dicom standard, which should be applied?\r\n\r\nThat should be up to the user\r\n\r\n> Separately to the above question about which is applied, if both LUT and WL sequences are supplied, is there a way in apply_voi_lut to specify applying one or the other? (ie force application of the WL instead of LUT etc)\r\n\r\nNot at the moment, although you could force which is applied by deleting the corresponding element(s) for the operation you don't want\r\n\r\n> Also, if an image has a sequence of WL values rather than being single valued (so 0028,1050 & 0028,1051 are sequences), does the index parameter to apply_voi_lut apply to specify which in the sequence you want to use?\r\n\r\nYes\nIs it possible that you could attach an anonymised dataset we could add to our test data? It'd be helpful to have a real dataset.\n", "created_at": "2020-10-26T20:49:06Z", "version": "2.0", "FAIL_TO_PASS": ["pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unknown_pixel_representation_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unknown_bits_allocated_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_unsupported_dtypes", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[1-0-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[1-1-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[8-0-False-uint8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[8-1-False-int8]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[16-0-False-uint16]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[16-1-False-int16]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-0-False-uint32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-1-False-int32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[32-0-True-float32]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_supported_dtypes[64-0-True-float64]", "pydicom/tests/test_handler_util.py::TestNumpy_PixelDtype::test_byte_swapping", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_1frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_1frame_3sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_2frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_reference_2frame_3sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_3sample_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_1frame_3sample_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_1sample", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_3sample_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_2frame_3sample_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_compressed_syntaxes_0conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_compressed_syntaxes_1conf", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_uncompressed_syntaxes", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_nr_frames_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_samples_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ReshapePixelArray::test_invalid_planar_conf_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_unknown_current_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_unknown_desired_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[RGB-RGB]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL-YBR_FULL]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL-YBR_FULL_422]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL_422-YBR_FULL_422]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_current_is_desired[YBR_FULL_422-YBR_FULL]", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_rgb_ybr_rgb_single_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ConvertColourSpace::test_rgb_ybr_rgb_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_DtypeCorrectedForEndianness::test_byte_swapping", "pydicom/tests/test_handler_util.py::TestNumpy_DtypeCorrectedForEndianness::test_no_endian_raises", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_bytes[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_in_pixels[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape0-1-length0]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape1-1-length1]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape2-1-length2]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape3-1-length3]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape4-1-length4]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape5-1-length5]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape6-1-length6]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape7-1-length7]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape8-1-length8]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape9-8-length9]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape10-8-length10]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape11-8-length11]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape12-8-length12]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape13-8-length13]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape14-8-length14]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape15-16-length15]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape16-16-length16]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape17-16-length17]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape18-16-length18]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape19-16-length19]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape20-32-length20]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape21-32-length21]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape22-32-length22]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape23-32-length23]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape24-32-length24]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape25-1-length25]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape26-1-length26]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape27-1-length27]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape28-1-length28]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape29-1-length29]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape30-1-length30]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape31-1-length31]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape32-1-length32]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape33-1-length33]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape34-8-length34]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape35-8-length35]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape36-8-length36]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape37-8-length37]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape38-8-length38]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape39-8-length39]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape40-16-length40]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape41-16-length41]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape42-16-length42]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape43-32-length43]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape44-32-length44]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape45-32-length45]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape46-1-length46]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape47-1-length47]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape48-1-length48]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape49-1-length49]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape50-1-length50]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape51-1-length51]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape52-1-length52]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape53-1-length53]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape54-1-length54]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape55-8-length55]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape56-8-length56]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape57-8-length57]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape58-16-length58]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape59-16-length59]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape60-16-length60]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape61-32-length61]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape62-32-length62]", "pydicom/tests/test_handler_util.py::TestGetExpectedLength::test_length_ybr_422[shape63-32-length63]", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_slope_intercept", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lut_sequence", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lut_sequence_zero_entries", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ModalityLUT::test_lutdata_ow", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_neither_ds_nor_palette_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_palette_unknown_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_palette_unavailable_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_supplemental_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_invalid_bit_depth_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_invalid_lut_bit_depth_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_unequal_lut_length_raises", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_no_palette_color", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint08_16", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint08_16_2frame", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint16_16_segmented_litle", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_uint16_16_segmented_big", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_16_allocated_8_entries", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_alpha", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_well_known_palette", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_first_map_positive", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_first_map_negative", "pydicom/tests/test_handler_util.py::TestNumpy_PaletteColor::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_discrete", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_linear", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_indirect_08", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_indirect_16", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_spring", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_summer", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_fall", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_palettes_winter", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_first_linear_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_first_indirect_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ExpandSegmentedLUT::test_unknown_opcode_raises", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_single_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_multi_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_uint32", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_int32", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_rescale", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_modality_lut", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_photometric_interp", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_parameters", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_window_bad_index", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyWindowing::test_rescale_empty", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_single_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_multi_view", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_multi_frame", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_zero_entries", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_int8", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_int16", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_bad_depth", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_uint16_array_float", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOI::test_voi_lutdata_ow", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_unchanged", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_only_windowing", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_only_voi", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_voi_windowing", "pydicom/tests/test_handler_util.py::TestNumpy_ApplyVOILUT::test_voi_windowing_empty", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_precision", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_not_j2k", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_no_siz", "pydicom/tests/test_handler_util.py::TestGetJ2KParameters::test_short_bytestream", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_none", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_missing", "pydicom/tests/test_handler_util.py::TestGetNrFrames::test_existing"], "PASS_TO_PASS": [], "environment_setup_commit": "9d69811e539774f296c2f289839147e741251716"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1719", "base_commit": "5bb3ddef43b35c07485a84f90b6a453fc649e31d", "patch": "diff --git a/astroid/raw_building.py b/astroid/raw_building.py\n--- a/astroid/raw_building.py\n+++ b/astroid/raw_building.py\n@@ -388,9 +388,9 @@ def object_build(\n             pypy__class_getitem__ = IS_PYPY and name == \"__class_getitem__\"\n             try:\n                 with warnings.catch_warnings():\n-                    warnings.simplefilter(\"error\")\n+                    warnings.simplefilter(\"ignore\")\n                     member = getattr(obj, name)\n-            except (AttributeError, DeprecationWarning):\n+            except (AttributeError):\n                 # damned ExtensionClass.Base, I know you're there !\n                 attach_dummy_node(node, name)\n                 continue\n", "test_patch": "diff --git a/tests/testdata/python3/data/fake_module_with_warnings.py b/tests/testdata/python3/data/fake_module_with_warnings.py\nnew file mode 100644\n--- /dev/null\n+++ b/tests/testdata/python3/data/fake_module_with_warnings.py\n@@ -0,0 +1,22 @@\n+'''\n+This is a mock of a module like Pandas, which can throw warnings for deprecated attributes\n+'''\n+import warnings\n+\n+\n+def __dir__():\n+    # GH43028\n+    # Int64Index etc. are deprecated, but we still want them to be available in the dir.\n+    # Remove in Pandas 2.0, when we remove Int64Index etc. from the code base.\n+    return list(globals().keys()) + [\"Float64Index\"]\n+\n+\n+def __getattr__(name):\n+    if name == \"Float64Index\":\n+        warnings.warn(\"This is what pandas would do\", FutureWarning, stacklevel=2)\n+        return 5\n+    raise AttributeError(f\"module 'pandas' has no attribute '{name}'\")\n+\n+\n+__all__ = [\"Float64Index\"]  # pylint: disable=E0603\n+__doc__ = \"\"\ndiff --git a/tests/unittest_raw_building.py b/tests/unittest_raw_building.py\n--- a/tests/unittest_raw_building.py\n+++ b/tests/unittest_raw_building.py\n@@ -2,11 +2,14 @@\n # For details: https://github.com/PyCQA/astroid/blob/main/LICENSE\n # Copyright (c) https://github.com/PyCQA/astroid/blob/main/CONTRIBUTORS.txt\n \n+import types\n import unittest\n \n import _io\n import pytest\n \n+# A fake module to simulate pandas in unittest below\n+import tests.testdata.python3.data.fake_module_with_warnings as fm\n from astroid.builder import AstroidBuilder\n from astroid.const import IS_PYPY\n from astroid.raw_building import (\n@@ -86,6 +89,19 @@ def test_io_is__io(self):\n         buffered_reader = module.getattr(\"BufferedReader\")[0]\n         self.assertEqual(buffered_reader.root().name, \"io\")\n \n+    def test_build_function_deepinspect_deprecation(self) -> None:\n+        # Tests https://github.com/PyCQA/astroid/issues/1717\n+        # When astroid deep inspection of modules raises\n+        # attribute errors when getting all attributes\n+        # Create a mock module to simulate a Cython module\n+        m = types.ModuleType(\"test\")\n+\n+        # Attach a mock of pandas with the same behavior\n+        m.pd = fm\n+\n+        # This should not raise an exception\n+        AstroidBuilder().module_build(m, \"test\")\n+\n \n if __name__ == \"__main__\":\n     unittest.main()\n", "problem_statement": "Cython module with import triggers deep introspection for pandas, raises unhandled FutureWarning\nThis is a somewhat complicated situation to reproduce, but basically `pandas` throws `FutureWarning`s for certain attributes, and when you import it into a Cython module (triggering astroid's deep module inspection), these future warnings are not handled by astroid and bubble up as `AstroidError`s through to pylint. Here is a full repro:\r\n\r\n\r\n### Cython module `pyx.pyx`\r\n\r\n```python\r\n# distutils: language = c++\r\nimport pandas as pd\r\n\r\ncdef class Test:\r\n    def __cinit__(self):\r\n        ...\r\n```\r\n\r\n\r\n### Python module `test.py`\r\n\r\n```python\r\nimport pyx\r\n\r\npyx.Test()\r\n```\r\n\r\n\r\n\r\n### Commands\r\n```\r\ncythonize -a -i pyx.pyx\r\npylint --extension-pkg-allow-list=pyx,pandas test.py\r\n```\r\n\r\n\r\n### Exception\r\n```\r\nException on node <Import l.1 at 0x106b23ca0> in file '/Users/timkpaine/Programs/projects/other/astroid/test.py'\r\nTraceback (most recent call last):\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/pylint/checkers/imports.py\", line 765, in _get_imported_module\r\n    return importnode.do_import_module(modname)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/mixins.py\", line 102, in do_import_module\r\n    return mymodule.import_module(\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/nodes/scoped_nodes/scoped_nodes.py\", line 527, in import_module\r\n    return AstroidManager().ast_from_module_name(absmodname)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/manager.py\", line 168, in ast_from_module_name\r\n    return self.ast_from_module(module, modname)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/manager.py\", line 265, in ast_from_module\r\n    return AstroidBuilder(self).module_build(module, modname)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/builder.py\", line 91, in module_build\r\n    node = self.inspect_build(module, modname=modname, path=path)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/raw_building.py\", line 311, in inspect_build\r\n    self.object_build(node, module)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/raw_building.py\", line 367, in object_build\r\n    self.object_build(module, member)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/astroid/raw_building.py\", line 325, in object_build\r\n    member = getattr(obj, name)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/pandas/__init__.py\", line 198, in __getattr__\r\n    warnings.warn(\r\nFutureWarning: pandas.Float64Index is deprecated and will be removed from pandas in a future version. Use pandas.Index with the appropriate dtype instead.\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 \"/opt/homebrew/lib/python3.9/site-packages/pylint/utils/ast_walker.py\", line 90, in walk\r\n    callback(astroid)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/pylint/checkers/imports.py\", line 472, in visit_import\r\n    imported_module = self._get_imported_module(node, name)\r\n  File \"/opt/homebrew/lib/python3.9/site-packages/pylint/checkers/imports.py\", line 788, in _get_imported_module\r\n    raise astroid.AstroidError from e\r\nastroid.exceptions.AstroidError\r\n************* Module test\r\ntest.py:1:0: F0002: test.py: Fatal error while checking 'test.py'. Please open an issue in our bug tracker so we address this. There is a pre-filled template that you can use in '/Users/timkpaine/Library/Caches/pylint/pylint-crash-2022-07-19-17.txt'. (astroid-error)\r\n```\r\n\r\n\r\n\r\n\r\n### Standalone (Non Cython) repro for convenience\r\n\r\n```python\r\nimport types\r\nimport pandas as pd\r\nfrom astroid.builder import AstroidBuilder\r\n\r\n\r\nm = types.ModuleType(\"test\")\r\nm.pd = pd\r\n\r\nAstroidBuilder().module_build(m, \"test\")\r\n```\r\n\r\n\r\nxref: https://github.com/PyCQA/pylint/issues/7205\r\nxref: https://github.com/PyCQA/astroid/pull/1719\n", "hints_text": "", "created_at": "2022-07-19T21:09:19Z", "version": "2.13", "FAIL_TO_PASS": ["tests/unittest_raw_building.py::RawBuildingTC::test_build_function_deepinspect_deprecation"], "PASS_TO_PASS": ["tests/unittest_raw_building.py::RawBuildingTC::test_attach_dummy_node", "tests/unittest_raw_building.py::RawBuildingTC::test_build_class", "tests/unittest_raw_building.py::RawBuildingTC::test_build_from_import", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_args", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_defaults", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_kwonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_posonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_module", "tests/unittest_raw_building.py::RawBuildingTC::test_io_is__io"], "environment_setup_commit": "fe058bff95745371df5796286d33677c21137847"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1368", "base_commit": "93e84041387c80bf8738fe96409a0cfd5852c29a", "patch": "diff --git a/pvlib/iotools/crn.py b/pvlib/iotools/crn.py\n--- a/pvlib/iotools/crn.py\n+++ b/pvlib/iotools/crn.py\n@@ -2,15 +2,14 @@\n \"\"\"\n \n import pandas as pd\n-import numpy as np\n \n \n-HEADERS = (\n-    'WBANNO UTC_DATE UTC_TIME LST_DATE LST_TIME CRX_VN LONGITUDE LATITUDE '\n-    'AIR_TEMPERATURE PRECIPITATION SOLAR_RADIATION SR_FLAG '\n-    'SURFACE_TEMPERATURE ST_TYPE ST_FLAG RELATIVE_HUMIDITY RH_FLAG '\n-    'SOIL_MOISTURE_5 SOIL_TEMPERATURE_5 WETNESS WET_FLAG WIND_1_5 WIND_FLAG'\n-)\n+HEADERS = [\n+    'WBANNO', 'UTC_DATE', 'UTC_TIME', 'LST_DATE', 'LST_TIME', 'CRX_VN',\n+    'LONGITUDE', 'LATITUDE', 'AIR_TEMPERATURE', 'PRECIPITATION',\n+    'SOLAR_RADIATION', 'SR_FLAG', 'SURFACE_TEMPERATURE', 'ST_TYPE', 'ST_FLAG',\n+    'RELATIVE_HUMIDITY', 'RH_FLAG', 'SOIL_MOISTURE_5', 'SOIL_TEMPERATURE_5',\n+    'WETNESS', 'WET_FLAG', 'WIND_1_5', 'WIND_FLAG']\n \n VARIABLE_MAP = {\n     'LONGITUDE': 'longitude',\n@@ -24,6 +23,21 @@\n     'WIND_FLAG': 'wind_speed_flag'\n }\n \n+NAN_DICT = {\n+    'CRX_VN': -99999,\n+    'AIR_TEMPERATURE': -9999,\n+    'PRECIPITATION': -9999,\n+    'SOLAR_RADIATION': -99999,\n+    'SURFACE_TEMPERATURE': -9999,\n+    'RELATIVE_HUMIDITY': -9999,\n+    'SOIL_MOISTURE_5': -99,\n+    'SOIL_TEMPERATURE_5': -9999,\n+    'WETNESS': -9999,\n+    'WIND_1_5': -99}\n+\n+# Add NUL characters to possible NaN values for all columns\n+NAN_DICT = {k: [v, '\\x00\\x00\\x00\\x00\\x00\\x00'] for k, v in NAN_DICT.items()}\n+\n # as specified in CRN README.txt file. excludes 1 space between columns\n WIDTHS = [5, 8, 4, 8, 4, 6, 7, 7, 7, 7, 6, 1, 7, 1, 1, 5, 1, 7, 7, 5, 1, 6, 1]\n # add 1 to make fields contiguous (required by pandas.read_fwf)\n@@ -40,15 +54,22 @@\n ]\n \n \n-def read_crn(filename):\n-    \"\"\"\n-    Read a NOAA USCRN fixed-width file into pandas dataframe.  The CRN is\n-    described in [1]_ and [2]_.\n+def read_crn(filename, map_variables=True):\n+    \"\"\"Read a NOAA USCRN fixed-width file into a pandas dataframe.\n+\n+    The CRN network consists of over 100 meteorological stations covering the\n+    U.S. and is described in [1]_ and [2]_. The primary goal of CRN is to\n+    provide long-term measurements of temperature, precipitation, and soil\n+    moisture and temperature. Additionally, global horizontal irradiance (GHI)\n+    is measured at each site using a photodiode pyranometer.\n \n     Parameters\n     ----------\n     filename: str, path object, or file-like\n         filepath or url to read for the fixed-width file.\n+    map_variables: boolean, default: True\n+        When true, renames columns of the Dataframe to pvlib variable names\n+        where applicable. See variable :const:`VARIABLE_MAP`.\n \n     Returns\n     -------\n@@ -60,12 +81,12 @@ def read_crn(filename):\n     -----\n     CRN files contain 5 minute averages labeled by the interval ending\n     time. Here, missing data is flagged as NaN, rather than the lowest\n-    possible integer for a field (e.g. -999 or -99). Air temperature in\n-    deg C. Wind speed in m/s at a height of 1.5 m above ground level.\n+    possible integer for a field (e.g. -999 or -99). Air temperature is in\n+    deg C and wind speed is in m/s at a height of 1.5 m above ground level.\n \n-    Variables corresponding to standard pvlib variables are renamed,\n+    Variables corresponding to standard pvlib variables are by default renamed,\n     e.g. `SOLAR_RADIATION` becomes `ghi`. See the\n-    `pvlib.iotools.crn.VARIABLE_MAP` dict for the complete mapping.\n+    :const:`pvlib.iotools.crn.VARIABLE_MAP` dict for the complete mapping.\n \n     CRN files occasionally have a set of null characters on a line\n     instead of valid data. This function drops those lines. Sometimes\n@@ -85,16 +106,13 @@ def read_crn(filename):\n        Amer. Meteor. Soc., 94, 489-498. :doi:`10.1175/BAMS-D-12-00170.1`\n     \"\"\"\n \n-    # read in data. set fields with NUL characters to NaN\n-    data = pd.read_fwf(filename, header=None, names=HEADERS.split(' '),\n-                       widths=WIDTHS, na_values=['\\x00\\x00\\x00\\x00\\x00\\x00'])\n-    # at this point we only have NaNs from NUL characters, not -999 etc.\n-    # these bad rows need to be removed so that dtypes can be set.\n-    # NaNs require float dtype so we run into errors if we don't do this.\n-    data = data.dropna(axis=0)\n-    # loop here because dtype kwarg not supported in read_fwf until 0.20\n-    for (col, _dtype) in zip(data.columns, DTYPES):\n-        data[col] = data[col].astype(_dtype)\n+    # read in data\n+    data = pd.read_fwf(filename, header=None, names=HEADERS, widths=WIDTHS,\n+                       na_values=NAN_DICT)\n+    # Remove rows with all nans\n+    data = data.dropna(axis=0, how='all')\n+    # set dtypes here because dtype kwarg not supported in read_fwf until 0.20\n+    data = data.astype(dict(zip(HEADERS, DTYPES)))\n \n     # set index\n     # UTC_TIME does not have leading 0s, so must zfill(4) to comply\n@@ -103,19 +121,8 @@ def read_crn(filename):\n     dtindex = pd.to_datetime(dts['UTC_DATE'] + dts['UTC_TIME'].str.zfill(4),\n                              format='%Y%m%d%H%M', utc=True)\n     data = data.set_index(dtindex)\n-    try:\n-        # to_datetime(utc=True) does not work in older versions of pandas\n-        data = data.tz_localize('UTC')\n-    except TypeError:\n-        pass\n-\n-    # Now we can set nans. This could be done a per column basis to be\n-    # safer, since in principle a real -99 value could occur in a -9999\n-    # column. Very unlikely to see that in the real world.\n-    for val in [-99, -999, -9999]:\n-        # consider replacing with .replace([-99, -999, -9999])\n-        data = data.where(data != val, np.nan)\n-\n-    data = data.rename(columns=VARIABLE_MAP)\n+\n+    if map_variables:\n+        data = data.rename(columns=VARIABLE_MAP)\n \n     return data\n", "test_patch": "diff --git a/pvlib/tests/iotools/test_crn.py b/pvlib/tests/iotools/test_crn.py\n--- a/pvlib/tests/iotools/test_crn.py\n+++ b/pvlib/tests/iotools/test_crn.py\n@@ -3,11 +3,11 @@\n from numpy import dtype, nan\n import pytest\n from pvlib.iotools import crn\n-from ..conftest import DATA_DIR, assert_frame_equal\n+from ..conftest import DATA_DIR, assert_frame_equal, assert_index_equal\n \n \n @pytest.fixture\n-def columns():\n+def columns_mapped():\n     return [\n         'WBANNO', 'UTC_DATE', 'UTC_TIME', 'LST_DATE', 'LST_TIME', 'CRX_VN',\n         'longitude', 'latitude', 'temp_air', 'PRECIPITATION', 'ghi',\n@@ -17,6 +17,16 @@ def columns():\n         'WETNESS', 'WET_FLAG', 'wind_speed', 'wind_speed_flag']\n \n \n+@pytest.fixture\n+def columns_unmapped():\n+    return [\n+        'WBANNO', 'UTC_DATE', 'UTC_TIME', 'LST_DATE', 'LST_TIME', 'CRX_VN',\n+        'LONGITUDE', 'LATITUDE', 'AIR_TEMPERATURE', 'PRECIPITATION',\n+        'SOLAR_RADIATION', 'SR_FLAG', 'SURFACE_TEMPERATURE', 'ST_TYPE',\n+        'ST_FLAG', 'RELATIVE_HUMIDITY', 'RH_FLAG', 'SOIL_MOISTURE_5',\n+        'SOIL_TEMPERATURE_5', 'WETNESS', 'WET_FLAG', 'WIND_1_5', 'WIND_FLAG']\n+\n+\n @pytest.fixture\n def dtypes():\n     return [\n@@ -39,7 +49,7 @@ def testfile_problems():\n     return DATA_DIR / 'CRN_with_problems.txt'\n \n \n-def test_read_crn(testfile, columns, dtypes):\n+def test_read_crn(testfile, columns_mapped, dtypes):\n     index = pd.DatetimeIndex(['2019-01-01 16:10:00',\n                               '2019-01-01 16:15:00',\n                               '2019-01-01 16:20:00',\n@@ -54,25 +64,31 @@ def test_read_crn(testfile, columns, dtypes):\n          0.0, 340.0, 0, 4.3, 'C', 0, 83.0, 0, nan, nan, 1183, 0, 0.53, 0],\n         [53131, 20190101, 1625, 20190101, 925, 3, -111.17, 32.24, 4.0,\n          0.0, 393.0, 0, 4.8, 'C', 0, 81.0, 0, nan, nan, 1223, 0, 0.64, 0]])\n-    expected = pd.DataFrame(values, columns=columns, index=index)\n+    expected = pd.DataFrame(values, columns=columns_mapped, index=index)\n     for (col, _dtype) in zip(expected.columns, dtypes):\n         expected[col] = expected[col].astype(_dtype)\n     out = crn.read_crn(testfile)\n     assert_frame_equal(out, expected)\n \n \n-def test_read_crn_problems(testfile_problems, columns, dtypes):\n+# Test map_variables=False returns correct column names\n+def test_read_crn_map_variables(testfile, columns_unmapped, dtypes):\n+    out = crn.read_crn(testfile, map_variables=False)\n+    assert_index_equal(out.columns, pd.Index(columns_unmapped))\n+\n+\n+def test_read_crn_problems(testfile_problems, columns_mapped, dtypes):\n     # GH1025\n     index = pd.DatetimeIndex(['2020-07-06 12:00:00',\n                               '2020-07-06 13:10:00'],\n                              freq=None).tz_localize('UTC')\n     values = np.array([\n-        [92821, 20200706, 1200, 20200706, 700, '3', -80.69, 28.62, 24.9,\n-         0.0, 190.0, 0, 25.5, 'C', 0, 93.0, 0, nan, nan, 990, 0, 1.57, 0],\n+        [92821, 20200706, 1200, 20200706, 700, '3.0', -80.69, 28.62, 24.9,\n+         0.0, np.nan, 0, 25.5, 'C', 0, 93.0, 0, nan, nan, 990, 0, 1.57, 0],\n         [92821, 20200706, 1310, 20200706, 810, '2.623', -80.69, 28.62,\n          26.9, 0.0, 430.0, 0, 30.2, 'C', 0, 87.0, 0, nan, nan, 989, 0,\n          1.64, 0]])\n-    expected = pd.DataFrame(values, columns=columns, index=index)\n+    expected = pd.DataFrame(values, columns=columns_mapped, index=index)\n     for (col, _dtype) in zip(expected.columns, dtypes):\n         expected[col] = expected[col].astype(_dtype)\n     out = crn.read_crn(testfile_problems)\n", "problem_statement": "`read_crn` returns -99999 instead of `NaN`\n**Describe the bug**\r\n`read_crn` fails to map -99999 to `NaN`\r\n\r\n**To Reproduce**\r\n```python\r\nfrom pvlib.iotools import read_crn\r\ncrn = read_crn('https://www.ncei.noaa.gov/pub/data/uscrn/products/subhourly01/2021/CRNS0101-05-2021-NY_Millbrook_3_W.txt')\r\ncrn.loc['2021-12-14 0930':'2021-12-14 1130', 'ghi']\r\n2021-12-14 09:30:00+00:00        0.0\r\n2021-12-14 09:35:00+00:00        0.0\r\n2021-12-14 09:40:00+00:00        0.0\r\n2021-12-14 09:45:00+00:00        0.0\r\n2021-12-14 09:50:00+00:00        0.0\r\n2021-12-14 09:55:00+00:00        0.0\r\n2021-12-14 10:00:00+00:00        0.0\r\n2021-12-14 10:05:00+00:00   -99999.0\r\n2021-12-14 10:10:00+00:00   -99999.0\r\n2021-12-14 10:15:00+00:00   -99999.0\r\n2021-12-14 10:20:00+00:00   -99999.0\r\n2021-12-14 10:25:00+00:00   -99999.0\r\n2021-12-14 10:30:00+00:00   -99999.0\r\n2021-12-14 10:35:00+00:00   -99999.0\r\n2021-12-14 10:40:00+00:00   -99999.0\r\n2021-12-14 10:45:00+00:00   -99999.0\r\n2021-12-14 10:50:00+00:00   -99999.0\r\n2021-12-14 10:55:00+00:00   -99999.0\r\n2021-12-14 11:00:00+00:00   -99999.0\r\n2021-12-14 11:05:00+00:00        0.0\r\n2021-12-14 11:10:00+00:00        0.0\r\n2021-12-14 11:15:00+00:00        0.0\r\n2021-12-14 11:20:00+00:00        0.0\r\n2021-12-14 11:25:00+00:00        0.0\r\n2021-12-14 11:30:00+00:00        0.0\r\nName: ghi, dtype: float64\r\n```\r\n\r\n**Expected behavior**\r\nShould return `NaN` instead of -99999\r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: 0.9.0\r\n - ``pandas.__version__``: 1.0.3 (doesn't matter)\r\n - python: 3.7\r\n\r\n**Additional context**\r\n\r\nDocumentation [here](https://www.ncei.noaa.gov/pub/data/uscrn/products/subhourly01/) says\r\n\r\n>          C.  Missing data are indicated by the lowest possible integer for a \r\n>             given column format, such as -9999.0 for 7-character fields with \r\n>             one decimal place or -99.000 for 7-character fields with three\r\n>             decimal places.\r\n\r\nSo we should change \r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/1ab0eb20f9cd9fb9f7a0ddf35f81283f2648e34a/pvlib/iotools/crn.py#L112-L117\r\n\r\nto include -99999 and perhaps -999999. Or do the smarter thing as discussed in the comment.\r\n\r\nalso https://github.com/SolarArbiter/solarforecastarbiter-core/issues/773\n", "hints_text": "", "created_at": "2022-01-05T22:59:26Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/iotools/test_crn.py::test_read_crn_map_variables"], "PASS_TO_PASS": ["pvlib/tests/iotools/test_crn.py::test_read_crn"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-807", "base_commit": "e326fa53038f616d949e4f981dab6187d2ca9470", "patch": "diff --git a/pvlib/scaling.py b/pvlib/scaling.py\nnew file mode 100644\n--- /dev/null\n+++ b/pvlib/scaling.py\n@@ -0,0 +1,242 @@\n+\"\"\"\n+The ``scaling`` module contains functions for manipulating irradiance\n+or other variables to account for temporal or spatial characteristics.\n+\"\"\"\n+\n+import numpy as np\n+import pandas as pd\n+\n+\n+def wvm(clearsky_index, positions, cloud_speed, dt=None):\n+    \"\"\"\n+    Compute spatial aggregation time series smoothing on clear sky index based\n+    on the Wavelet Variability model of Lave et al [1-2]. Implementation is\n+    basically a port of the Matlab version of the code [3].\n+\n+    Parameters\n+    ----------\n+    clearsky_index : numeric or pandas.Series\n+        Clear Sky Index time series that will be smoothed.\n+\n+    positions : numeric\n+        Array of coordinate distances as (x,y) pairs representing the\n+        easting, northing of the site positions in meters [m]. Distributed\n+        plants could be simulated by gridded points throughout the plant\n+        footprint.\n+\n+    cloud_speed : numeric\n+        Speed of cloud movement in meters per second [m/s].\n+\n+    dt : float, default None\n+        The time series time delta. By default, is inferred from the\n+        clearsky_index. Must be specified for a time series that doesn't\n+        include an index. Units of seconds [s].\n+\n+    Returns\n+    -------\n+    smoothed : numeric or pandas.Series\n+        The Clear Sky Index time series smoothed for the described plant.\n+\n+    wavelet: numeric\n+        The individual wavelets for the time series before smoothing.\n+\n+    tmscales: numeric\n+        The timescales associated with the wavelets in seconds [s].\n+\n+    References\n+    ----------\n+    [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n+    Model (WVM) for Solar PV Power Plants. IEEE Transactions on Sustainable\n+    Energy, vol. 4, no. 2, pp. 501-509, 2013.\n+\n+    [2] M. Lave and J. Kleissl. Cloud speed impact on solar variability\n+    scaling - Application to the wavelet variability model. Solar Energy,\n+    vol. 91, pp. 11-21, 2013.\n+\n+    [3] Wavelet Variability Model - Matlab Code:\n+    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    \"\"\"\n+\n+    # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n+\n+    try:\n+        import scipy.optimize\n+        from scipy.spatial.distance import pdist\n+    except ImportError:\n+        raise ImportError(\"The WVM function requires scipy.\")\n+\n+    pos = np.array(positions)\n+    dist = pdist(pos, 'euclidean')\n+    wavelet, tmscales = _compute_wavelet(clearsky_index, dt)\n+\n+    # Find effective length of position vector, 'dist' is full pairwise\n+    n_pairs = len(dist)\n+\n+    def fn(x):\n+        return np.abs((x ** 2 - x) / 2 - n_pairs)\n+    n_dist = np.round(scipy.optimize.fmin(fn, np.sqrt(n_pairs), disp=False))\n+\n+    # Compute VR\n+    A = cloud_speed / 2  # Resultant fit for A from [2]\n+    vr = np.zeros(tmscales.shape)\n+    for i, tmscale in enumerate(tmscales):\n+        rho = np.exp(-1 / A * dist / tmscale)  # Eq 5 from [1]\n+\n+        # 2*rho is because rho_ij = rho_ji. +n_dist accounts for sum(rho_ii=1)\n+        denominator = 2 * np.sum(rho) + n_dist\n+        vr[i] = n_dist ** 2 / denominator  # Eq 6 of [1]\n+\n+    # Scale each wavelet by VR (Eq 7 in [1])\n+    wavelet_smooth = np.zeros_like(wavelet)\n+    for i in np.arange(len(tmscales)):\n+        if i < len(tmscales) - 1:  # Treat the lowest freq differently\n+            wavelet_smooth[i, :] = wavelet[i, :] / np.sqrt(vr[i])\n+        else:\n+            wavelet_smooth[i, :] = wavelet[i, :]\n+\n+    outsignal = np.sum(wavelet_smooth, 0)\n+\n+    try:  # See if there's an index already, if so, return as a pandas Series\n+        smoothed = pd.Series(outsignal, index=clearsky_index.index)\n+    except AttributeError:\n+        smoothed = outsignal  # just output the numpy signal\n+\n+    return smoothed, wavelet, tmscales\n+\n+\n+def latlon_to_xy(coordinates):\n+    \"\"\"\n+    Convert latitude and longitude in degrees to a coordinate system measured\n+    in meters from zero deg latitude, zero deg longitude.\n+\n+    This is a convenience method to support inputs to wvm. Note that the\n+    methodology used is only suitable for short distances. For conversions of\n+    longer distances, users should consider use of Universal Transverse\n+    Mercator (UTM) or other suitable cartographic projection. Consider\n+    packages built for cartographic projection such as pyproj (e.g.\n+    pyproj.transform()) [2].\n+\n+    Parameters\n+    ----------\n+\n+    coordinates : numeric\n+        Array or list of (latitude, longitude) coordinate pairs. Use decimal\n+        degrees notation.\n+\n+    Returns\n+    -------\n+    xypos : numeric\n+        Array of coordinate distances as (x,y) pairs representing the\n+        easting, northing of the position in meters [m].\n+\n+    References\n+    ----------\n+    [1] H. Moritz. Geodetic Reference System 1980, Journal of Geodesy, vol. 74,\n+    no. 1, pp 128\u2013133, 2000.\n+\n+    [2] https://pypi.org/project/pyproj/\n+\n+    [3] Wavelet Variability Model - Matlab Code:\n+    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    \"\"\"\n+\n+    # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n+\n+    r_earth = 6371008.7714  # mean radius of Earth, in meters\n+    m_per_deg_lat = r_earth * np.pi / 180\n+    try:\n+        meanlat = np.mean([lat for (lat, lon) in coordinates])  # Mean latitude\n+    except TypeError:  # Assume it's a single value?\n+        meanlat = coordinates[0]\n+    m_per_deg_lon = r_earth * np.cos(np.pi/180 * meanlat) * np.pi/180\n+\n+    # Conversion\n+    pos = coordinates * np.array(m_per_deg_lat, m_per_deg_lon)\n+\n+    # reshape as (x,y) pairs to return\n+    try:\n+        return np.column_stack([pos[:, 1], pos[:, 0]])\n+    except IndexError:  # Assume it's a single value, which has a 1D shape\n+        return np.array((pos[1], pos[0]))\n+\n+\n+def _compute_wavelet(clearsky_index, dt=None):\n+    \"\"\"\n+    Compute the wavelet transform on the input clear_sky time series.\n+\n+    Parameters\n+    ----------\n+    clearsky_index : numeric or pandas.Series\n+        Clear Sky Index time series that will be smoothed.\n+\n+    dt : float, default None\n+        The time series time delta. By default, is inferred from the\n+        clearsky_index. Must be specified for a time series that doesn't\n+        include an index. Units of seconds [s].\n+\n+    Returns\n+    -------\n+    wavelet: numeric\n+        The individual wavelets for the time series\n+\n+    tmscales: numeric\n+        The timescales associated with the wavelets in seconds [s]\n+\n+    References\n+    ----------\n+    [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n+    Model (WVM) for Solar PV Power Plants. IEEE Transactions on Sustainable\n+    Energy, vol. 4, no. 2, pp. 501-509, 2013.\n+\n+    [3] Wavelet Variability Model - Matlab Code:\n+    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    \"\"\"\n+\n+    # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n+\n+    try:  # Assume it's a pandas type\n+        vals = clearsky_index.values.flatten()\n+    except AttributeError:  # Assume it's a numpy type\n+        vals = clearsky_index.flatten()\n+        if dt is None:\n+            raise ValueError(\"dt must be specified for numpy type inputs.\")\n+    else:  # flatten() succeeded, thus it's a pandas type, so get its dt\n+        try:  # Assume it's a time series type index\n+            dt = (clearsky_index.index[1] - clearsky_index.index[0]).seconds\n+        except AttributeError:  # It must just be a numeric index\n+            dt = (clearsky_index.index[1] - clearsky_index.index[0])\n+\n+    # Pad the series on both ends in time and place in a dataframe\n+    cs_long = np.pad(vals, (len(vals), len(vals)), 'symmetric')\n+    cs_long = pd.DataFrame(cs_long)\n+\n+    # Compute wavelet time scales\n+    min_tmscale = np.ceil(np.log(dt)/np.log(2))  # Minimum wavelet timescale\n+    max_tmscale = int(12 - min_tmscale)  # maximum wavelet timescale\n+\n+    tmscales = np.zeros(max_tmscale)\n+    csi_mean = np.zeros([max_tmscale, len(cs_long)])\n+    # Loop for all time scales we will consider\n+    for i in np.arange(0, max_tmscale):\n+        j = i+1\n+        tmscales[i] = 2**j * dt  # Wavelet integration time scale\n+        intvlen = 2**j  # Wavelet integration time series interval\n+        # Rolling average, retains only lower frequencies than interval\n+        df = cs_long.rolling(window=intvlen, center=True, min_periods=1).mean()\n+        # Fill nan's in both directions\n+        df = df.fillna(method='bfill').fillna(method='ffill')\n+        # Pop values back out of the dataframe and store\n+        csi_mean[i, :] = df.values.flatten()\n+\n+    # Calculate the wavelets by isolating the rolling mean frequency ranges\n+    wavelet_long = np.zeros(csi_mean.shape)\n+    for i in np.arange(0, max_tmscale-1):\n+        wavelet_long[i, :] = csi_mean[i, :] - csi_mean[i+1, :]\n+    wavelet_long[max_tmscale-1, :] = csi_mean[max_tmscale-1, :]  # Lowest freq\n+\n+    # Clip off the padding and just return the original time window\n+    wavelet = np.zeros([max_tmscale, len(vals)])\n+    for i in np.arange(0, max_tmscale):\n+        wavelet[i, :] = wavelet_long[i, len(vals)+1: 2*len(vals)+1]\n+\n+    return wavelet, tmscales\n", "test_patch": "diff --git a/pvlib/test/test_scaling.py b/pvlib/test/test_scaling.py\nnew file mode 100644\n--- /dev/null\n+++ b/pvlib/test/test_scaling.py\n@@ -0,0 +1,146 @@\n+import numpy as np\n+import pandas as pd\n+\n+import pytest\n+from numpy.testing import assert_almost_equal\n+\n+from pvlib import scaling\n+from conftest import requires_scipy\n+\n+\n+# Sample cloud speed\n+cloud_speed = 5\n+\n+# Sample dt\n+dt = 1\n+\n+\n+@pytest.fixture\n+def coordinates():\n+    # Sample positions in lat/lon\n+    lat = np.array((9.99, 10, 10.01))\n+    lon = np.array((4.99, 5, 5.01))\n+    coordinates = np.array([(lati, loni) for (lati, loni) in zip(lat, lon)])\n+    return coordinates\n+\n+\n+@pytest.fixture\n+def clear_sky_index():\n+    # Generate a sample clear_sky_index\n+    clear_sky_index = np.ones(10000)\n+    clear_sky_index[5000:5005] = np.array([1, 1, 1.1, 0.9, 1])\n+    return clear_sky_index\n+\n+\n+@pytest.fixture\n+def time(clear_sky_index):\n+    # Sample time vector\n+    return np.arange(0, len(clear_sky_index))\n+\n+\n+@pytest.fixture\n+def positions():\n+    # Sample positions based on the previous lat/lon (calculated manually)\n+    expect_xpos = np.array([554863.4, 555975.4, 557087.3])\n+    expect_ypos = np.array([1110838.8, 1111950.8, 1113062.7])\n+    return np.array([pt for pt in zip(expect_xpos, expect_ypos)])\n+\n+\n+@pytest.fixture\n+def expect_tmscale():\n+    # Expected timescales for dt = 1\n+    return [2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096]\n+\n+\n+@pytest.fixture\n+def expect_wavelet():\n+    # Expected wavelet for indices 5000:5004 for clear_sky_index above (Matlab)\n+    return np.array([[-0.025, 0.05, 0., -0.05, 0.025],\n+                     [0.025, 0., 0., 0., -0.025],\n+                     [0., 0., 0., 0., 0.]])\n+\n+\n+@pytest.fixture\n+def expect_cs_smooth():\n+    # Expected smoothed clear sky index for indices 5000:5004 (Matlab)\n+    return np.array([1., 1.0289, 1., 0.9711, 1.])\n+\n+\n+def test_latlon_to_xy_zero():\n+    coord = [0, 0]\n+    pos_e = [0, 0]\n+    pos = scaling.latlon_to_xy(coord)\n+    assert_almost_equal(pos, pos_e, decimal=1)\n+\n+\n+def test_latlon_to_xy_single(coordinates, positions):\n+    # Must test against central value, because latlon_to_xy uses the mean\n+    coord = coordinates[1]\n+    pos = scaling.latlon_to_xy(coord)\n+    assert_almost_equal(pos, positions[1], decimal=1)\n+\n+\n+def test_latlon_to_xy_array(coordinates, positions):\n+    pos = scaling.latlon_to_xy(coordinates)\n+    assert_almost_equal(pos, positions, decimal=1)\n+\n+\n+def test_latlon_to_xy_list(coordinates, positions):\n+    pos = scaling.latlon_to_xy(coordinates.tolist())\n+    assert_almost_equal(pos, positions, decimal=1)\n+\n+\n+def test_compute_wavelet_series(clear_sky_index, time,\n+                                expect_tmscale, expect_wavelet):\n+    csi_series = pd.Series(clear_sky_index, index=time)\n+    wavelet, tmscale = scaling._compute_wavelet(csi_series)\n+    assert_almost_equal(tmscale, expect_tmscale)\n+    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+\n+\n+def test_compute_wavelet_series_numindex(clear_sky_index, time,\n+                                         expect_tmscale, expect_wavelet):\n+    dtindex = pd.to_datetime(time, unit='s')\n+    csi_series = pd.Series(clear_sky_index, index=dtindex)\n+    wavelet, tmscale = scaling._compute_wavelet(csi_series)\n+    assert_almost_equal(tmscale, expect_tmscale)\n+    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+\n+\n+def test_compute_wavelet_array(clear_sky_index,\n+                               expect_tmscale, expect_wavelet):\n+    wavelet, tmscale = scaling._compute_wavelet(clear_sky_index, dt)\n+    assert_almost_equal(tmscale, expect_tmscale)\n+    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+\n+\n+def test_compute_wavelet_array_invalid(clear_sky_index):\n+    with pytest.raises(ValueError):\n+        scaling._compute_wavelet(clear_sky_index)\n+\n+\n+@requires_scipy\n+def test_wvm_series(clear_sky_index, time, positions, expect_cs_smooth):\n+    csi_series = pd.Series(clear_sky_index, index=time)\n+    cs_sm, _, _ = scaling.wvm(csi_series, positions, cloud_speed)\n+    assert_almost_equal(cs_sm[5000:5005], expect_cs_smooth, decimal=4)\n+\n+\n+@requires_scipy\n+def test_wvm_array(clear_sky_index, positions, expect_cs_smooth):\n+    cs_sm, _, _ = scaling.wvm(clear_sky_index, positions, cloud_speed, dt=dt)\n+    assert_almost_equal(cs_sm[5000:5005], expect_cs_smooth, decimal=4)\n+\n+\n+@requires_scipy\n+def test_wvm_series_xyaslist(clear_sky_index, time, positions,\n+                             expect_cs_smooth):\n+    csi_series = pd.Series(clear_sky_index, index=time)\n+    cs_sm, _, _ = scaling.wvm(csi_series, positions.tolist(), cloud_speed)\n+    assert_almost_equal(cs_sm[5000:5005], expect_cs_smooth, decimal=4)\n+\n+\n+@requires_scipy\n+def test_wvm_invalid(clear_sky_index, positions):\n+    with pytest.raises(ValueError):\n+        scaling.wvm(clear_sky_index, positions, cloud_speed)\n", "problem_statement": "Add Wavelet Variability Model (WVM) for calculating spatial smoothing of irradiance\n> > Should I spin this off to a separate issue, since it might be different (and more compartmented) than the broader downscaling discussion?\r\n> \r\n> Yes. Let's start a new module with this submission, `scaling.py` comes to mind, but I'm not enamored of it. Scope will be functions that operate on irradiance, perhaps other variables, to transform temporal or spatial characteristics.\r\n\r\nSpinoff from [issue #788 ](https://github.com/pvlib/pvlib-python/issues/788). Implementation is a python port of WVM, released as an auxiliary to the Matlab pvlib [here](https://pvpmc.sandia.gov/applications/wavelet-variability-model/). My implementation ports the original model logic, but deviates from the overall package, in that I begin at the point where the user already has a clear sky index to operate on (original starts from GHI and calculates POA clear sky index). I thought this would allow for more flexibility in choice of transposition model, etc, but it does ask a bit more work up front for a user to run the WVM.\r\n\r\nI am close to completion of a draft and will create a pull request when ready. This is my first contribution to the project (or any open source project really), so please accept my apologies in advance if it takes some guidance.\n", "hints_text": "> This is my first contribution to the project (or any open source project really), so please accept my apologies in advance if it takes some guidance.\r\n\r\nWelcome!  Asking for a clear-sky index as input seems appropriate; there's no need to rigidly follow the MATLAB implementation. I'll ask for your patience with the review process, which can involve multiple iterations and reviewers.", "created_at": "2019-11-01T14:54:52Z", "version": "0.6", "FAIL_TO_PASS": ["pvlib/test/test_scaling.py::test_latlon_to_xy_zero", "pvlib/test/test_scaling.py::test_latlon_to_xy_single", "pvlib/test/test_scaling.py::test_latlon_to_xy_array", "pvlib/test/test_scaling.py::test_latlon_to_xy_list", "pvlib/test/test_scaling.py::test_compute_wavelet_series", "pvlib/test/test_scaling.py::test_compute_wavelet_series_numindex", "pvlib/test/test_scaling.py::test_compute_wavelet_array", "pvlib/test/test_scaling.py::test_compute_wavelet_array_invalid", "pvlib/test/test_scaling.py::test_wvm_series", "pvlib/test/test_scaling.py::test_wvm_array", "pvlib/test/test_scaling.py::test_wvm_series_xyaslist", "pvlib/test/test_scaling.py::test_wvm_invalid"], "PASS_TO_PASS": [], "environment_setup_commit": "b91d178868d193afd56f8e3b013661a473d699c3"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-1017", "base_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3", "patch": "diff --git a/pydicom/filewriter.py b/pydicom/filewriter.py\n--- a/pydicom/filewriter.py\n+++ b/pydicom/filewriter.py\n@@ -955,35 +955,40 @@ def dcmwrite(filename, dataset, write_like_original=True):\n # for write_numbers, the Writer maps to a tuple (function, struct_format)\n #   (struct_format is python's struct module format)\n writers = {\n-    'UL': (write_numbers, 'L'),\n-    'SL': (write_numbers, 'l'),\n-    'US': (write_numbers, 'H'),\n-    'SS': (write_numbers, 'h'),\n-    'FL': (write_numbers, 'f'),\n+    'AE': (write_string, None),\n+    'AS': (write_string, None),\n+    'AT': (write_ATvalue, None),\n+    'CS': (write_string, None),\n+    'DA': (write_DA, None),\n+    'DS': (write_number_string, None),\n+    'DT': (write_DT, None),\n     'FD': (write_numbers, 'd'),\n-    'OF': (write_numbers, 'f'),\n+    'FL': (write_numbers, 'f'),\n+    'IS': (write_number_string, None),\n+    'LO': (write_text, None),\n+    'LT': (write_text, None),\n     'OB': (write_OBvalue, None),\n     'OD': (write_OWvalue, None),\n+    'OF': (write_numbers, 'f'),\n     'OL': (write_OWvalue, None),\n-    'UI': (write_UI, None),\n-    'SH': (write_text, None),\n-    'DA': (write_DA, None),\n-    'TM': (write_TM, None),\n-    'CS': (write_string, None),\n+    'OW': (write_OWvalue, None),\n+    'OV': (write_OWvalue, None),\n     'PN': (write_PN, None),\n-    'LO': (write_text, None),\n-    'IS': (write_number_string, None),\n-    'DS': (write_number_string, None),\n-    'AE': (write_string, None),\n-    'AS': (write_string, None),\n-    'LT': (write_text, None),\n+    'SH': (write_text, None),\n+    'SL': (write_numbers, 'l'),\n     'SQ': (write_sequence, None),\n+    'SS': (write_numbers, 'h'),\n+    'ST': (write_text, None),\n+    'SV': (write_numbers, 'q'),\n+    'TM': (write_TM, None),\n     'UC': (write_text, None),\n+    'UI': (write_UI, None),\n+    'UL': (write_numbers, 'L'),\n     'UN': (write_UN, None),\n     'UR': (write_string, None),\n-    'AT': (write_ATvalue, None),\n-    'ST': (write_text, None),\n-    'OW': (write_OWvalue, None),\n+    'US': (write_numbers, 'H'),\n+    'UT': (write_text, None),\n+    'UV': (write_numbers, 'Q'),\n     'US or SS': (write_OWvalue, None),\n     'US or OW': (write_OWvalue, None),\n     'US or SS or OW': (write_OWvalue, None),\n@@ -991,6 +996,4 @@ def dcmwrite(filename, dataset, write_like_original=True):\n     'OB/OW': (write_OBvalue, None),\n     'OB or OW': (write_OBvalue, None),\n     'OW or OB': (write_OBvalue, None),\n-    'DT': (write_DT, None),\n-    'UT': (write_text, None),\n }  # note OW/OB depends on other items, which we don't know at write time\ndiff --git a/pydicom/values.py b/pydicom/values.py\n--- a/pydicom/values.py\n+++ b/pydicom/values.py\n@@ -300,6 +300,15 @@ def convert_OWvalue(byte_string, is_little_endian, struct_format=None):\n     return convert_OBvalue(byte_string, is_little_endian)\n \n \n+def convert_OVvalue(byte_string, is_little_endian, struct_format=None):\n+    \"\"\"Return the encoded 'OV' value as :class:`bytes` or :class:`str`.\n+\n+    No byte swapping will be performed.\n+    \"\"\"\n+    # for now, Maybe later will have own routine\n+    return convert_OBvalue(byte_string, is_little_endian)\n+\n+\n def convert_PN(byte_string, encodings=None):\n     \"\"\"Return a decoded 'PN' value.\n \n@@ -635,35 +644,40 @@ def convert_value(VR, raw_data_element, encodings=None):\n # (function, struct_format)\n # (struct_format in python struct module style)\n converters = {\n-    'UL': (convert_numbers, 'L'),\n-    'SL': (convert_numbers, 'l'),\n-    'US': (convert_numbers, 'H'),\n-    'SS': (convert_numbers, 'h'),\n-    'FL': (convert_numbers, 'f'),\n+    'AE': convert_AE_string,\n+    'AS': convert_string,\n+    'AT': convert_ATvalue,\n+    'CS': convert_string,\n+    'DA': convert_DA_string,\n+    'DS': convert_DS_string,\n+    'DT': convert_DT_string,\n     'FD': (convert_numbers, 'd'),\n-    'OF': (convert_numbers, 'f'),\n+    'FL': (convert_numbers, 'f'),\n+    'IS': convert_IS_string,\n+    'LO': convert_text,\n+    'LT': convert_single_string,\n     'OB': convert_OBvalue,\n     'OD': convert_OBvalue,\n+    'OF': (convert_numbers, 'f'),\n     'OL': convert_OBvalue,\n-    'UI': convert_UI,\n-    'SH': convert_text,\n-    'DA': convert_DA_string,\n-    'TM': convert_TM_string,\n-    'CS': convert_string,\n+    'OW': convert_OWvalue,\n+    'OV': convert_OVvalue,\n     'PN': convert_PN,\n-    'LO': convert_text,\n-    'IS': convert_IS_string,\n-    'DS': convert_DS_string,\n-    'AE': convert_AE_string,\n-    'AS': convert_string,\n-    'LT': convert_single_string,\n+    'SH': convert_text,\n+    'SL': (convert_numbers, 'l'),\n     'SQ': convert_SQ,\n+    'SS': (convert_numbers, 'h'),\n+    'ST': convert_single_string,\n+    'SV': (convert_numbers, 'q'),\n+    'TM': convert_TM_string,\n     'UC': convert_text,\n+    'UI': convert_UI,\n+    'UL': (convert_numbers, 'L'),\n     'UN': convert_UN,\n     'UR': convert_UR_string,\n-    'AT': convert_ATvalue,\n-    'ST': convert_single_string,\n-    'OW': convert_OWvalue,\n+    'US': (convert_numbers, 'H'),\n+    'UT': convert_single_string,\n+    'UV': (convert_numbers, 'Q'),\n     'OW/OB': convert_OBvalue,  # note OW/OB depends on other items,\n     'OB/OW': convert_OBvalue,  # which we don't know at read time\n     'OW or OB': convert_OBvalue,\n@@ -672,6 +686,4 @@ def convert_value(VR, raw_data_element, encodings=None):\n     'US or OW': convert_OWvalue,\n     'US or SS or OW': convert_OWvalue,\n     'US\\\\US or SS\\\\US': convert_OWvalue,\n-    'DT': convert_DT_string,\n-    'UT': convert_single_string,\n }\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@@ -17,6 +17,7 @@\n from pydicom import config\n from pydicom.dataset import Dataset, FileDataset\n from pydicom.data import get_testdata_files\n+from pydicom.datadict import add_dict_entries\n from pydicom.filereader import dcmread, read_dataset\n from pydicom.dataelem import DataElement, DataElement_from_raw\n from pydicom.errors import InvalidDicomError\n@@ -837,6 +838,31 @@ def setup(self):\n         ds.URNCodeValue = 'http://test.com'  # VR of UR\n         ds.RetrieveURL = 'ftp://test.com  '  # Test trailing spaces ignored\n         ds.DestinationAE = '    TEST  12    '  # 16 characters max for AE\n+        # 8-byte values\n+        ds.ExtendedOffsetTable = (  # VR of OV\n+            b'\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00'\n+            b'\\x01\\x02\\x03\\x04\\x05\\x06\\x07\\x08'\n+        )\n+\n+        # No public elements with VR of SV or UV yet...\n+        add_dict_entries({\n+            0xFFFE0001: (\n+                'SV', '1', 'SV Element Minimum', '', 'SVElementMinimum'\n+            ),\n+            0xFFFE0002: (\n+                'SV', '1', 'SV Element Maximum', '', 'SVElementMaximum'\n+            ),\n+            0xFFFE0003: (\n+                'UV', '1', 'UV Element Minimum', '', 'UVElementMinimum'\n+            ),\n+            0xFFFE0004: (\n+                'UV', '1', 'UV Element Maximum', '', 'UVElementMaximum'\n+            ),\n+        })\n+        ds.SVElementMinimum = -2**63\n+        ds.SVElementMaximum = 2**63 - 1\n+        ds.UVElementMinimum = 0\n+        ds.UVElementMaximum = 2**64 - 1\n \n         self.fp = BytesIO()  # Implicit little\n         file_ds = FileDataset(self.fp, ds)\n@@ -939,6 +965,113 @@ def test_read_AE(self):\n         ds = dcmread(self.fp, force=True)\n         assert 'TEST  12' == ds.DestinationAE\n \n+    def test_read_OV_implicit_little(self):\n+        \"\"\"Check reading element with VR of OV encoded as implicit\"\"\"\n+        ds = dcmread(self.fp, force=True)\n+        val = (\n+            b'\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00'\n+            b'\\x01\\x02\\x03\\x04\\x05\\x06\\x07\\x08'\n+        )\n+        elem = ds['ExtendedOffsetTable']\n+        assert 'OV' == elem.VR\n+        assert 0x7FE00001 == elem.tag\n+        assert val == elem.value\n+\n+        new = DataElement(0x7FE00001, 'OV', val)\n+        assert elem == new\n+\n+    def test_read_OV_explicit_little(self):\n+        \"\"\"Check reading element with VR of OV encoded as explicit\"\"\"\n+        ds = dcmread(self.fp_ex, force=True)\n+        val = (\n+            b'\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00'\n+            b'\\x01\\x02\\x03\\x04\\x05\\x06\\x07\\x08'\n+        )\n+        elem = ds['ExtendedOffsetTable']\n+        assert 'OV' == elem.VR\n+        assert 0x7FE00001 == elem.tag\n+        assert val == elem.value\n+\n+        new = DataElement(0x7FE00001, 'OV', val)\n+        assert elem == new\n+\n+    def test_read_SV_implicit_little(self):\n+        \"\"\"Check reading element with VR of SV encoded as implicit\"\"\"\n+        ds = dcmread(self.fp, force=True)\n+        elem = ds['SVElementMinimum']\n+        assert 'SV' == elem.VR\n+        assert 0xFFFE0001 == elem.tag\n+        assert -2**63 == elem.value\n+\n+        new = DataElement(0xFFFE0001, 'SV', -2**63)\n+        assert elem == new\n+\n+        elem = ds['SVElementMaximum']\n+        assert 'SV' == elem.VR\n+        assert 0xFFFE0002 == elem.tag\n+        assert 2**63 - 1 == elem.value\n+\n+        new = DataElement(0xFFFE0002, 'SV', 2**63 - 1)\n+        assert elem == new\n+\n+    @pytest.mark.skip(\"No public elements with VR of SV\")\n+    def test_read_SV_explicit_little(self):\n+        \"\"\"Check reading element with VR of SV encoded as explicit\"\"\"\n+        ds = dcmread(self.fp_ex, force=True)\n+        elem = ds['SVElementMinimum']\n+        assert 'SV' == elem.VR\n+        assert 0xFFFE0001 == elem.tag\n+        assert -2**63 == elem.value\n+\n+        new = DataElement(0xFFFE0001, 'SV', -2**63)\n+        assert elem == new\n+\n+        elem = ds['SVElementMaximum']\n+        assert 'SV' == elem.VR\n+        assert 0xFFFE0002 == elem.tag\n+        assert 2**63 - 1 == elem.value\n+\n+        new = DataElement(0xFFFE0002, 'SV', 2**63 - 1)\n+        assert elem == new\n+\n+    def test_read_UV_implicit_little(self):\n+        \"\"\"Check reading element with VR of UV encoded as implicit\"\"\"\n+        ds = dcmread(self.fp, force=True)\n+        elem = ds['UVElementMinimum']\n+        assert 'UV' == elem.VR\n+        assert 0xFFFE0003 == elem.tag\n+        assert 0 == elem.value\n+\n+        new = DataElement(0xFFFE0003, 'UV', 0)\n+        assert elem == new\n+\n+        elem = ds['UVElementMaximum']\n+        assert 'UV' == elem.VR\n+        assert 0xFFFE0004 == elem.tag\n+        assert 2**64 - 1 == elem.value\n+\n+        new = DataElement(0xFFFE0004, 'UV', 2**64 - 1)\n+        assert elem == new\n+\n+    def test_read_UV_explicit_little(self):\n+        \"\"\"Check reading element with VR of UV encoded as explicit\"\"\"\n+        ds = dcmread(self.fp_ex, force=True)\n+        elem = ds['UVElementMinimum']\n+        assert 'UV' == elem.VR\n+        assert 0xFFFE0003 == elem.tag\n+        assert 0 == elem.value\n+\n+        new = DataElement(0xFFFE0003, 'UV', 0)\n+        assert elem == new\n+\n+        elem = ds['UVElementMaximum']\n+        assert 'UV' == elem.VR\n+        assert 0xFFFE0004 == elem.tag\n+        assert 2**64 - 1 == elem.value\n+\n+        new = DataElement(0xFFFE0004, 'UV', 2**64 - 1)\n+        assert elem == new\n+\n \n class TestDeferredRead(object):\n     \"\"\"Test that deferred data element reading (for large size)\n", "problem_statement": "Add support for missing VRs\nMissing: OV, SV, UV\r\n\r\n\n", "hints_text": "", "created_at": "2020-01-01T04:52:05Z", "version": "1.3", "FAIL_TO_PASS": ["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"], "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_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::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": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pyvista/pyvista", "instance_id": "pyvista__pyvista-4417", "base_commit": "a8921b94b91a7d9809c9b5ac2ef9c981b5f71ea1", "patch": "diff --git a/pyvista/core/composite.py b/pyvista/core/composite.py\n--- a/pyvista/core/composite.py\n+++ b/pyvista/core/composite.py\n@@ -1110,6 +1110,8 @@ def as_polydata_blocks(self, copy=False):\n             if block is not None:\n                 if isinstance(block, MultiBlock):\n                     dataset.replace(i, block.as_polydata_blocks(copy=copy))\n+                elif isinstance(block, pyvista.PointSet):\n+                    dataset.replace(i, block.cast_to_polydata(deep=True))\n                 elif not isinstance(block, pyvista.PolyData):\n                     dataset.replace(i, block.extract_surface())\n                 elif copy:\n", "test_patch": "diff --git a/tests/test_composite.py b/tests/test_composite.py\n--- a/tests/test_composite.py\n+++ b/tests/test_composite.py\n@@ -753,9 +753,13 @@ def test_set_active_scalars_mixed(multiblock_poly):\n \n \n def test_to_polydata(multiblock_all):\n+    if pyvista.vtk_version_info >= (9, 1, 0):\n+        multiblock_all.append(pyvista.PointSet([0.0, 0.0, 1.0]))  # missing pointset\n     assert not multiblock_all.is_all_polydata\n \n     dataset_a = multiblock_all.as_polydata_blocks()\n+    if pyvista.vtk_version_info >= (9, 1, 0):\n+        assert dataset_a[-1].n_points == 1\n     assert not multiblock_all.is_all_polydata\n     assert dataset_a.is_all_polydata\n \n", "problem_statement": "``Multiblock``.plot does not work when using ``PointSet``\n### Describe the bug, what's wrong, and what you expected.\n\nIt seems ``MultiBlock`` entities made of ``PointSet`` plot nothing when using ``plot`` method.\n\n### Steps to reproduce the bug.\n\n```python\r\nimport pyvista as pv\r\nimport numpy as np\r\n\r\npoints_arr = np.array(\r\n    [\r\n        [0.0, 1.0, 0.0],\r\n        [0.0, 0.0, 0.0],\r\n        [1.0, 1.0, 0.0],\r\n        [1.0, 0.0, 0.0],\r\n        [0.0, 0.0, 1.0],\r\n        [1.0, 0.0, 1.0],\r\n        [1.0, 1.0, 1.0],\r\n        [0.0, 1.0, 1.0],\r\n    ]\r\n)\r\n\r\npoints = pv.MultiBlock()\r\nfor each_kp in points_arr:\r\n    points.append(pv.PointSet(each_kp))\r\n\r\npoints.plot()\r\n```\n\n### System Information\n\n```shell\n--------------------------------------------------------------------------------\r\n  Date: Wed May 10 18:07:18 2023 CEST\r\n\r\n                OS : Darwin\r\n            CPU(s) : 8\r\n           Machine : arm64\r\n      Architecture : 64bit\r\n               RAM : 16.0 GiB\r\n       Environment : IPython\r\n       File system : apfs\r\n        GPU Vendor : Apple\r\n      GPU Renderer : Apple M2\r\n       GPU Version : 4.1 Metal - 83.1\r\n  MathText Support : False\r\n\r\n  Python 3.11.1 (main, Dec 23 2022, 09:28:24) [Clang 14.0.0\r\n  (clang-1400.0.29.202)]\r\n\r\n           pyvista : 0.39.0\r\n               vtk : 9.2.6\r\n             numpy : 1.24.3\r\n        matplotlib : 3.7.1\r\n            scooby : 0.7.1\r\n             pooch : v1.7.0\r\n           imageio : 2.28.0\r\n           IPython : 8.12.1\r\n        ipywidgets : 8.0.6\r\n             scipy : 1.10.1\r\n              tqdm : 4.65.0\r\n        jupyterlab : 3.6.3\r\n         pythreejs : 2.4.2\r\n      nest_asyncio : 1.5.6\r\n--------------------------------------------------------------------------------\n```\n\n\n### Screenshots\n\n<img width=\"624\" alt=\"image\" src=\"https://github.com/pyvista/pyvista/assets/28149841/a1b0999f-2d35-4911-a216-eb6503955860\">\r\n\n", "hints_text": "I can reproduce this problem. We'll add it for the v0.39.1 milestone.\nThe trick is this bit of code:\r\n\r\nhttps://github.com/pyvista/pyvista/blob/a8921b94b91a7d9809c9b5ac2ef9c981b5f71ea1/pyvista/plotting/plotting.py#L3218-L3224\r\n\r\nwhich isn't used for `MultiBlock` plotting because `add_mesh` forwards to `add_composite()`\r\n\r\nhttps://github.com/pyvista/pyvista/blob/a8921b94b91a7d9809c9b5ac2ef9c981b5f71ea1/pyvista/plotting/plotting.py#L3230\r\n\r\nBut then I realized this block should handle it\r\n\r\nhttps://github.com/pyvista/pyvista/blob/a8921b94b91a7d9809c9b5ac2ef9c981b5f71ea1/pyvista/plotting/plotting.py#L2544\r\n\r\n\r\nso maybe there's a bug in that method or the copy isn't propagating? Not sure...\r\n\r\n", "created_at": "2023-05-12T16:10:15Z", "version": "0.40", "FAIL_TO_PASS": ["tests/test_composite.py::test_to_polydata"], "PASS_TO_PASS": ["tests/test_composite.py::test_multi_block_init_vtk", "tests/test_composite.py::test_multi_block_init_dict", "tests/test_composite.py::test_multi_block_keys", "tests/test_composite.py::test_multi_block_init_list", "tests/test_composite.py::test_multi_block_append", "tests/test_composite.py::test_multi_block_set_get_ers", "tests/test_composite.py::test_replace", "tests/test_composite.py::test_pop", "tests/test_composite.py::test_del_slice", "tests/test_composite.py::test_slicing_multiple_in_setitem", "tests/test_composite.py::test_reverse", "tests/test_composite.py::test_insert", "tests/test_composite.py::test_extend", "tests/test_composite.py::test_multi_block_clean", "tests/test_composite.py::test_multi_block_repr", "tests/test_composite.py::test_multi_block_eq", "tests/test_composite.py::test_multi_block_io[True-.vtm-True]", "tests/test_composite.py::test_multi_block_io[True-.vtm-False]", "tests/test_composite.py::test_multi_block_io[True-.vtmb-True]", "tests/test_composite.py::test_multi_block_io[True-.vtmb-False]", "tests/test_composite.py::test_multi_block_io[False-.vtm-True]", "tests/test_composite.py::test_multi_block_io[False-.vtm-False]", "tests/test_composite.py::test_multi_block_io[False-.vtmb-True]", "tests/test_composite.py::test_multi_block_io[False-.vtmb-False]", "tests/test_composite.py::test_ensight_multi_block_io[vtm-True]", "tests/test_composite.py::test_ensight_multi_block_io[vtm-False]", "tests/test_composite.py::test_ensight_multi_block_io[vtmb-True]", "tests/test_composite.py::test_ensight_multi_block_io[vtmb-False]", "tests/test_composite.py::test_invalid_arg", "tests/test_composite.py::test_multi_io_erros", "tests/test_composite.py::test_extract_geometry", "tests/test_composite.py::test_combine_filter", "tests/test_composite.py::test_multi_block_copy", "tests/test_composite.py::test_multi_block_negative_index", "tests/test_composite.py::test_multi_slice_index", "tests/test_composite.py::test_slice_defaults", "tests/test_composite.py::test_slice_negatives", "tests/test_composite.py::test_multi_block_volume", "tests/test_composite.py::test_multi_block_length", "tests/test_composite.py::test_multi_block_save_lines", "tests/test_composite.py::test_multi_block_data_range", "tests/test_composite.py::test_multiblock_ref", "tests/test_composite.py::test_set_active_scalars", "tests/test_composite.py::test_set_active_scalars_multi", "tests/test_composite.py::test_set_active_scalars_components", "tests/test_composite.py::test_set_active_multi_multi", "tests/test_composite.py::test_set_active_scalars_mixed", "tests/test_composite.py::test_compute_normals", "tests/test_composite.py::test_activate_plotting_scalars"], "environment_setup_commit": "19938bf14e7705ed1e7d4c9aa6558d90df6de9d1"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-995", "base_commit": "29be72498a4f4131808a45843b15692234ae7652", "patch": "diff --git a/pydicom/dataset.py b/pydicom/dataset.py\n--- a/pydicom/dataset.py\n+++ b/pydicom/dataset.py\n@@ -34,8 +34,9 @@\n from pydicom.datadict import (tag_for_keyword, keyword_for_tag,\n                               repeater_has_keyword)\n from pydicom.dataelem import DataElement, DataElement_from_raw, RawDataElement\n-from pydicom.pixel_data_handlers.util import (convert_color_space,\n-                                              reshape_pixel_array)\n+from pydicom.pixel_data_handlers.util import (\n+    convert_color_space, reshape_pixel_array, get_image_pixel_ids\n+)\n from pydicom.tag import Tag, BaseTag, tag_in_exception\n from pydicom.uid import (ExplicitVRLittleEndian, ImplicitVRLittleEndian,\n                          ExplicitVRBigEndian, PYDICOM_IMPLEMENTATION_UID)\n@@ -1302,20 +1303,11 @@ def convert_pixel_data(self, handler_name=''):\n         decompressed and any related data elements are changed accordingly.\n         \"\"\"\n         # Check if already have converted to a NumPy array\n-        # Also check if self.PixelData has changed. If so, get new NumPy array\n-        keywords = ['PixelData', 'FloatPixelData', 'DoubleFloatPixelData']\n-        px_keyword = [kw for kw in keywords if kw in self]\n-        if len(px_keyword) != 1:\n-            raise AttributeError(\n-                \"Unable to convert the pixel data: one of Pixel Data, Float \"\n-                \"Pixel Data or Double Float Pixel Data must be present in \"\n-                \"the dataset\"\n-            )\n-\n+        # Also check if pixel data has changed. If so, get new NumPy array\n         already_have = True\n         if not hasattr(self, \"_pixel_array\"):\n             already_have = False\n-        elif self._pixel_id != id(getattr(self, px_keyword[0])):\n+        elif self._pixel_id != get_image_pixel_ids(self):\n             already_have = False\n \n         if already_have:\n@@ -1451,9 +1443,7 @@ def _do_pixel_data_conversion(self, handler):\n                 self._pixel_array, 'YBR_FULL', 'RGB'\n             )\n \n-        keywords = ['PixelData', 'FloatPixelData', 'DoubleFloatPixelData']\n-        px_keyword = [kw for kw in keywords if kw in self]\n-        self._pixel_id = id(getattr(self, px_keyword[0]))\n+        self._pixel_id = get_image_pixel_ids(self)\n \n     def decompress(self, handler_name=''):\n         \"\"\"Decompresses *Pixel Data* and modifies the :class:`Dataset`\ndiff --git a/pydicom/pixel_data_handlers/util.py b/pydicom/pixel_data_handlers/util.py\n--- a/pydicom/pixel_data_handlers/util.py\n+++ b/pydicom/pixel_data_handlers/util.py\n@@ -750,6 +750,59 @@ def get_expected_length(ds, unit='bytes'):\n     return length\n \n \n+def get_image_pixel_ids(ds):\n+    \"\"\"Return a dict of the pixel data affecting element's :func:`id` values.\n+\n+    +------------------------------------------------+\n+    | Element                                        |\n+    +-------------+---------------------------+------+\n+    | Tag         | Keyword                   | Type |\n+    +=============+===========================+======+\n+    | (0028,0002) | SamplesPerPixel           | 1    |\n+    +-------------+---------------------------+------+\n+    | (0028,0004) | PhotometricInterpretation | 1    |\n+    +-------------+---------------------------+------+\n+    | (0028,0006) | PlanarConfiguration       | 1C   |\n+    +-------------+---------------------------+------+\n+    | (0028,0008) | NumberOfFrames            | 1C   |\n+    +-------------+---------------------------+------+\n+    | (0028,0010) | Rows                      | 1    |\n+    +-------------+---------------------------+------+\n+    | (0028,0011) | Columns                   | 1    |\n+    +-------------+---------------------------+------+\n+    | (0028,0100) | BitsAllocated             | 1    |\n+    +-------------+---------------------------+------+\n+    | (0028,0103) | PixelRepresentation       | 1    |\n+    +-------------+---------------------------+------+\n+    | (7FE0,0008) | FloatPixelData            | 1C   |\n+    +-------------+---------------------------+------+\n+    | (7FE0,0009) | DoubleFloatPixelData      | 1C   |\n+    +-------------+---------------------------+------+\n+    | (7FE0,0010) | PixelData                 | 1C   |\n+    +-------------+---------------------------+------+\n+\n+    Parameters\n+    ----------\n+    ds : Dataset\n+        The :class:`~pydicom.dataset.Dataset` containing the pixel data.\n+\n+    Returns\n+    -------\n+    dict\n+        A dict containing the :func:`id` values for the elements that affect\n+        the pixel data.\n+\n+    \"\"\"\n+    keywords = [\n+        'SamplesPerPixel', 'PhotometricInterpretation', 'PlanarConfiguration',\n+        'NumberOfFrames', 'Rows', 'Columns', 'BitsAllocated',\n+        'PixelRepresentation', 'FloatPixelData', 'DoubleFloatPixelData',\n+        'PixelData'\n+    ]\n+\n+    return {kw: id(getattr(ds, kw, None)) for kw in keywords}\n+\n+\n def pixel_dtype(ds, as_float=False):\n     \"\"\"Return a :class:`numpy.dtype` for the pixel data in `ds`.\n \n", "test_patch": "diff --git a/pydicom/tests/test_dataset.py b/pydicom/tests/test_dataset.py\n--- a/pydicom/tests/test_dataset.py\n+++ b/pydicom/tests/test_dataset.py\n@@ -11,6 +11,7 @@\n from pydicom import dcmread\n from pydicom.filebase import DicomBytesIO\n from pydicom.overlay_data_handlers import numpy_handler as NP_HANDLER\n+from pydicom.pixel_data_handlers.util import get_image_pixel_ids\n from pydicom.sequence import Sequence\n from pydicom.tag import Tag\n from pydicom.uid import (\n@@ -1142,10 +1143,10 @@ def test_pixel_array_already_have(self):\n         # Test that _pixel_array is returned unchanged unless required\n         fpath = get_testdata_files(\"CT_small.dcm\")[0]\n         ds = dcmread(fpath)\n-        ds._pixel_id = id(ds.PixelData)\n+        ds._pixel_id = get_image_pixel_ids(ds)\n         ds._pixel_array = 'Test Value'\n         ds.convert_pixel_data()\n-        assert id(ds.PixelData) == ds._pixel_id\n+        assert get_image_pixel_ids(ds) == ds._pixel_id\n         assert 'Test Value' == ds._pixel_array\n \n     def test_pixel_array_id_changed(self):\n@@ -1277,17 +1278,6 @@ def test_update_with_dataset(self):\n         ds2.update(ds)\n         assert 'TestC' == ds2.PatientName\n \n-    def test_convert_pixel_data_no_px(self):\n-        \"\"\"Test convert_pixel_data() with no pixel data elements.\"\"\"\n-        ds = Dataset()\n-        msg = (\n-            r\"Unable to convert the pixel data: one of Pixel Data, Float \"\n-            r\"Pixel Data or Double Float Pixel Data must be present in \"\n-            r\"the dataset\"\n-        )\n-        with pytest.raises(AttributeError, match=msg):\n-            ds.convert_pixel_data()\n-\n \n class TestDatasetElements(object):\n     \"\"\"Test valid assignments of data elements\"\"\"\ndiff --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@@ -146,6 +146,9 @@\n JPEG_2K = get_testdata_files(\"JPEG2000.dcm\")[0]\n # RLE Lossless\n RLE = get_testdata_files(\"MR_small_RLE.dcm\")[0]\n+# No Image Pixel module\n+NO_PIXEL = get_testdata_files(\"rtplan.dcm\")[0]\n+\n \n # Transfer Syntaxes (non-retired + Explicit VR Big Endian)\n SUPPORTED_SYNTAXES = [\n@@ -471,6 +474,17 @@ def needs_convert(ds):\n         # Reset\n         NP_HANDLER.needs_to_convert_to_RGB = orig_fn\n \n+    def test_dataset_pixel_array_no_pixels(self):\n+        \"\"\"Test good exception message if no pixel data in dataset.\"\"\"\n+        ds = dcmread(NO_PIXEL)\n+        msg = (\n+            r\"Unable to convert the pixel data: one of Pixel Data, Float \"\n+            r\"Pixel Data or Double Float Pixel Data must be present in the \"\n+            r\"dataset\"\n+        )\n+        with pytest.raises(AttributeError, match=msg):\n+            ds.pixel_array\n+\n     @pytest.mark.parametrize(\"fpath, data\", REFERENCE_DATA_UNSUPPORTED)\n     def test_can_access_unsupported_dataset(self, fpath, data):\n         \"\"\"Test can read and access elements in unsupported datasets.\"\"\"\n", "problem_statement": "Dataset.pixel_array doesn't change unless PixelData does\n#### Description\r\nCurrently `ds.pixel_array` produces a numpy array that depends on element values for Rows, Columns, Samples Per Pixel, etc, however the code for `ds.pixel_array` only changes the returned array if the value for `ds.PixelData` changes. This may lead to confusion/undesirable behaviour if the values for related elements are changed after `ds.pixel_array` is called but not the underlying pixel data.\r\n\r\nI can't think of any real use cases except maybe in an interactive session when debugging a non-conformant dataset, but I suggest we change the way `Dataset._pixel_id` is calculated so that it takes into account changes in related elements as well.\r\n\n", "hints_text": "", "created_at": "2019-12-17T23:21:50Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_dataset.py::TestDataset::test_attribute_error_in_property", "pydicom/tests/test_dataset.py::TestDataset::test_for_stray_raw_data_element", "pydicom/tests/test_dataset.py::TestDataset::test_attribute_error_in_property_correct_debug", "pydicom/tests/test_dataset.py::TestDataset::test_tag_exception_print", "pydicom/tests/test_dataset.py::TestDataset::test_tag_exception_walk", "pydicom/tests/test_dataset.py::TestDataset::test_set_new_data_element_by_name", "pydicom/tests/test_dataset.py::TestDataset::test_set_existing_data_element_by_name", "pydicom/tests/test_dataset.py::TestDataset::test_set_non_dicom", "pydicom/tests/test_dataset.py::TestDataset::test_membership", "pydicom/tests/test_dataset.py::TestDataset::test_contains", "pydicom/tests/test_dataset.py::TestDataset::test_clear", "pydicom/tests/test_dataset.py::TestDataset::test_pop", "pydicom/tests/test_dataset.py::TestDataset::test_pop_using_tuple", "pydicom/tests/test_dataset.py::TestDataset::test_pop_using_keyword", "pydicom/tests/test_dataset.py::TestDataset::test_popitem", "pydicom/tests/test_dataset.py::TestDataset::test_setdefault", "pydicom/tests/test_dataset.py::TestDataset::test_setdefault_tuple", "pydicom/tests/test_dataset.py::TestDataset::test_setdefault_use_value", "pydicom/tests/test_dataset.py::TestDataset::test_setdefault_keyword", "pydicom/tests/test_dataset.py::TestDataset::test_get_exists1", "pydicom/tests/test_dataset.py::TestDataset::test_get_exists2", "pydicom/tests/test_dataset.py::TestDataset::test_get_exists3", "pydicom/tests/test_dataset.py::TestDataset::test_get_exists4", "pydicom/tests/test_dataset.py::TestDataset::test_get_default1", "pydicom/tests/test_dataset.py::TestDataset::test_get_default2", "pydicom/tests/test_dataset.py::TestDataset::test_get_default3", "pydicom/tests/test_dataset.py::TestDataset::test_get_default4", "pydicom/tests/test_dataset.py::TestDataset::test_get_raises", "pydicom/tests/test_dataset.py::TestDataset::test_get_from_raw", "pydicom/tests/test_dataset.py::TestDataset::test__setitem__", "pydicom/tests/test_dataset.py::TestDataset::test_matching_tags", "pydicom/tests/test_dataset.py::TestDataset::test_named_member_updated", "pydicom/tests/test_dataset.py::TestDataset::test_update", "pydicom/tests/test_dataset.py::TestDataset::test_dir_subclass", "pydicom/tests/test_dataset.py::TestDataset::test_dir", "pydicom/tests/test_dataset.py::TestDataset::test_dir_filter", "pydicom/tests/test_dataset.py::TestDataset::test_delete_dicom_attr", "pydicom/tests/test_dataset.py::TestDataset::test_delete_dicom_command_group_length", "pydicom/tests/test_dataset.py::TestDataset::test_delete_other_attr", "pydicom/tests/test_dataset.py::TestDataset::test_delete_dicom_attr_we_dont_have", "pydicom/tests/test_dataset.py::TestDataset::test_delete_item_long", "pydicom/tests/test_dataset.py::TestDataset::test_delete_item_tuple", "pydicom/tests/test_dataset.py::TestDataset::test_delete_non_existing_item", "pydicom/tests/test_dataset.py::TestDataset::test_equality_no_sequence", "pydicom/tests/test_dataset.py::TestDataset::test_equality_private", "pydicom/tests/test_dataset.py::TestDataset::test_equality_sequence", "pydicom/tests/test_dataset.py::TestDataset::test_equality_not_dataset", "pydicom/tests/test_dataset.py::TestDataset::test_equality_unknown", "pydicom/tests/test_dataset.py::TestDataset::test_equality_inheritance", "pydicom/tests/test_dataset.py::TestDataset::test_equality_elements", "pydicom/tests/test_dataset.py::TestDataset::test_inequality", "pydicom/tests/test_dataset.py::TestDataset::test_hash", "pydicom/tests/test_dataset.py::TestDataset::test_property", "pydicom/tests/test_dataset.py::TestDataset::test_add_repeater_elem_by_keyword", "pydicom/tests/test_dataset.py::TestDataset::test_setitem_slice_raises", "pydicom/tests/test_dataset.py::TestDataset::test_getitem_slice_raises", "pydicom/tests/test_dataset.py::TestDataset::test_empty_slice", "pydicom/tests/test_dataset.py::TestDataset::test_getitem_slice", "pydicom/tests/test_dataset.py::TestDataset::test_getitem_slice_ffff", "pydicom/tests/test_dataset.py::TestDataset::test_delitem_slice", "pydicom/tests/test_dataset.py::TestDataset::test_group_dataset", "pydicom/tests/test_dataset.py::TestDataset::test_get_item", "pydicom/tests/test_dataset.py::TestDataset::test_get_item_slice", "pydicom/tests/test_dataset.py::TestDataset::test_get_private_item", "pydicom/tests/test_dataset.py::TestDataset::test_private_block", "pydicom/tests/test_dataset.py::TestDataset::test_add_new_private_tag", "pydicom/tests/test_dataset.py::TestDataset::test_delete_private_tag", "pydicom/tests/test_dataset.py::TestDataset::test_private_creators", "pydicom/tests/test_dataset.py::TestDataset::test_is_original_encoding", "pydicom/tests/test_dataset.py::TestDataset::test_remove_private_tags", "pydicom/tests/test_dataset.py::TestDataset::test_data_element", "pydicom/tests/test_dataset.py::TestDataset::test_iterall", "pydicom/tests/test_dataset.py::TestDataset::test_save_as", "pydicom/tests/test_dataset.py::TestDataset::test_with", "pydicom/tests/test_dataset.py::TestDataset::test_exit_exception", "pydicom/tests/test_dataset.py::TestDataset::test_pixel_array_already_have", "pydicom/tests/test_dataset.py::TestDataset::test_pixel_array_id_changed", "pydicom/tests/test_dataset.py::TestDataset::test_pixel_array_unknown_syntax", "pydicom/tests/test_dataset.py::TestDataset::test_formatted_lines", "pydicom/tests/test_dataset.py::TestDataset::test_formatted_lines_known_uid", "pydicom/tests/test_dataset.py::TestDataset::test_set_convert_private_elem_from_raw", "pydicom/tests/test_dataset.py::TestDataset::test_top", "pydicom/tests/test_dataset.py::TestDataset::test_trait_names", "pydicom/tests/test_dataset.py::TestDataset::test_walk", "pydicom/tests/test_dataset.py::TestDataset::test_update_with_dataset", "pydicom/tests/test_dataset.py::TestDatasetElements::test_sequence_assignment", "pydicom/tests/test_dataset.py::TestDatasetElements::test_ensure_file_meta", "pydicom/tests/test_dataset.py::TestDatasetElements::test_fix_meta_info", "pydicom/tests/test_dataset.py::TestDatasetElements::test_validate_and_correct_file_meta", "pydicom/tests/test_dataset.py::TestFileDataset::test_pickle", "pydicom/tests/test_dataset.py::TestFileDataset::test_pickle_modified", "pydicom/tests/test_dataset.py::TestFileDataset::test_equality_file_meta", "pydicom/tests/test_dataset.py::TestFileDataset::test_creation_with_container", "pydicom/tests/test_dataset.py::TestFileDataset::test_works_as_expected_within_numpy_array", "pydicom/tests/test_dataset.py::TestFileDataset::test_dataset_overrides_all_dict_attributes", "pydicom/tests/test_dataset.py::TestDatasetOverlayArray::test_no_possible", "pydicom/tests/test_dataset.py::TestDatasetOverlayArray::test_possible_not_available", "pydicom/tests/test_dataset.py::TestDatasetOverlayArray::test_possible_available", "pydicom/tests/test_dataset.py::TestDatasetOverlayArray::test_handler_raises", "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__1.3/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__1.3/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__1.3/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__1.3/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[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/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__1.3/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__1.3/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__1.3/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__1.3/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__1.3/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__1.3/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[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/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__1.3/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__1.3/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__1.3/pydicom/data/test_files/liver_1frame.dcm-data0]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/liver.dcm-data1]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A.dcm-data2]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/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__1.3/pydicom/data/test_files/SC_ybr_full_422_uncompressed.dcm-data4]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A_2frame.dcm-data5]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb.dcm-data6]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_2frame.dcm-data7]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/MR_small.dcm-data8]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/emri_small.dcm-data9]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_16bit.dcm-data10]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_16bit_2frame.dcm-data11]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/rtdose_1frame.dcm-data12]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/rtdose.dcm-data13]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_32bit.dcm-data14]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_properties[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/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__1.3/pydicom/data/test_files/liver_1frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/liver_expb_1frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/liver.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/liver_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A_2frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/OBXXXX1A_expb_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_2frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_expb_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/MR_small.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/MR_small_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/emri_small.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/emri_small_big_endian.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_16bit.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_expb_16bit.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_16bit_2frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_expb_16bit_2frame.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/rtdose_1frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/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__1.3/pydicom/data/test_files/rtdose.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/rtdose_expb.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_32bit.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_expb_32bit.dcm]", "pydicom/tests/test_numpy_pixel_data.py::TestNumpy_NumpyHandler::test_big_endian_datasets[/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/SC_rgb_32bit_2frame.dcm-/project/temp/tmpqcg17m39/pydicom__pydicom__1.3/pydicom/data/test_files/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", "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_non_array_input", "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"], "PASS_TO_PASS": [], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pylint-dev/astroid", "instance_id": "pylint-dev__astroid-1978", "base_commit": "0c9ab0fe56703fa83c73e514a1020d398d23fa7f", "patch": "diff --git a/astroid/raw_building.py b/astroid/raw_building.py\n--- a/astroid/raw_building.py\n+++ b/astroid/raw_building.py\n@@ -10,11 +10,14 @@\n \n import builtins\n import inspect\n+import io\n+import logging\n import os\n import sys\n import types\n import warnings\n from collections.abc import Iterable\n+from contextlib import redirect_stderr, redirect_stdout\n from typing import Any, Union\n \n from astroid import bases, nodes\n@@ -22,6 +25,9 @@\n from astroid.manager import AstroidManager\n from astroid.nodes import node_classes\n \n+logger = logging.getLogger(__name__)\n+\n+\n _FunctionTypes = Union[\n     types.FunctionType,\n     types.MethodType,\n@@ -471,7 +477,26 @@ def imported_member(self, node, member, name: str) -> bool:\n             # check if it sounds valid and then add an import node, else use a\n             # dummy node\n             try:\n-                getattr(sys.modules[modname], name)\n+                with redirect_stderr(io.StringIO()) as stderr, redirect_stdout(\n+                    io.StringIO()\n+                ) as stdout:\n+                    getattr(sys.modules[modname], name)\n+                    stderr_value = stderr.getvalue()\n+                    if stderr_value:\n+                        logger.error(\n+                            \"Captured stderr while getting %s from %s:\\n%s\",\n+                            name,\n+                            sys.modules[modname],\n+                            stderr_value,\n+                        )\n+                    stdout_value = stdout.getvalue()\n+                    if stdout_value:\n+                        logger.info(\n+                            \"Captured stdout while getting %s from %s:\\n%s\",\n+                            name,\n+                            sys.modules[modname],\n+                            stdout_value,\n+                        )\n             except (KeyError, AttributeError):\n                 attach_dummy_node(node, name, member)\n             else:\n", "test_patch": "diff --git a/tests/unittest_raw_building.py b/tests/unittest_raw_building.py\n--- a/tests/unittest_raw_building.py\n+++ b/tests/unittest_raw_building.py\n@@ -8,8 +8,15 @@\n # For details: https://github.com/PyCQA/astroid/blob/main/LICENSE\n # Copyright (c) https://github.com/PyCQA/astroid/blob/main/CONTRIBUTORS.txt\n \n+from __future__ import annotations\n+\n+import logging\n+import os\n+import sys\n import types\n import unittest\n+from typing import Any\n+from unittest import mock\n \n import _io\n import pytest\n@@ -117,5 +124,45 @@ def test_module_object_with_broken_getattr(self) -> None:\n         AstroidBuilder().inspect_build(fm_getattr, \"test\")\n \n \n+@pytest.mark.skipif(\n+    \"posix\" not in sys.builtin_module_names, reason=\"Platform doesn't support posix\"\n+)\n+def test_build_module_getattr_catch_output(\n+    capsys: pytest.CaptureFixture[str],\n+    caplog: pytest.LogCaptureFixture,\n+) -> None:\n+    \"\"\"Catch stdout and stderr in module __getattr__ calls when building a module.\n+\n+    Usually raised by DeprecationWarning or FutureWarning.\n+    \"\"\"\n+    caplog.set_level(logging.INFO)\n+    original_sys = sys.modules\n+    original_module = sys.modules[\"posix\"]\n+    expected_out = \"INFO (TEST): Welcome to posix!\"\n+    expected_err = \"WARNING (TEST): Monkey-patched version of posix - module getattr\"\n+\n+    class CustomGetattr:\n+        def __getattr__(self, name: str) -> Any:\n+            print(f\"{expected_out}\")\n+            print(expected_err, file=sys.stderr)\n+            return getattr(original_module, name)\n+\n+    def mocked_sys_modules_getitem(name: str) -> types.ModuleType | CustomGetattr:\n+        if name != \"posix\":\n+            return original_sys[name]\n+        return CustomGetattr()\n+\n+    with mock.patch(\"astroid.raw_building.sys.modules\") as sys_mock:\n+        sys_mock.__getitem__.side_effect = mocked_sys_modules_getitem\n+        builder = AstroidBuilder()\n+        builder.inspect_build(os)\n+\n+    out, err = capsys.readouterr()\n+    assert expected_out in caplog.text\n+    assert expected_err in caplog.text\n+    assert not out\n+    assert not err\n+\n+\n if __name__ == \"__main__\":\n     unittest.main()\n", "problem_statement": "Deprecation warnings from numpy\n### Steps to reproduce\r\n\r\n1. Run pylint over the following test case:\r\n\r\n```\r\n\"\"\"Test case\"\"\"\r\n\r\nimport numpy as np\r\nvalue = np.random.seed(1234)\r\n```\r\n\r\n### Current behavior\r\n```\r\n/home/bje/source/nemo/myenv/lib/python3.10/site-packages/astroid/raw_building.py:470: FutureWarning: In the future `np.long` will be defined as the corresponding NumPy scalar.  (This may have returned Python scalars in past versions.\r\n  getattr(sys.modules[modname], name)\r\n/home/bje/source/nemo/myenv/lib/python3.10/site-packages/astroid/raw_building.py:470: FutureWarning: In the future `np.long` will be defined as the corresponding NumPy scalar.  (This may have returned Python scalars in past versions.\r\n  getattr(sys.modules[modname], name)\r\n```\r\n\r\n### Expected behavior\r\nThere should be no future warnings.\r\n\r\n### python -c \"from astroid import __pkginfo__; print(__pkginfo__.version)\" output\r\n2.12.13\n", "hints_text": "This seems very similar to https://github.com/PyCQA/astroid/pull/1514 that was fixed in 2.12.0.\nI'm running 2.12.13 (> 2.12.0), so the fix isn't working in this case?\nI don't know why #1514 did not fix this, I think we were capturing both stdout and stderr, so this will need some investigation. My guess would be that there's somewhere else to apply the same method to.\nHello, \r\nI see the same error with pylint on our tool [demcompare](https://github.com/CNES/demcompare). Pylint version:\r\n```\r\npylint --version\r\npylint 2.15.9\r\nastroid 2.12.13\r\nPython 3.8.10 (default, Nov 14 2022, 12:59:47) \r\n[GCC 9.4.0]\r\n```\r\nI confirm the weird astroid lower warning and I don't know how to bypass it with pylint checking. \r\n\r\n```\r\npylint demcompare \r\n/home/duboise/work/src/demcompare/venv/lib/python3.8/site-packages/astroid/raw_building.py:470: FutureWarning: In the future `np.long` will be defined as the corresponding NumPy scalar.  (This may have returned Python scalars in past versions.\r\n  getattr(sys.modules[modname], name)\r\n... (four times)\r\n```\r\n\r\nThanks in advance if there is a solution\r\nCordially\r\n\n> Thanks in advance if there is a solution\r\n\r\nwhile annoying the warning does not make pylint fail. Just ignore it. In a CI you can just check pylint return code. It will return 0 as expected\nI agree, even if annoying because it feels our code as a problem somewhere, the CI with pylint doesn't fail indeed. Thanks for the answer that confirm to not bother for now. \nThat might be fine in a CI environment, but for users, ultimately, ignoring warnings becomes difficult when there are too many such warnings. I would like to see this fixed.\nOh, it was not an argument in favour of not fixing it. It was just to point out that it is not a breaking problem. It is \"just\" a lot of quite annoying warnings. I am following the issue because it annoys me too. So I am in the same \"I hope they will fix it\" boat\n> I don't know why https://github.com/PyCQA/astroid/pull/1514 did not fix this, I think we were capturing both stdout and stderr, so this will need some investigation. My guess would be that there's somewhere else to apply the same method to.\r\n\r\nThat PR only addressed import-time. This `FutureWarning` is emitted by numpy's package-level `__getattr__` method, not during import.", "created_at": "2023-01-28T06:14:39Z", "version": "2.14", "FAIL_TO_PASS": ["tests/unittest_raw_building.py::test_build_module_getattr_catch_output"], "PASS_TO_PASS": ["tests/unittest_raw_building.py::RawBuildingTC::test_attach_dummy_node", "tests/unittest_raw_building.py::RawBuildingTC::test_build_class", "tests/unittest_raw_building.py::RawBuildingTC::test_build_from_import", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_args", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_deepinspect_deprecation", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_defaults", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_kwonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_function_posonlyargs", "tests/unittest_raw_building.py::RawBuildingTC::test_build_module", "tests/unittest_raw_building.py::RawBuildingTC::test_io_is__io", "tests/unittest_raw_building.py::RawBuildingTC::test_module_object_with_broken_getattr"], "environment_setup_commit": "0c9ab0fe56703fa83c73e514a1020d398d23fa7f"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-4764", "base_commit": "a820c139ccbe6d1865d73c4a459945cd69899f8f", "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@@ -44,6 +44,7 @@\n     dialect_selector,\n     dialect_readout,\n )\n+from sqlfluff.core.linter import LintingResult\n from sqlfluff.core.config import progress_bar_configuration\n \n from sqlfluff.core.enums import FormatType, Color\n@@ -691,12 +692,16 @@ def lint(\n         sys.exit(EXIT_SUCCESS)\n \n \n-def do_fixes(lnt, result, formatter=None, **kwargs):\n+def do_fixes(\n+    result: LintingResult, formatter: Optional[OutputStreamFormatter] = None, **kwargs\n+):\n     \"\"\"Actually do the fixes.\"\"\"\n-    click.echo(\"Persisting Changes...\")\n+    if formatter and formatter.verbosity >= 0:\n+        click.echo(\"Persisting Changes...\")\n     res = result.persist_changes(formatter=formatter, **kwargs)\n     if all(res.values()):\n-        click.echo(\"Done. Please check your files to confirm.\")\n+        if formatter and formatter.verbosity >= 0:\n+            click.echo(\"Done. Please check your files to confirm.\")\n         return True\n     # If some failed then return false\n     click.echo(\n@@ -708,7 +713,7 @@ def do_fixes(lnt, result, formatter=None, **kwargs):\n     return False  # pragma: no cover\n \n \n-def _stdin_fix(linter, formatter, fix_even_unparsable):\n+def _stdin_fix(linter: Linter, formatter, fix_even_unparsable):\n     \"\"\"Handle fixing from stdin.\"\"\"\n     exit_code = EXIT_SUCCESS\n     stdin = sys.stdin.read()\n@@ -751,7 +756,7 @@ def _stdin_fix(linter, formatter, fix_even_unparsable):\n \n \n def _paths_fix(\n-    linter,\n+    linter: Linter,\n     formatter,\n     paths,\n     processes,\n@@ -765,11 +770,12 @@ def _paths_fix(\n ):\n     \"\"\"Handle fixing from paths.\"\"\"\n     # Lint the paths (not with the fix argument at this stage), outputting as we go.\n-    click.echo(\"==== finding fixable violations ====\")\n+    if formatter.verbosity >= 0:\n+        click.echo(\"==== finding fixable violations ====\")\n     exit_code = EXIT_SUCCESS\n \n     with PathAndUserErrorHandler(formatter):\n-        result = linter.lint_paths(\n+        result: LintingResult = linter.lint_paths(\n             paths,\n             fix=True,\n             ignore_non_existent_files=False,\n@@ -781,20 +787,18 @@ def _paths_fix(\n \n     # NB: We filter to linting violations here, because they're\n     # the only ones which can be potentially fixed.\n-    if result.num_violations(types=SQLLintError, fixable=True) > 0:\n-        click.echo(\"==== fixing violations ====\")\n-        click.echo(\n-            f\"{result.num_violations(types=SQLLintError, fixable=True)} fixable \"\n-            \"linting violations found\"\n-        )\n+    num_fixable = result.num_violations(types=SQLLintError, fixable=True)\n+    if num_fixable > 0:\n+        if formatter.verbosity >= 0:\n+            click.echo(\"==== fixing violations ====\")\n+        click.echo(f\"{num_fixable} \" \"fixable linting violations found\")\n         if force:\n-            if warn_force:\n+            if warn_force and formatter.verbosity >= 0:\n                 click.echo(\n                     f\"{formatter.colorize('FORCE MODE', Color.red)}: \"\n                     \"Attempting fixes...\"\n                 )\n             success = do_fixes(\n-                linter,\n                 result,\n                 formatter,\n                 types=SQLLintError,\n@@ -809,9 +813,9 @@ def _paths_fix(\n             c = click.getchar().lower()\n             click.echo(\"...\")\n             if c in (\"y\", \"\\r\", \"\\n\"):\n-                click.echo(\"Attempting fixes...\")\n+                if formatter.verbosity >= 0:\n+                    click.echo(\"Attempting fixes...\")\n                 success = do_fixes(\n-                    linter,\n                     result,\n                     formatter,\n                     types=SQLLintError,\n@@ -829,8 +833,9 @@ def _paths_fix(\n                 click.echo(\"Aborting...\")\n                 exit_code = EXIT_FAIL\n     else:\n-        click.echo(\"==== no fixable linting violations found ====\")\n-        formatter.completion_message()\n+        if formatter.verbosity >= 0:\n+            click.echo(\"==== no fixable linting violations found ====\")\n+            formatter.completion_message()\n \n     error_types = [\n         (\n@@ -841,7 +846,7 @@ def _paths_fix(\n     ]\n     for num_violations_kwargs, message_format, error_level in error_types:\n         num_violations = result.num_violations(**num_violations_kwargs)\n-        if num_violations > 0:\n+        if num_violations > 0 and formatter.verbosity >= 0:\n             click.echo(message_format.format(num_violations))\n             exit_code = max(exit_code, error_level)\n \n@@ -880,10 +885,20 @@ def _paths_fix(\n     \"--force\",\n     is_flag=True,\n     help=(\n-        \"skip the confirmation prompt and go straight to applying \"\n+        \"Skip the confirmation prompt and go straight to applying \"\n         \"fixes. **Use this with caution.**\"\n     ),\n )\n+@click.option(\n+    \"-q\",\n+    \"--quiet\",\n+    is_flag=True,\n+    help=(\n+        \"Reduces the amount of output to stdout to a minimal level. \"\n+        \"This is effectively the opposite of -v. NOTE: It will only \"\n+        \"take effect if -f/--force is also set.\"\n+    ),\n+)\n @click.option(\n     \"-x\",\n     \"--fixed-suffix\",\n@@ -913,6 +928,7 @@ def fix(\n     force: bool,\n     paths: Tuple[str],\n     bench: bool = False,\n+    quiet: bool = False,\n     fixed_suffix: str = \"\",\n     logger: Optional[logging.Logger] = None,\n     processes: Optional[int] = None,\n@@ -932,6 +948,13 @@ def fix(\n     \"\"\"\n     # some quick checks\n     fixing_stdin = (\"-\",) == paths\n+    if quiet:\n+        if kwargs[\"verbose\"]:\n+            click.echo(\n+                \"ERROR: The --quiet flag can only be used if --verbose is not set.\",\n+            )\n+            sys.exit(EXIT_ERROR)\n+        kwargs[\"verbose\"] = -1\n \n     config = get_config(\n         extra_config_path, ignore_local_config, require_dialect=False, **kwargs\ndiff --git a/src/sqlfluff/cli/formatters.py b/src/sqlfluff/cli/formatters.py\n--- a/src/sqlfluff/cli/formatters.py\n+++ b/src/sqlfluff/cli/formatters.py\n@@ -94,7 +94,7 @@ def __init__(\n     ):\n         self._output_stream = output_stream\n         self.plain_output = self.should_produce_plain_output(nocolor)\n-        self._verbosity = verbosity\n+        self.verbosity = verbosity\n         self._filter_empty = filter_empty\n         self.output_line_length = output_line_length\n \n@@ -116,13 +116,13 @@ def _format_config(self, linter: Linter) -> str:\n         \"\"\"Format the config of a `Linter`.\"\"\"\n         text_buffer = StringIO()\n         # Only show version information if verbosity is high enough\n-        if self._verbosity > 0:\n+        if self.verbosity > 0:\n             text_buffer.write(\"==== sqlfluff ====\\n\")\n             config_content = [\n                 (\"sqlfluff\", get_package_version()),\n                 (\"python\", get_python_version()),\n                 (\"implementation\", get_python_implementation()),\n-                (\"verbosity\", self._verbosity),\n+                (\"verbosity\", self.verbosity),\n             ]\n             if linter.dialect:\n                 config_content.append((\"dialect\", linter.dialect.name))\n@@ -138,7 +138,7 @@ def _format_config(self, linter: Linter) -> str:\n                         col_width=41,\n                     )\n                 )\n-            if self._verbosity > 1:\n+            if self.verbosity > 1:\n                 text_buffer.write(\"\\n== Raw Config:\\n\")\n                 text_buffer.write(self.format_config_vals(linter.config.iter_vals()))\n         return text_buffer.getvalue()\n@@ -150,7 +150,7 @@ def dispatch_config(self, linter: Linter) -> None:\n     def dispatch_persist_filename(self, filename, result):\n         \"\"\"Dispatch filenames during a persist operation.\"\"\"\n         # Only show the skip records at higher levels of verbosity\n-        if self._verbosity >= 2 or result != \"SKIP\":\n+        if self.verbosity >= 2 or result != \"SKIP\":\n             self._dispatch(self.format_filename(filename=filename, success=result))\n \n     def _format_path(self, path: str) -> str:\n@@ -159,14 +159,14 @@ def _format_path(self, path: str) -> str:\n \n     def dispatch_path(self, path: str) -> None:\n         \"\"\"Dispatch paths for display.\"\"\"\n-        if self._verbosity > 0:\n+        if self.verbosity > 0:\n             self._dispatch(self._format_path(path))\n \n     def dispatch_template_header(\n         self, fname: str, linter_config: FluffConfig, file_config: FluffConfig\n     ) -> None:\n         \"\"\"Dispatch the header displayed before templating.\"\"\"\n-        if self._verbosity > 1:\n+        if self.verbosity > 1:\n             self._dispatch(self.format_filename(filename=fname, success=\"TEMPLATING\"))\n             # This is where we output config diffs if they exist.\n             if file_config:\n@@ -182,12 +182,12 @@ def dispatch_template_header(\n \n     def dispatch_parse_header(self, fname: str) -> None:\n         \"\"\"Dispatch the header displayed before parsing.\"\"\"\n-        if self._verbosity > 1:\n+        if self.verbosity > 1:\n             self._dispatch(self.format_filename(filename=fname, success=\"PARSING\"))\n \n     def dispatch_lint_header(self, fname: str, rules: List[str]) -> None:\n         \"\"\"Dispatch the header displayed before linting.\"\"\"\n-        if self._verbosity > 1:\n+        if self.verbosity > 1:\n             self._dispatch(\n                 self.format_filename(\n                     filename=fname, success=f\"LINTING ({', '.join(rules)})\"\n@@ -202,7 +202,7 @@ def dispatch_compilation_header(self, templater, message):\n \n     def dispatch_processing_header(self, processes: int) -> None:\n         \"\"\"Dispatch the header displayed before linting.\"\"\"\n-        if self._verbosity > 0:\n+        if self.verbosity > 0:\n             self._dispatch(  # pragma: no cover\n                 f\"{self.colorize('effective configured processes: ', Color.lightgrey)} \"\n                 f\"{processes}\"\n@@ -228,7 +228,7 @@ def _format_file_violations(\n         show = fails + warns > 0\n \n         # Only print the filename if it's either a failure or verbosity > 1\n-        if self._verbosity > 0 or show:\n+        if self.verbosity > 0 or show:\n             text_buffer.write(self.format_filename(fname, success=fails == 0))\n             text_buffer.write(\"\\n\")\n \n@@ -253,6 +253,8 @@ def dispatch_file_violations(\n         self, fname: str, linted_file: LintedFile, only_fixable: bool\n     ) -> None:\n         \"\"\"Dispatch any violations found in a file.\"\"\"\n+        if self.verbosity < 0:\n+            return\n         s = self._format_file_violations(\n             fname,\n             linted_file.get_violations(\n@@ -392,10 +394,13 @@ def format_filename(\n         if isinstance(success, str):\n             status_string = success\n         else:\n-            status_string = self.colorize(\n-                success_text if success else \"FAIL\",\n-                Color.green if success else Color.red,\n-            )\n+            status_string = success_text if success else \"FAIL\"\n+\n+        if status_string in (\"PASS\", \"FIXED\", success_text):\n+            status_string = self.colorize(status_string, Color.green)\n+        elif status_string in (\"FAIL\", \"ERROR\"):\n+            status_string = self.colorize(status_string, Color.red)\n+\n         return f\"== [{self.colorize(filename, Color.lightgrey)}] {status_string}\"\n \n     def format_violation(\ndiff --git a/src/sqlfluff/core/linter/linted_dir.py b/src/sqlfluff/core/linter/linted_dir.py\n--- a/src/sqlfluff/core/linter/linted_dir.py\n+++ b/src/sqlfluff/core/linter/linted_dir.py\n@@ -117,7 +117,11 @@ def persist_changes(\n         for file in self.files:\n             if file.num_violations(fixable=True, **kwargs) > 0:\n                 buffer[file.path] = file.persist_tree(suffix=fixed_file_suffix)\n-                result = buffer[file.path]\n+                result: Union[bool, str]\n+                if buffer[file.path] is True:\n+                    result = \"FIXED\"\n+                else:  # pragma: no cover\n+                    result = buffer[file.path]\n             else:  # pragma: no cover TODO?\n                 buffer[file.path] = True\n                 result = \"SKIP\"\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@@ -557,6 +557,18 @@ def test__cli__command_lint_parse(command):\n             ),\n             1,\n         ),\n+        # Test that setting --quiet with --verbose raises an error.\n+        (\n+            (\n+                fix,\n+                [\n+                    \"--quiet\",\n+                    \"--verbose\",\n+                    \"test/fixtures/cli/fail_many.sql\",\n+                ],\n+            ),\n+            2,\n+        ),\n     ],\n )\n def test__cli__command_lint_parse_with_retcode(command, ret_code):\n@@ -1891,7 +1903,7 @@ def test_cli_fix_disabled_progress_bar_deprecated_option(\n \n \n def test__cli__fix_multiple_errors_no_show_errors():\n-    \"\"\"Basic checking of lint functionality.\"\"\"\n+    \"\"\"Test the fix output.\"\"\"\n     result = invoke_assert_code(\n         ret_code=1,\n         args=[\n@@ -1910,8 +1922,57 @@ def test__cli__fix_multiple_errors_no_show_errors():\n     assert result.output.replace(\"\\\\\", \"/\").startswith(multiple_expected_output)\n \n \n+def test__cli__fix_multiple_errors_quiet_force():\n+    \"\"\"Test the fix --quiet option with --force.\"\"\"\n+    result = invoke_assert_code(\n+        ret_code=0,\n+        args=[\n+            fix,\n+            [\n+                \"--disable-progress-bar\",\n+                \"test/fixtures/linter/multiple_sql_errors.sql\",\n+                \"--force\",\n+                \"--quiet\",\n+                \"-x\",\n+                \"_fix\",\n+            ],\n+        ],\n+    )\n+    normalised_output = result.output.replace(\"\\\\\", \"/\")\n+    assert normalised_output.startswith(\n+        \"\"\"1 fixable linting violations found\n+== [test/fixtures/linter/multiple_sql_errors.sql] FIXED\"\"\"\n+    )\n+\n+\n+def test__cli__fix_multiple_errors_quiet_no_force():\n+    \"\"\"Test the fix --quiet option without --force.\"\"\"\n+    result = invoke_assert_code(\n+        ret_code=0,\n+        args=[\n+            fix,\n+            [\n+                \"--disable-progress-bar\",\n+                \"test/fixtures/linter/multiple_sql_errors.sql\",\n+                \"--quiet\",\n+                \"-x\",\n+                \"_fix\",\n+            ],\n+            # Test with the confirmation step.\n+            \"y\",\n+        ],\n+    )\n+    normalised_output = result.output.replace(\"\\\\\", \"/\")\n+    assert normalised_output.startswith(\n+        \"\"\"1 fixable linting violations found\n+Are you sure you wish to attempt to fix these? [Y/n] ...\n+== [test/fixtures/linter/multiple_sql_errors.sql] FIXED\n+All Finished\"\"\"\n+    )\n+\n+\n def test__cli__fix_multiple_errors_show_errors():\n-    \"\"\"Basic checking of lint functionality.\"\"\"\n+    \"\"\"Test the fix --show-lint-violations option.\"\"\"\n     result = invoke_assert_code(\n         ret_code=1,\n         args=[\n", "problem_statement": "Enable quiet mode/no-verbose in CLI for use in pre-commit hook\nThere seems to be only an option to increase the level of verbosity when using SQLFluff [CLI](https://docs.sqlfluff.com/en/stable/cli.html), not to limit it further.\r\n\r\nIt would be great to have an option to further limit the amount of prints when running `sqlfluff fix`, especially in combination with deployment using a pre-commit hook. For example, only print the return status and the number of fixes applied, similar to how it is when using `black` in a pre-commit hook:\r\n![image](https://user-images.githubusercontent.com/10177212/140480676-dc98d00b-4383-44f2-bb90-3301a6eedec2.png)\r\n\r\nThis hides the potentially long list of fixes that are being applied to the SQL files, which can get quite verbose.\n", "hints_text": "", "created_at": "2023-04-16T14:24:42Z", "version": "1.4", "FAIL_TO_PASS": ["test/cli/commands_test.py::test__cli__fix_multiple_errors_quiet_force", "test/cli/commands_test.py::test__cli__fix_multiple_errors_quiet_no_force"], "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[command26]", "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-0]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command4-0]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command5-2]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command6-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command7-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command8-1]", "test/cli/commands_test.py::test__cli__command_lint_parse_with_retcode[command9-2]", "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[LT01-test/fixtures/linter/indentation_errors.sql0]", "test/cli/commands_test.py::test__cli__command__fix[LT01-test/fixtures/linter/whitespace_errors.sql]", "test/cli/commands_test.py::test__cli__command__fix[LT01-test/fixtures/linter/indentation_errors.sql1]", "test/cli/commands_test.py::test__cli__command__fix[LT02-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_format_stdin[select", "test/cli/commands_test.py::test__cli__command_format_stdin[", "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[LT01-test/fixtures/linter/indentation_errors.sql-y-0-0]", "test/cli/commands_test.py::test__cli__command__fix_no_force[LT01-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[None-none]", "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_multiple_files[outfile-none]", "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::TestProgressBars::test_cli_fix_disabled_progress_bar", "test/cli/commands_test.py::TestProgressBars::test_cli_fix_disabled_progress_bar_deprecated_option", "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": "d19de0ecd16d298f9e3bfb91da122734c40c01e5"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1218", "base_commit": "40ba4bd5c8b91754aa73e638ed984ab9657847cd", "patch": "diff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -977,7 +977,7 @@ def infer_temperature_model(self):\n             return self.faiman_temp\n         elif {'noct_installed'} <= params:\n             return self.fuentes_temp\n-        elif {'noct', 'eta_m_ref'} <= params:\n+        elif {'noct', 'module_efficiency'} <= params:\n             return self.noct_sam_temp\n         else:\n             raise ValueError(f'could not infer temperature model from '\ndiff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\n--- a/pvlib/pvsystem.py\n+++ b/pvlib/pvsystem.py\n@@ -671,7 +671,9 @@ def pvsyst_celltemp(self, poa_global, temp_air, wind_speed=1.0):\n         wind_speed = self._validate_per_array(wind_speed, system_wide=True)\n \n         def build_celltemp_kwargs(array):\n-            return {**_build_kwargs(['eta_m', 'alpha_absorption'],\n+            # TODO remove 'eta_m' after deprecation of this parameter\n+            return {**_build_kwargs(['eta_m', 'module_efficiency',\n+                                     'alpha_absorption'],\n                                     array.module_parameters),\n                     **_build_kwargs(['u_c', 'u_v'],\n                                     array.temperature_model_parameters)}\n@@ -843,10 +845,10 @@ def _build_kwargs_noct_sam(array):\n                 # bundled with kwargs for simplicity\n                 temp_model_kwargs['noct'] = \\\n                     array.temperature_model_parameters['noct']\n-                temp_model_kwargs['eta_m_ref'] = \\\n-                    array.temperature_model_parameters['eta_m_ref']\n+                temp_model_kwargs['module_efficiency'] = \\\n+                    array.temperature_model_parameters['module_efficiency']\n             except KeyError:\n-                msg = ('Parameters noct and eta_m_ref are required.'\n+                msg = ('Parameters noct and module_efficiency are required.'\n                        ' Found {} in temperature_model_parameters.'\n                        .format(array.temperature_model_parameters))\n                 raise KeyError(msg)\ndiff --git a/pvlib/temperature.py b/pvlib/temperature.py\n--- a/pvlib/temperature.py\n+++ b/pvlib/temperature.py\n@@ -6,6 +6,7 @@\n import numpy as np\n import pandas as pd\n from pvlib.tools import sind\n+from pvlib._deprecation import warn_deprecated\n \n TEMPERATURE_MODEL_PARAMETERS = {\n     'sapm': {\n@@ -285,7 +286,7 @@ def sapm_cell_from_module(module_temperature, poa_global, deltaT,\n \n \n def pvsyst_cell(poa_global, temp_air, wind_speed=1.0, u_c=29.0, u_v=0.0,\n-                eta_m=0.1, alpha_absorption=0.9):\n+                eta_m=None, module_efficiency=0.1, alpha_absorption=0.9):\n     r\"\"\"\n     Calculate cell temperature using an empirical heat loss factor model\n     as implemented in PVsyst.\n@@ -313,12 +314,14 @@ def pvsyst_cell(poa_global, temp_air, wind_speed=1.0, u_c=29.0, u_v=0.0,\n     u_v : float, default 0.0\n         Combined heat loss factor influenced by wind. Parameter :math:`U_{v}`\n         in :eq:`pvsyst`.\n-        :math:`\\left[ \\frac{\\text{W}/\\text{m}^2}{\\text{C}\\ \\left( \\text{m/s} \\right)} \\right]`\n+        :math:`\\left[ \\frac{\\text{W}/\\text{m}^2}{\\text{C}\\ \\left( \\text{m/s} \\right)} \\right]`  # noQA: E501\n+\n+    eta_m : numeric, default None (deprecated, use module_efficiency instead)\n \n-    eta_m : numeric, default 0.1\n-        Module external efficiency as a fraction, i.e.,\n-        :math:`DC\\ power / (POA\\ irradiance \\times module\\ area)`.\n-        Parameter :math:`\\eta_{m}` in :eq:`pvsyst`.\n+    module_efficiency : numeric, default 0.1\n+        Module external efficiency as a fraction. Parameter :math:`\\eta_{m}`\n+        in :eq:`pvsyst`. Calculate as\n+        :math:`\\eta_{m} = DC\\ power / (POA\\ irradiance \\times module\\ area)`.\n \n     alpha_absorption : numeric, default 0.9\n         Absorption coefficient. Parameter :math:`\\alpha` in :eq:`pvsyst`.\n@@ -370,8 +373,13 @@ def pvsyst_cell(poa_global, temp_air, wind_speed=1.0, u_c=29.0, u_v=0.0,\n     37.93103448275862\n     \"\"\"\n \n+    if eta_m:\n+        warn_deprecated(\n+            since='v0.9', message='eta_m overwriting module_efficiency',\n+            name='eta_m', alternative='module_efficiency', removal='v0.10')\n+        module_efficiency = eta_m\n     total_loss_factor = u_c + u_v * wind_speed\n-    heat_input = poa_global * alpha_absorption * (1 - eta_m)\n+    heat_input = poa_global * alpha_absorption * (1 - module_efficiency)\n     temp_difference = heat_input / total_loss_factor\n     return temp_air + temp_difference\n \n@@ -719,7 +727,7 @@ def _adj_for_mounting_standoff(x):\n                         [0., 18., 11., 6., 2., 0.])\n \n \n-def noct_sam(poa_global, temp_air, wind_speed, noct, eta_m_ref,\n+def noct_sam(poa_global, temp_air, wind_speed, noct, module_efficiency,\n              effective_irradiance=None, transmittance_absorptance=0.9,\n              array_height=1, mount_standoff=4):\n     r'''\n@@ -744,9 +752,9 @@ def noct_sam(poa_global, temp_air, wind_speed, noct, eta_m_ref,\n         Nominal operating cell temperature [C], determined at conditions of\n         800 W/m^2 irradiance, 20 C ambient air temperature and 1 m/s wind.\n \n-    eta_m_ref : float\n+    module_efficiency : float\n         Module external efficiency [unitless] at reference conditions of\n-        1000 W/m^2 and 20C. Calculate as\n+        1000 W/m^2 and 20C. Denoted as :math:`eta_{m}` in [1]_. Calculate as\n         :math:`\\eta_{m} = \\frac{V_{mp} I_{mp}}{A \\times 1000 W/m^2}`\n         where A is module area [m^2].\n \n@@ -810,6 +818,6 @@ def noct_sam(poa_global, temp_air, wind_speed, noct, eta_m_ref,\n     # [1] Eq. 10.37 isn't clear on exactly what \"G\" is. SAM SSC code uses\n     # poa_global where G appears\n     cell_temp_init = poa_global / 800. * (noct_adj - 20.)\n-    heat_loss = 1 - eta_m_ref / tau_alpha\n+    heat_loss = 1 - module_efficiency / tau_alpha\n     wind_loss = 9.5 / (5.7 + 3.8 * wind_adj)\n     return temp_air + cell_temp_init * heat_loss * wind_loss\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@@ -201,7 +201,7 @@ def pvwatts_dc_pvwatts_ac_faiman_temp_system():\n @pytest.fixture(scope=\"function\")\n def pvwatts_dc_pvwatts_ac_pvsyst_temp_system():\n     module_parameters = {'pdc0': 220, 'gamma_pdc': -0.003}\n-    temp_model_params = {'u_c': 29.0, 'u_v': 0.0, 'eta_m': 0.1,\n+    temp_model_params = {'u_c': 29.0, 'u_v': 0.0, 'module_efficiency': 0.1,\n                          'alpha_absorption': 0.9}\n     inverter_parameters = {'pdc0': 220, 'eta_inv_nom': 0.95}\n     system = PVSystem(surface_tilt=32.2, surface_azimuth=180,\n@@ -226,7 +226,7 @@ def pvwatts_dc_pvwatts_ac_fuentes_temp_system():\n @pytest.fixture(scope=\"function\")\n def pvwatts_dc_pvwatts_ac_noct_sam_temp_system():\n     module_parameters = {'pdc0': 220, 'gamma_pdc': -0.003}\n-    temp_model_params = {'noct': 45, 'eta_m_ref': 0.2}\n+    temp_model_params = {'noct': 45, 'module_efficiency': 0.2}\n     inverter_parameters = {'pdc0': 220, 'eta_inv_nom': 0.95}\n     system = PVSystem(surface_tilt=32.2, surface_azimuth=180,\n                       module_parameters=module_parameters,\n@@ -710,7 +710,7 @@ def test_run_model_with_weather_noct_sam_temp(sapm_dc_snl_ac_system, location,\n     weather['wind_speed'] = 5\n     weather['temp_air'] = 10\n     sapm_dc_snl_ac_system.temperature_model_parameters = {\n-        'noct': 45, 'eta_m_ref': 0.2\n+        'noct': 45, 'module_efficiency': 0.2\n     }\n     mc = ModelChain(sapm_dc_snl_ac_system, location)\n     mc.temperature_model = 'noct_sam'\n@@ -941,7 +941,7 @@ def test__prepare_temperature_arrays_weather(sapm_dc_snl_ac_system_same_arrays,\n                            ModelChain.faiman_temp),\n                           ({'noct_installed': 45},\n                            ModelChain.fuentes_temp),\n-                          ({'noct': 45, 'eta_m_ref': 0.2},\n+                          ({'noct': 45, 'module_efficiency': 0.2},\n                            ModelChain.noct_sam_temp)])\n def test_temperature_models_arrays_multi_weather(\n         temp_params, temp_model,\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@@ -392,7 +392,7 @@ def two_array_system(pvsyst_module_params, cec_module_params):\n     temperature_model['noct_installed'] = 45\n     # parameters for noct_sam temperature model\n     temperature_model['noct'] = 45.\n-    temperature_model['eta_m_ref'] = 0.2\n+    temperature_model['module_efficiency'] = 0.2\n     module_params = {**pvsyst_module_params, **cec_module_params}\n     return pvsystem.PVSystem(\n         arrays=[\n@@ -471,8 +471,9 @@ def test_PVSystem_pvsyst_celltemp(mocker):\n     temp_model_params = temperature.TEMPERATURE_MODEL_PARAMETERS['pvsyst'][\n         parameter_set]\n     alpha_absorption = 0.85\n-    eta_m = 0.17\n-    module_parameters = {'alpha_absorption': alpha_absorption, 'eta_m': eta_m}\n+    module_efficiency = 0.17\n+    module_parameters = {'alpha_absorption': alpha_absorption,\n+                         'module_efficiency': module_efficiency}\n     system = pvsystem.PVSystem(module_parameters=module_parameters,\n                                temperature_model_parameters=temp_model_params)\n     mocker.spy(temperature, 'pvsyst_cell')\n@@ -481,8 +482,9 @@ def test_PVSystem_pvsyst_celltemp(mocker):\n     wind = 0.5\n     out = system.pvsyst_celltemp(irrad, temp, wind_speed=wind)\n     temperature.pvsyst_cell.assert_called_once_with(\n-        irrad, temp, wind, temp_model_params['u_c'], temp_model_params['u_v'],\n-        eta_m, alpha_absorption)\n+        irrad, temp, wind_speed=wind, u_c=temp_model_params['u_c'],\n+        u_v=temp_model_params['u_v'], module_efficiency=module_efficiency,\n+        alpha_absorption=alpha_absorption)\n     assert (out < 90) and (out > 70)\n \n \n@@ -500,16 +502,16 @@ def test_PVSystem_faiman_celltemp(mocker):\n \n \n def test_PVSystem_noct_celltemp(mocker):\n-    poa_global, temp_air, wind_speed, noct, eta_m_ref = (1000., 25., 1., 45.,\n-                                                         0.2)\n+    poa_global, temp_air, wind_speed, noct, module_efficiency = (\n+        1000., 25., 1., 45., 0.2)\n     expected = 55.230790492\n-    temp_model_params = {'noct': noct, 'eta_m_ref': eta_m_ref}\n+    temp_model_params = {'noct': noct, 'module_efficiency': module_efficiency}\n     system = pvsystem.PVSystem(temperature_model_parameters=temp_model_params)\n     mocker.spy(temperature, 'noct_sam')\n     out = system.noct_sam_celltemp(poa_global, temp_air, wind_speed)\n     temperature.noct_sam.assert_called_once_with(\n         poa_global, temp_air, wind_speed, effective_irradiance=None, noct=noct,\n-        eta_m_ref=eta_m_ref)\n+        module_efficiency=module_efficiency)\n     assert_allclose(out, expected)\n     # dufferent types\n     out = system.noct_sam_celltemp(np.array(poa_global), np.array(temp_air),\n@@ -533,8 +535,8 @@ def test_PVSystem_noct_celltemp(mocker):\n \n \n def test_PVSystem_noct_celltemp_error():\n-    poa_global, temp_air, wind_speed, eta_m_ref = (1000., 25., 1., 0.2)\n-    temp_model_params = {'eta_m_ref': eta_m_ref}\n+    poa_global, temp_air, wind_speed, module_efficiency = (1000., 25., 1., 0.2)\n+    temp_model_params = {'module_efficiency': module_efficiency}\n     system = pvsystem.PVSystem(temperature_model_parameters=temp_model_params)\n     with pytest.raises(KeyError):\n         system.noct_sam_celltemp(poa_global, temp_air, wind_speed)\ndiff --git a/pvlib/tests/test_temperature.py b/pvlib/tests/test_temperature.py\n--- a/pvlib/tests/test_temperature.py\n+++ b/pvlib/tests/test_temperature.py\n@@ -6,6 +6,7 @@\n from numpy.testing import assert_allclose\n \n from pvlib import temperature, tools\n+from pvlib._deprecation import pvlibDeprecationWarning\n \n \n @pytest.fixture\n@@ -72,7 +73,7 @@ def test_pvsyst_cell_default():\n \n def test_pvsyst_cell_kwargs():\n     result = temperature.pvsyst_cell(900, 20, wind_speed=5.0, u_c=23.5,\n-                                     u_v=6.25, eta_m=0.1)\n+                                     u_v=6.25, module_efficiency=0.1)\n     assert_allclose(result, 33.315, 0.001)\n \n \n@@ -96,6 +97,13 @@ def test_pvsyst_cell_series():\n     assert_series_equal(expected, result)\n \n \n+def test_pvsyst_cell_eta_m_deprecated():\n+    with pytest.warns(pvlibDeprecationWarning):\n+        result = temperature.pvsyst_cell(900, 20, wind_speed=5.0, u_c=23.5,\n+                                         u_v=6.25, eta_m=0.1)\n+        assert_allclose(result, 33.315, 0.001)\n+\n+\n def test_faiman_default():\n     result = temperature.faiman(900, 20, 5)\n     assert_allclose(result, 35.203, 0.001)\n@@ -215,16 +223,16 @@ def test_fuentes_timezone(tz):\n \n \n def test_noct_sam():\n-    poa_global, temp_air, wind_speed, noct, eta_m_ref = (1000., 25., 1., 45.,\n-                                                         0.2)\n+    poa_global, temp_air, wind_speed, noct, module_efficiency = (\n+        1000., 25., 1., 45., 0.2)\n     expected = 55.230790492\n     result = temperature.noct_sam(poa_global, temp_air, wind_speed, noct,\n-                                  eta_m_ref)\n+                                  module_efficiency)\n     assert_allclose(result, expected)\n     # test with different types\n     result = temperature.noct_sam(np.array(poa_global), np.array(temp_air),\n                                   np.array(wind_speed), np.array(noct),\n-                                  np.array(eta_m_ref))\n+                                  np.array(module_efficiency))\n     assert_allclose(result, expected)\n     dr = pd.date_range(start='2020-01-01 12:00:00', end='2020-01-01 13:00:00',\n                        freq='1H')\n@@ -232,7 +240,7 @@ def test_noct_sam():\n                                   pd.Series(index=dr, data=temp_air),\n                                   pd.Series(index=dr, data=wind_speed),\n                                   pd.Series(index=dr, data=noct),\n-                                  eta_m_ref)\n+                                  module_efficiency)\n     assert_series_equal(result, pd.Series(index=dr, data=expected))\n \n \n@@ -242,7 +250,7 @@ def test_noct_sam_against_sam():\n     # NOCT cell temperature model), with the only change being the soiling\n     # loss is set to 0. Weather input is TMY3 for Phoenix AZ.\n     # Values are taken from the Jan 1 12:00:00 timestamp.\n-    poa_total, temp_air, wind_speed, noct, eta_m_ref = (\n+    poa_total, temp_air, wind_speed, noct, module_efficiency = (\n         860.673, 25, 3, 46.4, 0.20551)\n     poa_total_after_refl = 851.458  # from SAM output\n     # compute effective irradiance\n@@ -259,7 +267,7 @@ def test_noct_sam_against_sam():\n     array_height = 1\n     mount_standoff = 4.0\n     result = temperature.noct_sam(poa_total, temp_air, wind_speed, noct,\n-                                  eta_m_ref, effective_irradiance,\n+                                  module_efficiency, effective_irradiance,\n                                   transmittance_absorptance, array_height,\n                                   mount_standoff)\n     expected = 43.0655\n@@ -268,14 +276,14 @@ def test_noct_sam_against_sam():\n \n \n def test_noct_sam_options():\n-    poa_global, temp_air, wind_speed, noct, eta_m_ref = (1000., 25., 1., 45.,\n-                                                         0.2)\n+    poa_global, temp_air, wind_speed, noct, module_efficiency = (\n+        1000., 25., 1., 45., 0.2)\n     effective_irradiance = 1100.\n     transmittance_absorptance = 0.8\n     array_height = 2\n     mount_standoff = 2.0\n     result = temperature.noct_sam(poa_global, temp_air, wind_speed, noct,\n-                                  eta_m_ref, effective_irradiance,\n+                                  module_efficiency, effective_irradiance,\n                                   transmittance_absorptance, array_height,\n                                   mount_standoff)\n     expected = 60.477703576\n", "problem_statement": "change eta_m to module_efficiency\n`temperature.noct_sam` uses `eta_m_ref` to describe the module efficiency at reference conditions and `temperature.pvsyst_cell` uses `eta_m` to describe the module efficiency generically.\r\n\r\nJust calling both of these `module_efficiency` would make the function signatures easily understandable by many more people. I'd be ok with `module_efficiency_ref` but I don't think that precision is very important.\r\n\r\nI skimmed [pvterms](https://duramat.github.io/pv-terms/) and didn't see a suggestion for this quantity.\r\n\r\n`temperature.noct_sam` has not yet been released and it's just a positional argument, so changing the name is trivial. `temperature.pvsyst_cell` would need a deprecation cycle.\r\n\r\nOriginally discussed in https://github.com/pvlib/pvlib-python/pull/1177#discussion_r589081257\r\n\r\nAssignment of milestone indicates that we will act on this or close it forever before 0.9 is released.\r\n\n", "hints_text": "I support this change.", "created_at": "2021-05-06T16:02:38Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_modelchain.py::test_run_model_with_weather_noct_sam_temp", "pvlib/tests/test_modelchain.py::test_temperature_models_arrays_multi_weather[temp_params4-noct_sam_temp]", "pvlib/tests/test_modelchain.py::test_infer_temp_model[noct_sam_temp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvsyst_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[noct_sam_celltemp]", "pvlib/tests/test_temperature.py::test_pvsyst_cell_kwargs", "pvlib/tests/test_temperature.py::test_pvsyst_cell_eta_m_deprecated"], "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_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_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_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_faiman_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_noct_celltemp_error", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_functions[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_temp[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_multi_wind[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_short[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_temp_too_long[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_short[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_wind_too_long[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[faiman_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[pvsyst_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[sapm_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[fuentes_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_multi_array_celltemp_poa_length_mismatch[noct_sam_celltemp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp_override", "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_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_change_surface_azimuth", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_albedo", "pvlib/tests/test_pvsystem.py::test_PVSystem_modules_per_string", "pvlib/tests/test_pvsystem.py::test_PVSystem_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_combine_loss_factors", "pvlib/tests/test_pvsystem.py::test_no_extra_kwargs", "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_temperature.py::test_sapm_cell", "pvlib/tests/test_temperature.py::test_sapm_module", "pvlib/tests/test_temperature.py::test_sapm_cell_from_module", "pvlib/tests/test_temperature.py::test_sapm_ndarray", "pvlib/tests/test_temperature.py::test_sapm_series", "pvlib/tests/test_temperature.py::test_pvsyst_cell_default", "pvlib/tests/test_temperature.py::test_pvsyst_cell_ndarray", "pvlib/tests/test_temperature.py::test_pvsyst_cell_series", "pvlib/tests/test_temperature.py::test_faiman_default", "pvlib/tests/test_temperature.py::test_faiman_kwargs", "pvlib/tests/test_temperature.py::test_faiman_list", "pvlib/tests/test_temperature.py::test_faiman_ndarray", "pvlib/tests/test_temperature.py::test_ross", "pvlib/tests/test_temperature.py::test_faiman_series", "pvlib/tests/test_temperature.py::test__temperature_model_params", "pvlib/tests/test_temperature.py::test_fuentes[pvwatts_8760_rackmount.csv-45]", "pvlib/tests/test_temperature.py::test_fuentes[pvwatts_8760_roofmount.csv-49]", "pvlib/tests/test_temperature.py::test_fuentes_timezone[None]", "pvlib/tests/test_temperature.py::test_fuentes_timezone[Etc/GMT+5]", "pvlib/tests/test_temperature.py::test_noct_sam", "pvlib/tests/test_temperature.py::test_noct_sam_against_sam", "pvlib/tests/test_temperature.py::test_noct_sam_options", "pvlib/tests/test_temperature.py::test_noct_sam_errors"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-2573", "base_commit": "8822bf4d831ccbf6bc63a44b34978daf6939d996", "patch": "diff --git a/src/sqlfluff/core/config.py b/src/sqlfluff/core/config.py\n--- a/src/sqlfluff/core/config.py\n+++ b/src/sqlfluff/core/config.py\n@@ -395,7 +395,7 @@ def load_config_up_to_path(\n \n     @classmethod\n     def find_ignore_config_files(\n-        cls, path, working_path=os.getcwd(), ignore_file_name=\".sqlfluffignore\"\n+        cls, path, working_path=Path.cwd(), ignore_file_name=\".sqlfluffignore\"\n     ):\n         \"\"\"Finds sqlfluff ignore files from both the path and its parent paths.\"\"\"\n         return set(\n@@ -417,8 +417,8 @@ def iter_config_locations_up_to_path(path, working_path=Path.cwd()):\n         The lowest priority is the user appdir, then home dir, then increasingly\n         the configs closest to the file being directly linted.\n         \"\"\"\n-        given_path = Path(path).resolve()\n-        working_path = Path(working_path).resolve()\n+        given_path = Path(path).absolute()\n+        working_path = Path(working_path).absolute()\n \n         # If we've been passed a file and not a directory,\n         # then go straight to the directory.\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@@ -843,6 +843,7 @@ def paths_from_path(\n         # matched, but we warn the users when that happens\n         is_exact_file = os.path.isfile(path)\n \n+        path_walk: WalkableType\n         if is_exact_file:\n             # When the exact file to lint is passed, we\n             # fill path_walk with an input that follows\n@@ -850,24 +851,26 @@ def paths_from_path(\n             #   (root, directories, files)\n             dirpath = os.path.dirname(path)\n             files = [os.path.basename(path)]\n-            ignore_file_paths = ConfigLoader.find_ignore_config_files(\n-                path=path, working_path=working_path, ignore_file_name=ignore_file_name\n-            )\n-            # Add paths that could contain \"ignore files\"\n-            # to the path_walk list\n-            path_walk_ignore_file = [\n-                (\n-                    os.path.dirname(ignore_file_path),\n-                    None,\n-                    # Only one possible file, since we only\n-                    # have one \"ignore file name\"\n-                    [os.path.basename(ignore_file_path)],\n-                )\n-                for ignore_file_path in ignore_file_paths\n-            ]\n-            path_walk: WalkableType = [(dirpath, None, files)] + path_walk_ignore_file\n+            path_walk = [(dirpath, None, files)]\n         else:\n-            path_walk = os.walk(path)\n+            path_walk = list(os.walk(path))\n+\n+        ignore_file_paths = ConfigLoader.find_ignore_config_files(\n+            path=path, working_path=working_path, ignore_file_name=ignore_file_name\n+        )\n+        # Add paths that could contain \"ignore files\"\n+        # to the path_walk list\n+        path_walk_ignore_file = [\n+            (\n+                os.path.dirname(ignore_file_path),\n+                None,\n+                # Only one possible file, since we only\n+                # have one \"ignore file name\"\n+                [os.path.basename(ignore_file_path)],\n+            )\n+            for ignore_file_path in ignore_file_paths\n+        ]\n+        path_walk += path_walk_ignore_file\n \n         # If it's a directory then expand the path!\n         buffer = []\n", "test_patch": "diff --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@@ -1,10 +1,12 @@\n \"\"\"The Test file for the linter class.\"\"\"\n \n-import pytest\n+import os\n import logging\n from typing import List\n from unittest.mock import patch\n \n+import pytest\n+\n from sqlfluff.core import Linter, FluffConfig\n from sqlfluff.core.linter import runner\n from sqlfluff.core.errors import SQLLexError, SQLBaseError, SQLLintError, SQLParseError\n@@ -91,6 +93,23 @@ def test__linter__path_from_paths__explicit_ignore():\n     assert len(paths) == 0\n \n \n+def test__linter__path_from_paths__sqlfluffignore_current_directory():\n+    \"\"\"Test that .sqlfluffignore in the current directory is read when dir given.\"\"\"\n+    oldcwd = os.getcwd()\n+    try:\n+        os.chdir(\"test/fixtures/linter/sqlfluffignore\")\n+        lntr = Linter()\n+        paths = lntr.paths_from_path(\n+            \"path_a/\",\n+            ignore_non_existent_files=True,\n+            ignore_files=True,\n+            working_path=\"test/fixtures/linter/sqlfluffignore/\",\n+        )\n+        assert len(paths) == 0\n+    finally:\n+        os.chdir(oldcwd)\n+\n+\n def test__linter__path_from_paths__dot():\n     \"\"\"Test extracting paths from a dot.\"\"\"\n     lntr = Linter()\n", "problem_statement": "Configuration from current working path not being loaded when path provided.\nI have the following directory structure.\r\n```\r\n~/GitHub/sqlfluff-bug\r\n\u279c  tree -a\r\n.\r\n\u251c\u2500\u2500 .sqlfluffignore\r\n\u251c\u2500\u2500 ignore_me_1.sql\r\n\u251c\u2500\u2500 path_a\r\n\u2502\u00a0\u00a0 \u2514\u2500\u2500 ignore_me_2.sql\r\n\u2514\u2500\u2500 path_b\r\n    \u251c\u2500\u2500 ignore_me_3.sql\r\n    \u2514\u2500\u2500 lint_me_1.sql\r\n\r\n2 directories, 5 files\r\n```\r\nAnd the following ignore file\r\n\r\n```\r\n~/GitHub/sqlfluff-bug\r\n\u279c  cat .sqlfluffignore\r\n\r\n~/GitHub/sqlfluff-bug\r\n\u279c  cat .sqlfluffignore\r\nignore_me_1.sql\r\npath_a/\r\npath_b/ignore_me_3.sql%\r\n```\r\n\r\nWhen I run the following I get the expected result. Sqlfluff only lints the one file that is not ignored.\r\n```\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint .\r\n\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint .\r\n== [path_b/lint_me_1.sql] FAIL\r\nL:   2 | P:   1 | L003 | Indent expected and not found compared to line #1\r\nL:   2 | P:  10 | L010 | Inconsistent capitalisation of keywords.\r\n```\r\n\r\nHowever when I run the lint explicitly on one of the two directories then ignored files are also linted.\r\n\r\n```\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint path_a\r\n\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint path_a\r\n== [path_a/ignore_me_2.sql] FAIL\r\nL:   2 | P:   1 | L003 | Indent expected and not found compared to line #1\r\nL:   2 | P:  10 | L010 | Inconsistent capitalisation of keywords.\r\n\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint path_b\r\n\r\n~/GitHub/sqlfluff-bug\r\n\u279c  sqlfluff lint path_b\r\n== [path_b/ignore_me_3.sql] FAIL\r\nL:   2 | P:   1 | L003 | Indent expected and not found compared to line #1\r\nL:   2 | P:  10 | L010 | Inconsistent capitalisation of keywords.\r\n== [path_b/lint_me_1.sql] FAIL\r\nL:   2 | P:   1 | L003 | Indent expected and not found compared to line #1\r\nL:   2 | P:  10 | L010 | Inconsistent capitalisation of keywords.\r\n```\r\n\r\nIf this is the expected behaviour then it might be worthwhile to add an example to the [docs](https://docs.sqlfluff.com/en/latest/configuration.html#sqlfluffignore).\r\n\r\nEdit: I've replicated this issue on sqlfluff version 0.3.2 to 0.3.6.\n", "hints_text": "This is currently functioning as expected in that it only looks for `.sqlfluffignore` files within the directories you specify. So if you point sqlfluff at `/path_b`, it would only looks for a `.sqlfluffignore` file at `/path_b/.sqlfluffignore` and any child directories of that. It won't check in parents of the given file.\r\n\r\nI think that's the expected behavior consistent with `.dockerignore` and `.gitignore` .\r\n\r\nI agree about clarifying the documentation, which uses the phrase `placed in the root of your project` which I think alone is misleading.\nI think the behavior described in this issue is desirable.\r\n\r\nFor CI with pre-commit for example, right now I would need to add a `.sqlfluffignore` to each sub-directory containing sql files I want to ignore. That's because pre-commit will give the full path pointing to each file that changed before commit.\r\n\r\nI'm not sure the behavior is consistent with `.gitignore` because the \"project root\" stays the same and `.gitignore` files are applied from top level down to the subdirectory of each file, while in `sqlfluff` we don't really have a project root, which I think could come from a new configuration in `.sqlfluff` (or we could assume `cwd` if it's parent directory of the file we're trying to lint).\nI've just hit this issue myself (but for configuration file) and I agree with @dmateusp on this one.\r\n\r\nI think this is more than a documentation issue.\r\n\r\nFrom putting together the initial configuration code, the config loader, *should* check the current working directory for config loading, but it feels like that isn't working right now.", "created_at": "2022-02-07T17:47:05Z", "version": "0.9", "FAIL_TO_PASS": ["test/core/linter_test.py::test__linter__path_from_paths__sqlfluffignore_current_directory"], "PASS_TO_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__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__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__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"], "environment_setup_commit": "a4dcf3f08d95cbde4efb39969b0ab8e33a791f21"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-2849", "base_commit": "0bbd70f38a3318b9a488d988d06e8005e222d6ac", "patch": "diff --git a/plugins/sqlfluff-templater-dbt/sqlfluff_templater_dbt/templater.py b/plugins/sqlfluff-templater-dbt/sqlfluff_templater_dbt/templater.py\n--- a/plugins/sqlfluff-templater-dbt/sqlfluff_templater_dbt/templater.py\n+++ b/plugins/sqlfluff-templater-dbt/sqlfluff_templater_dbt/templater.py\n@@ -522,17 +522,21 @@ def make_template(in_str):\n             # sliced_file to reflect the mapping of the added character(s) back\n             # to the raw SQL.\n             templated_sql = templated_sql + \"\\n\" * n_trailing_newlines\n-            sliced_file.append(\n-                TemplatedFileSlice(\n-                    slice_type=\"literal\",\n-                    source_slice=slice(\n-                        len(source_dbt_sql) - n_trailing_newlines, len(source_dbt_sql)\n-                    ),\n-                    templated_slice=slice(\n-                        len(templated_sql) - n_trailing_newlines, len(templated_sql)\n-                    ),\n+            if sliced_file and sliced_file[-1].templated_slice.stop != len(\n+                templated_sql\n+            ):\n+                sliced_file.append(\n+                    TemplatedFileSlice(\n+                        slice_type=\"literal\",\n+                        source_slice=slice(\n+                            len(source_dbt_sql) - n_trailing_newlines,\n+                            len(source_dbt_sql),\n+                        ),\n+                        templated_slice=slice(\n+                            len(templated_sql) - n_trailing_newlines, len(templated_sql)\n+                        ),\n+                    )\n                 )\n-            )\n         return (\n             TemplatedFile(\n                 source_str=source_dbt_sql,\ndiff --git a/src/sqlfluff/core/templaters/base.py b/src/sqlfluff/core/templaters/base.py\n--- a/src/sqlfluff/core/templaters/base.py\n+++ b/src/sqlfluff/core/templaters/base.py\n@@ -73,6 +73,7 @@ def __init__(\n         templated_str: Optional[str] = None,\n         sliced_file: Optional[List[TemplatedFileSlice]] = None,\n         raw_sliced: Optional[List[RawFileSlice]] = None,\n+        check_consistency=True,\n     ):\n         \"\"\"Initialise the TemplatedFile.\n \n@@ -104,6 +105,36 @@ def __init__(\n         self._source_newlines = list(iter_indices_of_newlines(self.source_str))\n         self._templated_newlines = list(iter_indices_of_newlines(self.templated_str))\n \n+        # NOTE: The \"check_consistency\" flag should always be True when using\n+        # SQLFluff in real life. This flag was only added because some legacy\n+        # templater tests in test/core/templaters/jinja_test.py use hardcoded\n+        # test data with issues that will trigger errors here. It would be cool\n+        # to fix that data someday. I (Barry H.) started looking into it, but\n+        # it was much trickier than I expected, because bits of the same data\n+        # are shared across multiple tests.\n+        if check_consistency:\n+            # Sanity check raw string and slices.\n+            pos = 0\n+            rfs: RawFileSlice\n+            for idx, rfs in enumerate(self.raw_sliced):\n+                assert rfs.source_idx == pos\n+                pos += len(rfs.raw)\n+            assert pos == len(self.source_str)\n+\n+            # Sanity check templated string and slices.\n+            previous_slice = None\n+            tfs: Optional[TemplatedFileSlice] = None\n+            for idx, tfs in enumerate(self.sliced_file):\n+                if previous_slice:\n+                    assert (\n+                        tfs.templated_slice.start == previous_slice.templated_slice.stop\n+                    )\n+                else:\n+                    assert tfs.templated_slice.start == 0\n+                previous_slice = tfs\n+            if self.sliced_file and templated_str is not None:\n+                assert tfs.templated_slice.stop == len(templated_str)\n+\n     @classmethod\n     def from_string(cls, raw):\n         \"\"\"Create TemplatedFile from a string.\"\"\"\ndiff --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@@ -77,9 +77,6 @@ def trace(self) -> JinjaTrace:\n             except IndexError:\n                 pos2 = len(trace_template_output)\n             p = trace_template_output[pos1 + 1 : pos2]\n-            is_set_or_macro = p[:3] == \"set\"\n-            if is_set_or_macro:\n-                p = p[3:]\n             m_id = regex.match(r\"^([0-9a-f]+)(_(\\d+))?\", p)\n             if not m_id:\n                 raise ValueError(  # pragma: no cover\n@@ -98,18 +95,7 @@ def trace(self) -> JinjaTrace:\n             alt_id, content_info, literal = value\n             target_slice_idx = self.find_slice_index(alt_id)\n             slice_length = content_info if literal else len(str(content_info))\n-            if not is_set_or_macro:\n-                self.move_to_slice(target_slice_idx, slice_length)\n-            else:\n-                # If we find output from a {% set %} directive or a macro,\n-                # record a trace without reading or updating the program\n-                # counter. Such slices are always treated as \"templated\"\n-                # because they are inserted during expansion of templated\n-                # code (i.e. {% set %} variable or macro defined within the\n-                # file).\n-                self.record_trace(\n-                    slice_length, target_slice_idx, slice_type=\"templated\"\n-                )\n+            self.move_to_slice(target_slice_idx, slice_length)\n         return JinjaTrace(\n             self.make_template(self.raw_str).render(), self.raw_sliced, self.sliced_file\n         )\n@@ -241,9 +227,17 @@ def _slice_template(self) -> List[RawFileSlice]:\n                         idx,\n                     )\n                 )\n-                self.raw_slice_info[result[-1]] = self.slice_info_for_literal(\n-                    len(raw), \"\" if set_idx is None else \"set\"\n-                )\n+                if set_idx is None:\n+                    rsi = self.slice_info_for_literal(\n+                        len(raw), \"\" if set_idx is None else \"set\"\n+                    )\n+                else:\n+                    # For \"set\" blocks, don't generate alternate ID or code.\n+                    # Sometimes, dbt users use {% set %} blocks to generate\n+                    # queries that get sent to actual databases, thus causing\n+                    # errors if we tamper with it.\n+                    rsi = RawSliceInfo(None, None, [])\n+                self.raw_slice_info[result[-1]] = rsi\n                 idx += len(raw)\n                 continue\n             str_buff += raw\n@@ -326,15 +320,20 @@ def _slice_template(self) -> List[RawFileSlice]:\n                         # effects, but return a unique slice ID.\n                         if trimmed_content:\n                             assert m_open and m_close\n-                            unique_id = self.next_slice_id()\n-                            unique_alternate_id = unique_id\n-                            prefix = \"set\" if set_idx is not None else \"\"\n-                            open_ = m_open.group(1)\n-                            close_ = m_close.group(1)\n-                            alternate_code = (\n-                                f\"\\0{prefix}{unique_alternate_id} {open_} \"\n-                                f\"{trimmed_content} {close_}\"\n-                            )\n+                            # For \"set\" blocks, don't generate alternate ID or\n+                            # code. Sometimes, dbt users use {% set %} blocks to\n+                            # generate queries that get sent to actual\n+                            # databases, thus causing errors if we tamper with\n+                            # it.\n+                            if set_idx is None:\n+                                unique_id = self.next_slice_id()\n+                                unique_alternate_id = unique_id\n+                                open_ = m_open.group(1)\n+                                close_ = m_close.group(1)\n+                                alternate_code = (\n+                                    f\"\\0{unique_alternate_id} {open_} \"\n+                                    f\"{trimmed_content} {close_}\"\n+                                )\n                 if block_type == \"block_start\" and trimmed_content.split()[0] in (\n                     \"macro\",\n                     \"set\",\n@@ -343,16 +342,24 @@ def _slice_template(self) -> List[RawFileSlice]:\n                     # - {% set variable = value %}\n                     # - {% set variable %}value{% endset %}\n                     # https://jinja.palletsprojects.com/en/2.10.x/templates/#block-assignments\n-                    # When the second format is used, set the variable 'is_set'\n+                    # When the second format is used, set the variable 'set_idx'\n                     # to a non-None value. This info is used elsewhere, as\n                     # literals inside a {% set %} block require special handling\n                     # during the trace.\n                     trimmed_content_parts = trimmed_content.split(maxsplit=2)\n-                    if len(trimmed_content_parts) <= 2 or not trimmed_content_parts[\n-                        2\n-                    ].startswith(\"=\"):\n+                    if len(trimmed_content_parts) <= 2 or (\n+                        not trimmed_content_parts[1].endswith(\"=\")\n+                        and not trimmed_content_parts[2].startswith(\"=\")\n+                    ):\n                         set_idx = len(result)\n-                elif block_type == \"block_end\" and set_idx is not None:\n+                elif (\n+                    block_type == \"block_end\"\n+                    and set_idx is not None\n+                    and (\n+                        trimmed_content.startswith(\"endset\")\n+                        or trimmed_content.startswith(\"endmacro\")\n+                    )\n+                ):\n                     # Exiting a {% set %} block. Clear the indicator variable.\n                     set_idx = None\n                 m = regex.search(r\"\\s+$\", raw, regex.MULTILINE | regex.DOTALL)\n", "test_patch": "diff --git a/test/core/templaters/base_test.py b/test/core/templaters/base_test.py\n--- a/test/core/templaters/base_test.py\n+++ b/test/core/templaters/base_test.py\n@@ -134,6 +134,7 @@ def test__templated_file_get_line_pos_of_char_pos(\n         templated_str=templated_str,\n         sliced_file=file_slices,\n         fname=\"test\",\n+        check_consistency=False,\n     )\n     res_line_no, res_line_pos = file.get_line_pos_of_char_pos(in_charpos)\n     assert res_line_no == out_line_no\n@@ -287,6 +288,7 @@ def test__templated_file_templated_slice_to_source_slice(\n             for rs in raw_slices\n         ],\n         fname=\"test\",\n+        check_consistency=False,\n     )\n     source_slice = file.templated_slice_to_source_slice(in_slice)\n     literal_test = file.is_source_slice_literal(source_slice)\n@@ -303,5 +305,6 @@ def test__templated_file_source_only_slices():\n             RawFileSlice(\"b\" * 7, \"comment\", 10),\n             RawFileSlice(\"a\" * 10, \"literal\", 17),\n         ],\n+        check_consistency=False,\n     )\n     assert file.source_only_slices() == [RawFileSlice(\"b\" * 7, \"comment\", 10)]\ndiff --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@@ -370,6 +370,35 @@ def test__templater_jinja_slices(case: RawTemplatedTestCase):\n     assert actual_rs_source_list == case.expected_raw_sliced__source_list\n \n \n+def test_templater_set_block_handling():\n+    \"\"\"Test handling of literals in {% set %} blocks.\n+\n+    Specifically, verify they are not modified in the alternate template.\n+    \"\"\"\n+\n+    def run_query(sql):\n+        # Prior to the bug fix, this assertion failed. This was bad because,\n+        # inside JinjaTracer, dbt templates similar to the one in this test\n+        # would call the database with funky SQL (including weird strings it\n+        # uses internally like: 00000000000000000000000000000002.\n+        assert sql == \"\\n\\nselect 1 from foobarfoobarfoobarfoobar_dev\\n\\n\"\n+        return sql\n+\n+    t = JinjaTemplater(override_context=dict(run_query=run_query))\n+    instr = \"\"\"{% set my_query1 %}\n+select 1 from foobarfoobarfoobarfoobar_{{ \"dev\" }}\n+{% endset %}\n+{% set my_query2 %}\n+{{ my_query1 }}\n+{% endset %}\n+\n+{{ run_query(my_query2) }}\n+\"\"\"\n+    outstr, vs = t.process(in_str=instr, fname=\"test\", config=FluffConfig())\n+    assert str(outstr) == \"\\n\\n\\n\\n\\nselect 1 from foobarfoobarfoobarfoobar_dev\\n\\n\\n\"\n+    assert len(vs) == 0\n+\n+\n def test__templater_jinja_error_variable():\n     \"\"\"Test missing variable error handling in the jinja templater.\"\"\"\n     t = JinjaTemplater(override_context=dict(blah=\"foo\"))\n@@ -846,6 +875,45 @@ def test__templater_jinja_slice_template(test, result):\n                 (\"literal\", slice(312, 327, None), slice(27, 42, None)),\n             ],\n         ),\n+        (\n+            # Test for issue 2835. There's no space between \"col\" and \"=\"\n+            \"\"\"{% set col= \"col1\" %}\n+SELECT {{ col }}\n+\"\"\",\n+            None,\n+            [\n+                (\"block_start\", slice(0, 21, None), slice(0, 0, None)),\n+                (\"literal\", slice(21, 29, None), slice(0, 8, None)),\n+                (\"templated\", slice(29, 38, None), slice(8, 12, None)),\n+                (\"literal\", slice(38, 39, None), slice(12, 13, None)),\n+            ],\n+        ),\n+        (\n+            # Another test for issue 2835. The {% for %} loop inside the\n+            # {% set %} caused JinjaTracer to think the {% set %} ended\n+            # at the {% endfor %}\n+            \"\"\"{% set some_part_of_the_query %}\n+    {% for col in [\"col1\"] %}\n+    {{col}}\n+    {% endfor %}\n+{% endset %}\n+\n+SELECT {{some_part_of_the_query}}\n+FROM SOME_TABLE\n+\"\"\",\n+            None,\n+            [\n+                (\"block_start\", slice(0, 32, None), slice(0, 0, None)),\n+                (\"literal\", slice(32, 37, None), slice(0, 0, None)),\n+                (\"block_start\", slice(37, 62, None), slice(0, 0, None)),\n+                (\"block_end\", slice(79, 91, None), slice(0, 0, None)),\n+                (\"literal\", slice(91, 92, None), slice(0, 0, None)),\n+                (\"block_end\", slice(92, 104, None), slice(0, 0, None)),\n+                (\"literal\", slice(104, 113, None), slice(0, 9, None)),\n+                (\"templated\", slice(113, 139, None), slice(9, 29, None)),\n+                (\"literal\", slice(139, 156, None), slice(29, 46, None)),\n+            ],\n+        ),\n     ],\n )\n def test__templater_jinja_slice_file(raw_file, override_context, result, caplog):\n", "problem_statement": "Lint and fix throws exception when having jinja for loop inside set\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\nTo reproduce the error, create test.template.sql\r\n```\r\n{% set whitelisted= [\r\n    {'name': 'COL_1'},\r\n    {'name': 'COL_2'},\r\n    {'name': 'COL_3'}\r\n] %}\r\n\r\n{% set some_part_of_the_query %}\r\n    {% for col in whitelisted %}\r\n    {{col.name}}{{ \", \" if not loop.last }}\r\n    {% endfor %}\r\n{% endset %}\r\n\r\nSELECT {{some_part_of_the_query}}\r\nFROM SOME_TABLE\r\n\r\n```\r\n\r\nwhen running lint i get this error:\r\n```\r\n==== sqlfluff ====\r\nsqlfluff:               0.11.0 python:                 3.8.12\r\nimplementation:        cpython dialect:             snowflake\r\nverbosity:                   1 templater:               jinja\r\n\r\n==== readout ====\r\n\r\n=== [ path: test.template.sql ] ===\r\n\r\nWARNING    Unable to lint test.template.sql due to an internal error. Please report this as an issue with your query's contents and stacktrace below!\r\nTo hide this warning, add the failing file to .sqlfluffignore\r\nTraceback (most recent call last):\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/linter/runner.py\", line 103, in run\r\n    yield partial()\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/linter/linter.py\", line 666, in lint_rendered\r\n    parsed = cls.parse_rendered(rendered)\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/linter/linter.py\", line 352, in parse_rendered\r\n    tokens, lvs, config = cls._lex_templated_file(\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/linter/linter.py\", line 139, in _lex_templated_file\r\n    tokens, lex_vs = lexer.lex(templated_file)\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/parser/lexer.py\", line 321, in lex\r\n    segments: Tuple[RawSegment, ...] = self.elements_to_segments(\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/parser/lexer.py\", line 348, in elements_to_segments\r\n    source_slice = templated_file.templated_slice_to_source_slice(\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/templaters/base.py\", line 258, in templated_slice_to_source_slice\r\n    ts_stop_sf_start, ts_stop_sf_stop = self._find_slice_indices_of_templated_pos(\r\n  File \"lib/python3.8/site-packages/sqlfluff/core/templaters/base.py\", line 177, in _find_slice_indices_of_templated_pos\r\n    raise ValueError(\"Position Not Found\")\r\nValueError: Position Not Found\r\n \r\n==== summary ====\r\nviolations:        0 status:         PASS\r\nAll Finished \ud83d\udcdc \ud83c\udf89!\r\n\r\n```\r\n\r\nThis is the rendered query:\r\n```\r\n SELECT\r\n\r\n    COL_1,\r\n\r\n    COL_2,\r\n\r\n    COL_3\r\n\r\n\r\nFROM SOME_TABLE\r\n\r\n```\r\n\r\nAnd when trying around to make this work i removed the new lines between the selected columns like this:\r\n```\r\n{% set whitelisted= [\r\n    {'name': 'COL_1'},\r\n    {'name': 'COL_2'},\r\n    {'name': 'COL_3'}\r\n] %}\r\n\r\n{% set some_part_of_the_query %}\r\n    {% for col in whitelisted -%}\r\n    {{col.name}}{{ \", \" if not loop.last }}\r\n    {% endfor -%}\r\n{% endset %}\r\n\r\nSELECT {{some_part_of_the_query}}\r\nFROM SOME_TABLE\r\n\r\n```\r\n\r\nwhich renders:\r\n```\r\nSELECT\r\n    COL_1,\r\n    COL_2,\r\n    COL_3\r\n\r\nFROM SOME_TABLE\r\n\r\n```\r\n\r\nAnd this will make the linter pass:\r\n\r\n```\r\n==== sqlfluff ====\r\nsqlfluff:               0.11.0 python:                 3.8.12\r\nimplementation:        cpython dialect:             snowflake\r\nverbosity:                   1 templater:               jinja\r\n\r\n==== readout ====\r\n\r\n=== [ path: test.template.sql ] ===\r\n\r\n== [test.template.sql] PASS                                                                                                                          \r\n==== summary ====\r\nviolations:        0 status:         PASS\r\nAll Finished \ud83d\udcdc \ud83c\udf89!\r\n\r\n```\r\n\r\n\n\n### Expected Behaviour\n\nMy expectations is that the linter and fix should pass.\n\n### Observed Behaviour\n\nRight now lint and fix throws exception (see \"What Happened\" section)\n\n### How to reproduce\n\nMentioned above.\n\n### Dialect\n\nsnowflake\n\n### Version\n\nsqlfluff, version 0.11.0\n\n### Configuration\n\n[sqlfluff]\r\nverbose = 1\r\ndialect = snowflake\r\ntemplater = jinja\r\nexclude_rules = L027,L031,L032,L036,L044,L046,L034,L050\r\noutput_line_length = 121\r\nsql_file_exts=.sql\r\n\r\n[sqlfluff:rules]\r\ntab_space_size = 4\r\nmax_line_length = 250\r\nindent_unit = space\r\ncomma_style = trailing\r\nallow_scalar = True\r\nsingle_table_references = consistent\r\nunquoted_identifiers_policy = aliases\r\n\r\n[sqlfluff:rules:L042]\r\nforbid_subquery_in = both\r\n\r\n[sqlfluff:rules:L010]  # Keywords\r\ncapitalisation_policy = upper\r\n\r\n[sqlfluff:rules:L014]\r\nextended_capitalisation_policy = lower\r\n\r\n[sqlfluff:rules:L030]  # function names\r\nextended_capitalisation_policy = upper\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": "", "created_at": "2022-03-12T21:48:15Z", "version": "0.10", "FAIL_TO_PASS": ["test/core/templaters/base_test.py::test__templated_file_get_line_pos_of_char_pos[01234\\n6789{{foo}}fo\\nbarss-01234\\n6789x\\nfo\\nbarfss-file_slices0-0-1-1]", "test/core/templaters/base_test.py::test__templated_file_get_line_pos_of_char_pos[01234\\n6789{{foo}}fo\\nbarss-01234\\n6789x\\nfo\\nbarfss-file_slices1-20-3-1]", "test/core/templaters/base_test.py::test__templated_file_get_line_pos_of_char_pos[01234\\n6789{{foo}}fo\\nbarss-01234\\n6789x\\nfo\\nbarfss-file_slices2-24-3-5]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice0-out_slice0-True-file_slices0-raw_slices0]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice1-out_slice1-True-file_slices1-raw_slices1]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice2-out_slice2-True-file_slices2-raw_slices2]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice3-out_slice3-False-file_slices3-raw_slices3]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice4-out_slice4-False-file_slices4-raw_slices4]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice5-out_slice5-True-file_slices5-raw_slices5]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice6-out_slice6-True-file_slices6-raw_slices6]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice7-out_slice7-True-file_slices7-raw_slices7]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice8-out_slice8-True-file_slices8-raw_slices8]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice9-out_slice9-True-file_slices9-raw_slices9]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice10-out_slice10-True-file_slices10-raw_slices10]", "test/core/templaters/base_test.py::test__templated_file_templated_slice_to_source_slice[in_slice11-out_slice11-False-file_slices11-raw_slices11]", "test/core/templaters/base_test.py::test__templated_file_source_only_slices", "test/core/templaters/jinja_test.py::test_templater_set_block_handling", "test/core/templaters/jinja_test.py::test__templater_jinja_slice_file[{%"], "PASS_TO_PASS": ["test/core/templaters/base_test.py::test__indices_of_newlines[-positions0]", "test/core/templaters/base_test.py::test__indices_of_newlines[foo-positions1]", "test/core/templaters/base_test.py::test__indices_of_newlines[foo\\nbar-positions2]", "test/core/templaters/base_test.py::test__indices_of_newlines[\\nfoo\\n\\nbar\\nfoo\\n\\nbar\\n-positions3]", "test/core/templaters/base_test.py::test__templater_raw", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[100-True-file_slices0-10-11]", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[13-True-file_slices1-0-3]", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[28-True-file_slices2-2-5]", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[12-True-file_slices3-1-3]", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[20-True-file_slices4-2-3]", "test/core/templaters/base_test.py::test__templated_file_find_slice_indices_of_templated_pos[13-False-file_slices5-0-1]", "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_error_variable", "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-True-False]", "test/core/templaters/jinja_test.py::test__templater_full[jinja_c_dbt/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_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"], "environment_setup_commit": "3d52e8270d82aeccf4c516d059a80a6947919aea"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1213", "base_commit": "40ba4bd5c8b91754aa73e638ed984ab9657847cd", "patch": "diff --git a/pvlib/scaling.py b/pvlib/scaling.py\n--- a/pvlib/scaling.py\n+++ b/pvlib/scaling.py\n@@ -13,8 +13,8 @@\n def wvm(clearsky_index, positions, cloud_speed, dt=None):\n     \"\"\"\n     Compute spatial aggregation time series smoothing on clear sky index based\n-    on the Wavelet Variability model of Lave et al [1-2]. Implementation is\n-    basically a port of the Matlab version of the code [3].\n+    on the Wavelet Variability model of Lave et al. [1]_, [2]_. Implementation\n+    is basically a port of the Matlab version of the code [3]_.\n \n     Parameters\n     ----------\n@@ -48,16 +48,16 @@ def wvm(clearsky_index, positions, cloud_speed, dt=None):\n \n     References\n     ----------\n-    [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n-    Model (WVM) for Solar PV Power Plants. IEEE Transactions on Sustainable\n-    Energy, vol. 4, no. 2, pp. 501-509, 2013.\n+    .. [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n+       Model (WVM) for Solar PV Power Plants. IEEE Transactions on Sustainable\n+       Energy, vol. 4, no. 2, pp. 501-509, 2013.\n \n-    [2] M. Lave and J. Kleissl. Cloud speed impact on solar variability\n-    scaling - Application to the wavelet variability model. Solar Energy,\n-    vol. 91, pp. 11-21, 2013.\n+    .. [2] M. Lave and J. Kleissl. Cloud speed impact on solar variability\n+       scaling - Application to the wavelet variability model. Solar Energy,\n+       vol. 91, pp. 11-21, 2013.\n \n-    [3] Wavelet Variability Model - Matlab Code:\n-    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    .. [3] Wavelet Variability Model - Matlab Code:\n+       https://github.com/sandialabs/wvm\n     \"\"\"\n \n     # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n@@ -128,13 +128,13 @@ def latlon_to_xy(coordinates):\n \n     References\n     ----------\n-    [1] H. Moritz. Geodetic Reference System 1980, Journal of Geodesy, vol. 74,\n-    no. 1, pp 128\u2013133, 2000.\n+    .. [1] H. Moritz. Geodetic Reference System 1980, Journal of Geodesy, vol.\n+       74, no. 1, pp 128\u2013133, 2000.\n \n-    [2] https://pypi.org/project/pyproj/\n+    .. [2] https://pypi.org/project/pyproj/\n \n-    [3] Wavelet Variability Model - Matlab Code:\n-    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    .. [3] Wavelet Variability Model - Matlab Code:\n+       https://github.com/sandialabs/wvm\n     \"\"\"\n \n     # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n@@ -159,7 +159,12 @@ def latlon_to_xy(coordinates):\n \n def _compute_wavelet(clearsky_index, dt=None):\n     \"\"\"\n-    Compute the wavelet transform on the input clear_sky time series.\n+    Compute the wavelet transform on the input clear_sky time series. Uses a\n+    top hat wavelet [-1,1,1,-1] shape, based on the difference of successive\n+    centered moving averages. Smallest scale (filter size of 2) is a degenerate\n+    case that resembles a Haar wavelet. Returns one level of approximation\n+    coefficient (CAn) and n levels of detail coefficients (CD1, CD2, ...,\n+    CDn-1, CDn).\n \n     Parameters\n     ----------\n@@ -174,19 +179,20 @@ def _compute_wavelet(clearsky_index, dt=None):\n     Returns\n     -------\n     wavelet: numeric\n-        The individual wavelets for the time series\n+        The individual wavelets for the time series. Format follows increasing\n+        scale (decreasing frequency): [CD1, CD2, ..., CDn, CAn]\n \n     tmscales: numeric\n         The timescales associated with the wavelets in seconds [s]\n \n     References\n     ----------\n-    [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n-    Model (WVM) for Solar PV Power Plants. IEEE Transactions on Sustainable\n-    Energy, vol. 4, no. 2, pp. 501-509, 2013.\n+    .. [1] M. Lave, J. Kleissl and J.S. Stein. A Wavelet-Based Variability\n+       Model (WVM) for Solar PV Power Plants. IEEE Transactions on\n+       Sustainable Energy, vol. 4, no. 2, pp. 501-509, 2013.\n \n-    [3] Wavelet Variability Model - Matlab Code:\n-    https://pvpmc.sandia.gov/applications/wavelet-variability-model/\n+    .. [2] Wavelet Variability Model - Matlab Code:\n+       https://github.com/sandialabs/wvm\n     \"\"\"\n \n     # Added by Joe Ranalli (@jranalli), Penn State Hazleton, 2019\n@@ -209,31 +215,37 @@ def _compute_wavelet(clearsky_index, dt=None):\n \n     # Compute wavelet time scales\n     min_tmscale = np.ceil(np.log(dt)/np.log(2))  # Minimum wavelet timescale\n-    max_tmscale = int(12 - min_tmscale)  # maximum wavelet timescale\n+    max_tmscale = int(13 - min_tmscale)  # maximum wavelet timescale\n \n     tmscales = np.zeros(max_tmscale)\n     csi_mean = np.zeros([max_tmscale, len(cs_long)])\n+    # Skip averaging for the 0th scale\n+    csi_mean[0, :] = cs_long.values.flatten()\n+    tmscales[0] = 1\n     # Loop for all time scales we will consider\n-    for i in np.arange(0, max_tmscale):\n-        j = i+1\n-        tmscales[i] = 2**j * dt  # Wavelet integration time scale\n-        intvlen = 2**j  # Wavelet integration time series interval\n+    for i in np.arange(1, max_tmscale):\n+        tmscales[i] = 2**i * dt  # Wavelet integration time scale\n+        intvlen = 2**i  # Wavelet integration time series interval\n         # Rolling average, retains only lower frequencies than interval\n+        # Produces slightly different end effects than the MATLAB version\n         df = cs_long.rolling(window=intvlen, center=True, min_periods=1).mean()\n         # Fill nan's in both directions\n         df = df.fillna(method='bfill').fillna(method='ffill')\n         # Pop values back out of the dataframe and store\n         csi_mean[i, :] = df.values.flatten()\n+        # Shift to account for different indexing in MATLAB moving average\n+        csi_mean[i, :] = np.roll(csi_mean[i, :], -1)\n+        csi_mean[i, -1] = csi_mean[i, -2]\n \n-    # Calculate the wavelets by isolating the rolling mean frequency ranges\n+    # Calculate detail coefficients by difference between successive averages\n     wavelet_long = np.zeros(csi_mean.shape)\n     for i in np.arange(0, max_tmscale-1):\n         wavelet_long[i, :] = csi_mean[i, :] - csi_mean[i+1, :]\n-    wavelet_long[max_tmscale-1, :] = csi_mean[max_tmscale-1, :]  # Lowest freq\n+    wavelet_long[-1, :] = csi_mean[-1, :]  # Lowest freq (CAn)\n \n     # Clip off the padding and just return the original time window\n     wavelet = np.zeros([max_tmscale, len(vals)])\n     for i in np.arange(0, max_tmscale):\n-        wavelet[i, :] = wavelet_long[i, len(vals)+1: 2*len(vals)+1]\n+        wavelet[i, :] = wavelet_long[i, len(vals): 2*len(vals)]\n \n     return wavelet, tmscales\n", "test_patch": "diff --git a/pvlib/tests/test_scaling.py b/pvlib/tests/test_scaling.py\n--- a/pvlib/tests/test_scaling.py\n+++ b/pvlib/tests/test_scaling.py\n@@ -48,21 +48,24 @@ def positions():\n @pytest.fixture\n def expect_tmscale():\n     # Expected timescales for dt = 1\n-    return [2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096]\n+    return [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096]\n \n \n @pytest.fixture\n def expect_wavelet():\n     # Expected wavelet for indices 5000:5004 for clear_sky_index above (Matlab)\n-    return np.array([[-0.025, 0.05, 0., -0.05, 0.025],\n-                     [0.025, 0., 0., 0., -0.025],\n-                     [0., 0., 0., 0., 0.]])\n+    e = np.zeros([13, 5])\n+    e[0, :] = np.array([0, -0.05, 0.1, -0.05, 0])\n+    e[1, :] = np.array([-0.025, 0.05, 0., -0.05, 0.025])\n+    e[2, :] = np.array([0.025, 0., 0., 0., -0.025])\n+    e[-1, :] = np.array([1, 1, 1, 1, 1])\n+    return e\n \n \n @pytest.fixture\n def expect_cs_smooth():\n     # Expected smoothed clear sky index for indices 5000:5004 (Matlab)\n-    return np.array([1., 1.0289, 1., 0.9711, 1.])\n+    return np.array([1., 1., 1.05774, 0.94226, 1.])\n \n \n def test_latlon_to_xy_zero():\n@@ -94,7 +97,7 @@ def test_compute_wavelet_series(clear_sky_index, time,\n     csi_series = pd.Series(clear_sky_index, index=time)\n     wavelet, tmscale = scaling._compute_wavelet(csi_series)\n     assert_almost_equal(tmscale, expect_tmscale)\n-    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+    assert_almost_equal(wavelet[:, 5000:5005], expect_wavelet)\n \n \n def test_compute_wavelet_series_numindex(clear_sky_index, time,\n@@ -103,14 +106,14 @@ def test_compute_wavelet_series_numindex(clear_sky_index, time,\n     csi_series = pd.Series(clear_sky_index, index=dtindex)\n     wavelet, tmscale = scaling._compute_wavelet(csi_series)\n     assert_almost_equal(tmscale, expect_tmscale)\n-    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+    assert_almost_equal(wavelet[:, 5000:5005], expect_wavelet)\n \n \n def test_compute_wavelet_array(clear_sky_index,\n                                expect_tmscale, expect_wavelet):\n     wavelet, tmscale = scaling._compute_wavelet(clear_sky_index, dt)\n     assert_almost_equal(tmscale, expect_tmscale)\n-    assert_almost_equal(wavelet[0:3, 5000:5005], expect_wavelet)\n+    assert_almost_equal(wavelet[:, 5000:5005], expect_wavelet)\n \n \n def test_compute_wavelet_array_invalid(clear_sky_index):\n@@ -118,6 +121,14 @@ def test_compute_wavelet_array_invalid(clear_sky_index):\n         scaling._compute_wavelet(clear_sky_index)\n \n \n+def test_compute_wavelet_dwttheory(clear_sky_index, time,\n+                                   expect_tmscale, expect_wavelet):\n+    # Confirm detail coeffs sum to original signal\n+    csi_series = pd.Series(clear_sky_index, index=time)\n+    wavelet, tmscale = scaling._compute_wavelet(csi_series)\n+    assert_almost_equal(np.sum(wavelet, 0), csi_series)\n+\n+\n def test_wvm_series(clear_sky_index, time, positions, expect_cs_smooth):\n     csi_series = pd.Series(clear_sky_index, index=time)\n     cs_sm, _, _ = scaling.wvm(csi_series, positions, cloud_speed)\n", "problem_statement": "Bug within scaling.py wavelet calculation methodology\n**Describe the bug**\r\nMathematical error within the wavelet computation for the scaling.py WVM implementation. Error arises from the methodology, as opposed to just a software bug. \r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n```\r\nimport numpy as np\r\nfrom pvlib import scaling\r\ncs = np.random.rand(2**14)\r\nw, ts = scaling._compute_wavelet(cs,1)\r\nprint(np.all( (sum(w)-cs) < 1e-8 ))  # Returns False, expect True\r\n```\r\n\r\n**Expected behavior**\r\nFor a discrete wavelet transform (DWT) the sum of all wavelet modes should equate to the original data. \r\n\r\n**Versions:**\r\n - ``pvlib.__version__``: 0.7.2\r\n - ``pandas.__version__``: 1.2.3\r\n - python: 3.8.8\r\n\r\n**Additional context**\r\nThis bug is also present in the [PV_LIB](https://pvpmc.sandia.gov/applications/wavelet-variability-model/) Matlab version that was used as the basis for this code (I did reach out to them using the PVLIB MATLAB email form, but don't know who actually wrote that code). Essentially, the existing code throws away the highest level of Detail Coefficient in the transform and keeps an extra level of Approximation coefficient. The impact on the calculation is small, but leads to an incorrect DWT and reconstruction. I have a fix that makes the code pass the theoretical test about the DWT proposed under 'To Reproduce' but there may be some question as to whether this should be corrected or left alone to match the MATLAB code it was based on. \r\n\n", "hints_text": "@jranalli thanks for finding and reporting this. Can I ask how you contacted PVLIB MATLAB? Because I maintain that repository and I didn't see the email, so we need to fix something on our end with communications.\n@cwhanse Now that I look again, I think I used the wrong form. It was just a general Questions and Comments link for the PV Performance Modeling Collaborative at the bottom of the page. I didn't see any contact point for the PV_LIB MATLAB library and I also didn't know about the github repo for it, but now I do! \r\n\r\nI do have a fix for the MATLAB code as well, but I don't see that part of the library on github. If you'd like me to open an issue on that repository as well, I'd be happy to do so, but if there's some other pathway or contact point since that's kind of listed as a separate package of the code, please let me know.\r\n\r\nEither way, do you think it's appropriate to fix this, or does there need to be a conversation with the original author of that MATLAB code? If everything is fine to go ahead with it here, I can just put together my fix as a pull request for review.\r\n\r\n\nAnd did my own looking: pvl_WVM is in it's own Matlab archive, separate from PVLIB for Matlab. The WVM code is only available as a download from pvpmc.sandia.gov, whereas PVLIB for Matlab is on [github](https://github.com/sandialabs/MATLAB_PV_LIB).\r\n\r\nI've sent the bug report to Matt Lave, the originator of the WVM algorithm and code. We'll likely welcome the bug fix but I'd like to hear Matt's view first.\nOK sounds good. If he or you want to connect for more detail on the issue, you can get contact info for me at my [Faculty Page](http://personal.psu.edu/jar339/about.html).\nfor the record: bug is confirmed via separate communication with the WVM algorithm author.", "created_at": "2021-04-23T01:38:03Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/test_scaling.py::test_compute_wavelet_series", "pvlib/tests/test_scaling.py::test_compute_wavelet_series_numindex", "pvlib/tests/test_scaling.py::test_compute_wavelet_array", "pvlib/tests/test_scaling.py::test_compute_wavelet_dwttheory", "pvlib/tests/test_scaling.py::test_wvm_series", "pvlib/tests/test_scaling.py::test_wvm_array", "pvlib/tests/test_scaling.py::test_wvm_series_xyaslist"], "PASS_TO_PASS": ["pvlib/tests/test_scaling.py::test_latlon_to_xy_zero", "pvlib/tests/test_scaling.py::test_latlon_to_xy_single", "pvlib/tests/test_scaling.py::test_latlon_to_xy_array", "pvlib/tests/test_scaling.py::test_latlon_to_xy_list", "pvlib/tests/test_scaling.py::test_compute_wavelet_array_invalid", "pvlib/tests/test_scaling.py::test_wvm_invalid"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-3109", "base_commit": "38c31c99f6be6c2ffcc9e4132387edb7af8c5d06", "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@@ -1,6 +1,7 @@\n \"\"\"Contains the CLI.\"\"\"\n \n from itertools import chain\n+import os\n import sys\n import json\n import logging\n@@ -29,9 +30,10 @@\n     colorize,\n     format_dialect_warning,\n     format_dialects,\n-    CallbackFormatter,\n+    OutputStreamFormatter,\n )\n from sqlfluff.cli.helpers import cli_table, get_package_version\n+from sqlfluff.cli.outputstream import make_output_stream, OutputStream\n \n # Import from sqlfluff core.\n from sqlfluff.core import (\n@@ -332,28 +334,9 @@ def get_config(\n         sys.exit(66)\n \n \n-def _callback_handler(cfg: FluffConfig) -> Callable:\n-    \"\"\"Returns function which will be bound as a callback for printing passed message.\n-\n-    Called in `get_linter_and_formatter`.\n-    \"\"\"\n-\n-    def _echo_with_tqdm_lock(message: str) -> None:\n-        \"\"\"Makes sure that message printing (echoing) will be not in conflict with tqdm.\n-\n-        It may happen that progressbar conflicts with extra printing. Nothing very\n-        serious happens then, except that there is printed (not removed) progressbar\n-        line. The `external_write_mode` allows to disable tqdm for writing time.\n-        \"\"\"\n-        with tqdm.external_write_mode():\n-            click.echo(message=message, color=cfg.get(\"color\"))\n-\n-    return _echo_with_tqdm_lock\n-\n-\n def get_linter_and_formatter(\n-    cfg: FluffConfig, silent: bool = False\n-) -> Tuple[Linter, CallbackFormatter]:\n+    cfg: FluffConfig, output_stream: Optional[OutputStream] = None\n+) -> Tuple[Linter, OutputStreamFormatter]:\n     \"\"\"Get a linter object given a config.\"\"\"\n     try:\n         # We're just making sure it exists at this stage.\n@@ -364,20 +347,12 @@ def get_linter_and_formatter(\n     except KeyError:  # pragma: no cover\n         click.echo(f\"Error: Unknown dialect '{cfg.get('dialect')}'\")\n         sys.exit(66)\n-\n-    if not silent:\n-        # Instantiate the linter and return it (with an output function)\n-        formatter = CallbackFormatter(\n-            callback=_callback_handler(cfg=cfg),\n-            verbosity=cfg.get(\"verbose\"),\n-            output_line_length=cfg.get(\"output_line_length\"),\n-        )\n-        return Linter(config=cfg, formatter=formatter), formatter\n-    else:\n-        # Instantiate the linter and return. NB: No formatter\n-        # in the Linter and a black formatter otherwise.\n-        formatter = CallbackFormatter(callback=lambda m: None, verbosity=0)\n-        return Linter(config=cfg), formatter\n+    formatter = OutputStreamFormatter(\n+        output_stream=output_stream or make_output_stream(cfg),\n+        verbosity=cfg.get(\"verbose\"),\n+        output_line_length=cfg.get(\"output_line_length\"),\n+    )\n+    return Linter(config=cfg, formatter=formatter), formatter\n \n \n @click.group(context_settings={\"help_option_names\": [\"-h\", \"--help\"]})\n@@ -521,7 +496,8 @@ def lint(\n     )\n     non_human_output = (format != FormatType.human.value) or (write_output is not None)\n     file_output = None\n-    lnt, formatter = get_linter_and_formatter(config, silent=non_human_output)\n+    output_stream = make_output_stream(config, format, write_output)\n+    lnt, formatter = get_linter_and_formatter(config, output_stream)\n \n     verbose = config.get(\"verbose\")\n     progress_bar_configuration.disable_progress_bar = disable_progress_bar\n@@ -610,6 +586,7 @@ def lint(\n     if file_output:\n         dump_file_payload(write_output, cast(str, file_output))\n \n+    output_stream.close()\n     if bench:\n         click.echo(\"==== overall timings ====\")\n         click.echo(cli_table([(\"Clock time\", result.total_time)]))\n@@ -733,7 +710,10 @@ def fix(\n         extra_config_path, ignore_local_config, require_dialect=False, **kwargs\n     )\n     fix_even_unparsable = config.get(\"fix_even_unparsable\")\n-    lnt, formatter = get_linter_and_formatter(config, silent=fixing_stdin)\n+    output_stream = make_output_stream(\n+        config, None, os.devnull if fixing_stdin else None\n+    )\n+    lnt, formatter = get_linter_and_formatter(config, output_stream)\n \n     verbose = config.get(\"verbose\")\n     progress_bar_configuration.disable_progress_bar = disable_progress_bar\n@@ -974,7 +954,8 @@ def parse(\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     non_human_output = (format != FormatType.human.value) or (write_output is not None)\n-    lnt, formatter = get_linter_and_formatter(c, silent=non_human_output)\n+    output_stream = make_output_stream(c, format, write_output)\n+    lnt, formatter = get_linter_and_formatter(c, output_stream)\n     verbose = c.get(\"verbose\")\n     recurse = c.get(\"recurse\")\n \n@@ -1020,7 +1001,7 @@ def parse(\n         # iterative print for human readout\n         if format == FormatType.human.value:\n             violations_count = _print_out_violations_and_timing(\n-                bench, code_only, total_time, verbose, parsed_strings\n+                output_stream, bench, code_only, total_time, verbose, parsed_strings\n             )\n         else:\n             parsed_strings_dict = [\n@@ -1072,6 +1053,7 @@ def parse(\n \n \n def _print_out_violations_and_timing(\n+    output_stream: OutputStream,\n     bench: bool,\n     code_only: bool,\n     total_time: float,\n@@ -1086,30 +1068,30 @@ def _print_out_violations_and_timing(\n         timing.add(parsed_string.time_dict)\n \n         if parsed_string.tree:\n-            click.echo(parsed_string.tree.stringify(code_only=code_only))\n+            output_stream.write(parsed_string.tree.stringify(code_only=code_only))\n         else:\n             # TODO: Make this prettier\n-            click.echo(\"...Failed to Parse...\")  # pragma: no cover\n+            output_stream.write(\"...Failed to Parse...\")  # pragma: no cover\n \n         violations_count += len(parsed_string.violations)\n         if parsed_string.violations:\n-            click.echo(\"==== parsing violations ====\")  # pragma: no cover\n+            output_stream.write(\"==== parsing violations ====\")  # pragma: no cover\n         for v in parsed_string.violations:\n-            click.echo(format_violation(v))  # pragma: no cover\n+            output_stream.write(format_violation(v))  # pragma: no cover\n         if parsed_string.violations and parsed_string.config.get(\"dialect\") == \"ansi\":\n-            click.echo(format_dialect_warning())  # pragma: no cover\n+            output_stream.write(format_dialect_warning())  # pragma: no cover\n \n         if verbose >= 2:\n-            click.echo(\"==== timings ====\")\n-            click.echo(cli_table(parsed_string.time_dict.items()))\n+            output_stream.write(\"==== timings ====\")\n+            output_stream.write(cli_table(parsed_string.time_dict.items()))\n \n     if verbose >= 2 or bench:\n-        click.echo(\"==== overall timings ====\")\n-        click.echo(cli_table([(\"Clock time\", total_time)]))\n+        output_stream.write(\"==== overall timings ====\")\n+        output_stream.write(cli_table([(\"Clock time\", total_time)]))\n         timing_summary = timing.summary()\n         for step in timing_summary:\n-            click.echo(f\"=== {step} ===\")\n-            click.echo(cli_table(timing_summary[step].items()))\n+            output_stream.write(f\"=== {step} ===\")\n+            output_stream.write(cli_table(timing_summary[step].items()))\n \n     return violations_count\n \ndiff --git a/src/sqlfluff/cli/formatters.py b/src/sqlfluff/cli/formatters.py\n--- a/src/sqlfluff/cli/formatters.py\n+++ b/src/sqlfluff/cli/formatters.py\n@@ -2,7 +2,7 @@\n \n \n from io import StringIO\n-from typing import Callable, List, Union\n+from typing import List, Union\n \n from sqlfluff.cli.helpers import (\n     colorize,\n@@ -12,6 +12,7 @@\n     get_python_implementation,\n     pad_line,\n )\n+from sqlfluff.cli.outputstream import OutputStream\n from sqlfluff.core import SQLBaseError, FluffConfig, Linter\n from sqlfluff.core.enums import Color\n from sqlfluff.core.linter import LintedFile\n@@ -199,8 +200,8 @@ def format_dialect_warning():  # pragma: no cover\n     )\n \n \n-class CallbackFormatter:\n-    \"\"\"Formatter which uses a callback to output information.\n+class OutputStreamFormatter:\n+    \"\"\"Formatter which writes to an OutputStream.\n \n     On instantiation, this formatter accepts a function to\n     dispatch messages. Each public method accepts an object\n@@ -212,23 +213,20 @@ class CallbackFormatter:\n \n \n     Args:\n-        callback (:obj:`callable`): A callable which can be\n-            be called with a string to be output.\n-        verbosity (:obj:`int`): An integer specifying how\n-            verbose the output should be.\n-        filter_empty (:obj:`bool`): If True, empty messages\n-            will not be dispatched.\n-\n+        output_stream: Output is sent here\n+        verbosity: Specifies how verbose output should be\n+        filter_empty: If True, empty messages will not be dispatched\n+        output_line_length: Maximum line length\n     \"\"\"\n \n     def __init__(\n         self,\n-        callback: Callable,\n+        output_stream: OutputStream,\n         verbosity: int = 0,\n         filter_empty: bool = True,\n         output_line_length: int = 80,\n     ):\n-        self._callback = callback\n+        self._output_stream = output_stream\n         self._verbosity = verbosity\n         self._filter_empty = filter_empty\n         self.output_line_length = output_line_length\n@@ -240,7 +238,7 @@ def _dispatch(self, s: str) -> None:\n         \"\"\"\n         # The strip here is to filter out any empty messages\n         if (not self._filter_empty) or s.strip(\" \\n\\t\"):\n-            self._callback(s)\n+            self._output_stream.write(s)\n \n     def _format_config(self, linter: Linter) -> str:\n         \"\"\"Format the config of a `Linter`.\"\"\"\ndiff --git a/src/sqlfluff/cli/outputstream.py b/src/sqlfluff/cli/outputstream.py\nnew file mode 100644\n--- /dev/null\n+++ b/src/sqlfluff/cli/outputstream.py\n@@ -0,0 +1,76 @@\n+\"\"\"Classes for managing linter output, used with OutputStreamFormatter.\"\"\"\n+import abc\n+import os\n+from typing import Any, Optional\n+\n+import click\n+from tqdm import tqdm\n+\n+from sqlfluff.core import FluffConfig\n+from sqlfluff.core.enums import FormatType\n+\n+\n+class OutputStream(abc.ABC):\n+    \"\"\"Base class for linter output stream.\"\"\"\n+\n+    def __init__(self, config: FluffConfig, context: Any = None):\n+        self.config = config\n+\n+    def write(self, message: str) -> None:\n+        \"\"\"Write message to output.\"\"\"\n+        raise NotImplementedError  # pragma: no cover\n+\n+    def close(self):\n+        \"\"\"Close output stream.\"\"\"\n+        pass\n+\n+\n+class TqdmOutput(OutputStream):\n+    \"\"\"Outputs to stdout, coordinates to avoid conflict with tqdm.\n+\n+    It may happen that progressbar conflicts with extra printing. Nothing very\n+    serious happens then, except that there is printed (not removed) progressbar\n+    line. The `external_write_mode` allows to disable tqdm for writing time.\n+    \"\"\"\n+\n+    def __init__(self, config: FluffConfig):\n+        super().__init__(config)\n+\n+    def write(self, message: str) -> None:\n+        \"\"\"Write message to stdout.\"\"\"\n+        with tqdm.external_write_mode():\n+            click.echo(message=message, color=self.config.get(\"color\"))\n+\n+\n+class FileOutput(OutputStream):\n+    \"\"\"Outputs to a specified file.\"\"\"\n+\n+    def __init__(self, config: FluffConfig, output_path: str):\n+        super().__init__(config)\n+        self.file = open(output_path, \"w\")\n+\n+    def write(self, message: str) -> None:\n+        \"\"\"Write message to output_path.\"\"\"\n+        print(message, file=self.file)\n+\n+    def close(self):\n+        \"\"\"Close output file.\"\"\"\n+        self.file.close()\n+\n+\n+def make_output_stream(\n+    config: FluffConfig,\n+    format: Optional[str] = None,\n+    output_path: Optional[str] = None,\n+) -> OutputStream:\n+    \"\"\"Create and return appropriate OutputStream instance.\"\"\"\n+    if format is None or format == FormatType.human.value:\n+        if not output_path:\n+            # Human-format output to stdout.\n+            return TqdmOutput(config)\n+        else:\n+            # Human-format output to a file.\n+            return FileOutput(config, output_path)\n+    else:\n+        # Discard human output as not required\n+        return FileOutput(config, os.devnull)\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@@ -968,11 +968,12 @@ def test__cli__command_fail_nice_not_found(command):\n \n \n @pytest.mark.parametrize(\n-    \"serialize\", [\"yaml\", \"json\", \"github-annotation\", \"github-annotation-native\"]\n+    \"serialize\",\n+    [\"human\", \"yaml\", \"json\", \"github-annotation\", \"github-annotation-native\"],\n )\n @pytest.mark.parametrize(\"write_file\", [None, \"outfile\"])\n def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_path):\n-    \"\"\"Check the general format of JSON output for multiple files.\n+    \"\"\"Test the output output formats for multiple files.\n \n     This tests runs both stdout checking and file checking.\n     \"\"\"\n@@ -987,9 +988,11 @@ def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_\n     )\n \n     if write_file:\n-        target_file = os.path.join(\n-            tmp_path, write_file + (\".yaml\" if serialize == \"yaml\" else \".json\")\n-        )\n+        ext = {\n+            \"human\": \".txt\",\n+            \"yaml\": \".yaml\",\n+        }\n+        target_file = os.path.join(tmp_path, write_file + ext.get(serialize, \".json\"))\n         cmd_args += (\"--write-output\", target_file)\n \n     # note the file is in here twice. two files = two payloads.\n@@ -1004,7 +1007,9 @@ def test__cli__command_lint_serialize_multiple_files(serialize, write_file, tmp_\n     else:\n         result_payload = result.output\n \n-    if serialize == \"json\":\n+    if serialize == \"human\":\n+        assert len(result_payload.split(\"\\n\")) == 29 if write_file else 30\n+    elif serialize == \"json\":\n         result = json.loads(result_payload)\n         assert len(result) == 2\n     elif serialize == \"yaml\":\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@@ -11,7 +11,8 @@\n from sqlfluff.core.dialects import load_raw_dialect\n from sqlfluff.core.linter import runner\n from sqlfluff.core.errors import SQLLexError, SQLBaseError, SQLLintError, SQLParseError\n-from sqlfluff.cli.formatters import CallbackFormatter\n+from sqlfluff.cli.formatters import OutputStreamFormatter\n+from sqlfluff.cli.outputstream import make_output_stream\n from sqlfluff.core.linter import LintingResult, NoQaDirective\n import sqlfluff.core.linter as linter\n from sqlfluff.core.parser import GreedyUntil, Ref\n@@ -256,8 +257,10 @@ def imap_unordered(self, *args, **kwargs):\n \n         monkeypatch.setattr(runner.MultiProcessRunner, \"_create_pool\", _create_pool)\n \n+    config = FluffConfig(overrides={\"dialect\": \"ansi\"})\n+    output_stream = make_output_stream(config, None, os.devnull)\n     lntr = Linter(\n-        formatter=CallbackFormatter(callback=lambda m: None, verbosity=0),\n+        formatter=OutputStreamFormatter(output_stream, verbosity=0),\n         dialect=\"ansi\",\n     )\n     result = lntr.lint_paths(\n", "problem_statement": "Write-output human format does not produce result\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 running SQLFluff using the following statement:\r\n`python -m sqlfluff lint --write-output test.txt --config=config/sql-lint.cfg`\r\nno result was produced.\n\n### Expected Behaviour\n\nI expect a file to appear, in this case called test,txt, containing all violations found.\n\n### Observed Behaviour\n\nLooking through the code I saw human was the default format so expected adding --format=human would not make a difference. To be sure, I also ran the statement using the flag and it still produced nothing.\r\n\r\nTo make sure it was just the human format which was having problems, I also executed the statement using --format=json,yaml,github-annotations, all of which did produce the expected result which leads me to believe there is something wrong with the human format.\n\n### How to reproduce\n\nI imagine simply executing `sqlfluff lint --write-output test.txt example.sql`\n\n### Dialect\n\nT-SQL\n\n### Version\n\n0.11.2\n\n### Configuration\n\n[sqlfluff]\r\ndialect = tsql\r\nexclude_rules = L014,\r\n                L016,\r\n                L031,\r\n                L035,\r\n                L059\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 line in cli.py seems largely to blame -- it somewhat conflates output _format_ with writing to a file or not.\r\n```\r\nnon_human_output = (format != FormatType.human.value) or (write_output is not None)\r\n```\r\n\r\nIt will require some care to fix this. Simply removing `or (write_output is not None)` didn't seem to fix it.\r\n\r\nAs a workaround until this is fixed, you may be able to use output redirection, e.g.\r\n```\r\npython -m sqlfluff lint --config=config/sql-lint.cfg > test.txt\r\n```\r\n\r\n\nYour workaround does work for me, thank you. Seeing as this solution is only a workaround I imagine closing the ticket is not preferable.", "created_at": "2022-04-17T20:42:13Z", "version": "0.12", "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_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_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_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_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_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-65-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_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_pass_on_correct_encoding_argument", "test/cli/commands_test.py::test_cli_fail_on_wrong_encoding_argument", "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_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/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__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__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": "8f6fd1d8a8d69b2c463fbcf5bd1131c47f12ad88"}, {"repo": "sqlfluff/sqlfluff", "instance_id": "sqlfluff__sqlfluff-3662", "base_commit": "f9a3fe8f639d279226f16bdc51326dfa5c142c3e", "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@@ -486,8 +486,12 @@ def dump_file_payload(filename: Optional[str], payload: str):\n     \"-p\",\n     \"--processes\",\n     type=int,\n-    default=1,\n-    help=\"The number of parallel processes to run.\",\n+    default=None,\n+    help=(\n+        \"The number of parallel processes to run. Positive numbers work as \"\n+        \"expected. Zero and negative numbers will work as number_of_cpus - \"\n+        \"number. e.g  -1 means all cpus except one. 0 means all cpus.\"\n+    ),\n )\n @click.option(\n     \"--disable_progress_bar\",\n@@ -497,7 +501,6 @@ def dump_file_payload(filename: Optional[str], payload: str):\n @click.argument(\"paths\", nargs=-1, type=click.Path(allow_dash=True))\n def lint(\n     paths: Tuple[str],\n-    processes: int,\n     format: str,\n     write_output: Optional[str],\n     annotation_level: str,\n@@ -505,6 +508,7 @@ def lint(\n     disregard_sqlfluffignores: bool,\n     logger: Optional[logging.Logger] = None,\n     bench: bool = False,\n+    processes: Optional[int] = None,\n     disable_progress_bar: Optional[bool] = False,\n     extra_config_path: Optional[str] = None,\n     ignore_local_config: bool = False,\n@@ -675,8 +679,12 @@ def do_fixes(lnt, result, formatter=None, **kwargs):\n     \"-p\",\n     \"--processes\",\n     type=int,\n-    default=1,\n-    help=\"The number of parallel processes to run.\",\n+    default=None,\n+    help=(\n+        \"The number of parallel processes to run. Positive numbers work as \"\n+        \"expected. Zero and negative numbers will work as number_of_cpus - \"\n+        \"number. e.g  -1 means all cpus except one. 0 means all cpus.\"\n+    ),\n )\n @click.option(\n     \"--disable_progress_bar\",\n@@ -705,10 +713,10 @@ def do_fixes(lnt, result, formatter=None, **kwargs):\n def fix(\n     force: bool,\n     paths: Tuple[str],\n-    processes: int,\n     bench: bool = False,\n     fixed_suffix: str = \"\",\n     logger: Optional[logging.Logger] = None,\n+    processes: Optional[int] = None,\n     disable_progress_bar: Optional[bool] = False,\n     extra_config_path: Optional[str] = None,\n     ignore_local_config: bool = False,\ndiff --git a/src/sqlfluff/cli/formatters.py b/src/sqlfluff/cli/formatters.py\n--- a/src/sqlfluff/cli/formatters.py\n+++ b/src/sqlfluff/cli/formatters.py\n@@ -181,6 +181,14 @@ def dispatch_compilation_header(self, templater, message):\n             f\"=== [{self.colorize(templater, Color.lightgrey)}] {message}\"\n         )  # pragma: no cover\n \n+    def dispatch_processing_header(self, processes: int) -> None:\n+        \"\"\"Dispatch the header displayed before linting.\"\"\"\n+        if self._verbosity > 0:\n+            self._dispatch(  # pragma: no cover\n+                f\"{self.colorize('effective configured processes: ', Color.lightgrey)} \"\n+                f\"{processes}\"\n+            )\n+\n     def dispatch_dialect_warning(self, dialect) -> None:\n         \"\"\"Dispatch a warning for dialects.\"\"\"\n         self._dispatch(self.format_dialect_warning(dialect))  # pragma: no cover\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@@ -1066,7 +1066,7 @@ def lint_path(\n         fix: bool = False,\n         ignore_non_existent_files: bool = False,\n         ignore_files: bool = True,\n-        processes: int = 1,\n+        processes: Optional[int] = None,\n     ) -> LintedDir:\n         \"\"\"Lint a path.\"\"\"\n         linted_path = LintedDir(path)\n@@ -1080,16 +1080,22 @@ def lint_path(\n             )\n         )\n \n+        if processes is None:\n+            processes = self.config.get(\"processes\", default=1)\n+\n         # to avoid circular import\n         from sqlfluff.core.linter.runner import get_runner\n \n-        runner = get_runner(\n+        runner, effective_processes = get_runner(\n             self,\n             self.config,\n             processes=processes,\n             allow_process_parallelism=self.allow_process_parallelism,\n         )\n \n+        if self.formatter and effective_processes != 1:\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         progress_bar_files = tqdm(\n@@ -1124,7 +1130,7 @@ def lint_paths(\n         fix: bool = False,\n         ignore_non_existent_files: bool = False,\n         ignore_files: bool = True,\n-        processes: int = 1,\n+        processes: Optional[int] = None,\n     ) -> LintingResult:\n         \"\"\"Lint an iterable of paths.\"\"\"\n         paths_count = len(paths)\ndiff --git a/src/sqlfluff/core/linter/runner.py b/src/sqlfluff/core/linter/runner.py\n--- a/src/sqlfluff/core/linter/runner.py\n+++ b/src/sqlfluff/core/linter/runner.py\n@@ -10,6 +10,7 @@\n import bdb\n import functools\n import logging\n+import multiprocessing\n import multiprocessing.dummy\n import signal\n import sys\n@@ -227,15 +228,29 @@ def get_runner(\n     config: FluffConfig,\n     processes: int,\n     allow_process_parallelism: bool = True,\n-) -> BaseRunner:\n-    \"\"\"Generate a runner instance based on parallel and system configuration.\"\"\"\n+) -> Tuple[BaseRunner, int]:\n+    \"\"\"Generate a runner instance based on parallel and system configuration.\n+\n+    The processes argument can be positive or negative.\n+    - If positive, the integer is interpreted as the number of processes.\n+    - If negative or zero, the integer is interpreted as number_of_cpus - processes.\n+\n+    e.g.\n+    -1 = all cpus but one.\n+    0 = all cpus\n+    1 = 1 cpu\n+\n+    \"\"\"\n+    if processes <= 0:\n+        processes = max(multiprocessing.cpu_count() + processes, 1)\n+\n     if processes > 1:\n         # Process parallelism isn't really supported during testing\n         # so this flag allows us to fall back to a threaded runner\n         # in those cases.\n         if allow_process_parallelism:\n-            return MultiProcessRunner(linter, config, processes=processes)\n+            return MultiProcessRunner(linter, config, processes=processes), processes\n         else:\n-            return MultiThreadRunner(linter, config, processes=processes)\n+            return MultiThreadRunner(linter, config, processes=processes), processes\n     else:\n-        return SequentialRunner(linter, config)\n+        return SequentialRunner(linter, config), processes\n", "test_patch": "diff --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@@ -14,6 +14,7 @@\n from sqlfluff.cli.formatters import OutputStreamFormatter\n from sqlfluff.cli.outputstream import make_output_stream\n from sqlfluff.core.linter import LintingResult, NoQaDirective\n+from sqlfluff.core.linter.runner import get_runner\n import sqlfluff.core.linter as linter\n from sqlfluff.core.parser import GreedyUntil, Ref\n from sqlfluff.core.templaters import TemplatedFile\n@@ -289,6 +290,36 @@ def test_lint_path_parallel_wrapper_exception(patched_lint):\n             result.reraise()\n \n \n+@pytest.mark.parametrize(\n+    \"mock_cpu,in_processes,exp_processes\",\n+    [\n+        # Make the mocked cpu count a really high value which is\n+        # unlikely to collide with the real value. We can then\n+        # test all the different combos.\n+        (512, 1, 1),\n+        (512, 0, 512),\n+        (512, -12, 500),\n+        (512, 5, 5),\n+        # Check that we can't go lower than 1 in a 1 cpu case\n+        (1, -1, 1),\n+    ],\n+)\n+@patch(\"multiprocessing.cpu_count\")\n+def test__linter__get_runner_processes(\n+    patched_cpu_count, mock_cpu, in_processes, exp_processes\n+):\n+    \"\"\"Test that get_runner handles processes correctly.\"\"\"\n+    # Make the mocked cpu count a really high value which is\n+    # unlikely to collide with the real value.\n+    patched_cpu_count.return_value = mock_cpu\n+    _, return_processes = get_runner(\n+        linter=Linter(),\n+        config=FluffConfig(overrides={\"dialect\": \"ansi\"}),\n+        processes=in_processes,\n+    )\n+    assert return_processes == exp_processes\n+\n+\n @patch(\"sqlfluff.core.linter.runner.linter_logger\")\n @patch(\"sqlfluff.core.linter.Linter.lint_rendered\")\n def test__linter__linting_unexpected_error_handled_gracefully(\n", "problem_statement": "Number of processes configurable in .sqlfluff\nBeing able to set the number of processes to run with in .sqlfluff might be useful to avoid having to pass it in the CLI every time.\n", "hints_text": "One thought on this: The same `.sqlfluff` file will sometimes be used on different machines (e.g. various development machines, CI server). We should allow the setting to be somewhat \"context sensitive\" if desired. Proposal:\r\n* Positive values indicate the number of processes to create\r\n* Zero or negative values are interpreted as `number_of_cpus - specified_number`. Thus, a value of `0` means \"use all processors\" and `-1` means \"use all processors except one\".\nIs there a standard way in python to detect the effective available cpus?\n@alanmcruickshank: Yes. Use [`multiprocessing.cpu_count()`](https://docs.python.org/3/library/multiprocessing.html#multiprocessing.cpu_count).\nI'm happy to pick this one up. This actually fits with a small reorg of how I think threads should be configured. I think it fits better if the thread argument is passed in to the `Linter` object on instantiation, rather than when calling `lint_paths`. @barrywhart - does that sit well with you? I realise that changes a little of the structure you originally envisaged.\nSounds good -- no concerns from me.", "created_at": "2022-07-25T09:10:25Z", "version": "1.2", "FAIL_TO_PASS": ["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]"], "PASS_TO_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__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__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"], "environment_setup_commit": "388dd01e05c7dcb880165c7241ed4027d9d0171e"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-897", "base_commit": "b682e1dc71eda183786ad724da25f2fb30b5a621", "patch": "diff --git a/pydicom/dataelem.py b/pydicom/dataelem.py\n--- a/pydicom/dataelem.py\n+++ b/pydicom/dataelem.py\n@@ -378,7 +378,7 @@ def _convert(self, val):\n         elif self.VR == 'TM' and config.datetime_conversion:\n             return pydicom.valuerep.TM(val)\n         elif self.VR == \"UI\":\n-            return UID(val)\n+            return UID(val if val else '')\n         elif not in_py2 and self.VR == \"PN\":\n             return PersonName(val)\n         # Later may need this for PersonName as for UI,\ndiff --git a/pydicom/multival.py b/pydicom/multival.py\n--- a/pydicom/multival.py\n+++ b/pydicom/multival.py\n@@ -2,6 +2,7 @@\n \"\"\"Code for multi-value data elements values,\n or any list of items that must all be the same type.\n \"\"\"\n+from pydicom import compat\n \n try:\n     from collections.abc import MutableSequence\n@@ -30,7 +31,7 @@ def __init__(self, type_constructor, iterable):\n         :param type_constructor: a constructor for the required\n                            type for all list items. Could be the\n                            class, or a factory function. For DICOM\n-                           mult-value data elements, this will be the\n+                           multi-value data elements, this will be the\n                            class or type corresponding to the VR.\n         :param iterable: an iterable (e.g. list, tuple) of items\n                         to initialize the MultiValue list\n@@ -59,8 +60,11 @@ def __setitem__(self, i, val):\n             self._list.__setitem__(i, self.type_constructor(val))\n \n     def __str__(self):\n-        lines = [str(x) for x in self]\n-        return \"['\" + \"', '\".join(lines) + \"']\"\n+        if not self:\n+            return ''\n+        lines = [\"'{}'\".format(x) if isinstance(x, compat.char_types)\n+                 else str(x) for x in self]\n+        return \"[\" + \", \".join(lines) + \"]\"\n \n     __repr__ = __str__\n \ndiff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -603,8 +603,6 @@ def _encode_personname(components, encodings):\n \n class PersonName3(object):\n     def __init__(self, val, encodings=None, original_string=None):\n-        # handle None `val` as empty string\n-        val = val or ''\n         if isinstance(val, PersonName3):\n             encodings = val.encodings\n             self.original_string = val.original_string\n@@ -614,6 +612,9 @@ def __init__(self, val, encodings=None, original_string=None):\n             self.original_string = val\n             self._components = None\n         else:\n+            # handle None `val` as empty string\n+            val = val or ''\n+\n             # this is the decoded string - save the original string if\n             # available for easier writing back\n             self.original_string = original_string\n@@ -774,6 +775,12 @@ def formatted(self, format_str):\n         self._create_dict()\n         return format_str % self._dict\n \n+    def __bool__(self):\n+        if self.original_string is None:\n+            return (self._components is not None and\n+                    (len(self._components) > 1 or bool(self._components[0])))\n+        return bool(self.original_string)\n+\n \n class PersonNameBase(object):\n     \"\"\"Base class for Person Name classes\"\"\"\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@@ -15,6 +15,7 @@\n     DataElement_from_raw,\n )\n from pydicom.dataset import Dataset\n+from pydicom.multival import MultiValue\n from pydicom.tag import Tag\n from pydicom.uid import UID\n from pydicom.valuerep import DSfloat\n@@ -368,6 +369,92 @@ def test_private_repeater_tag(self):\n         assert '[Overlay ID]' == private_data_elem.name\n         assert 'UN' == private_data_elem.VR\n \n+    def test_empty_text_values(self):\n+        \"\"\"Test that assigning an empty value behaves as expected.\"\"\"\n+        def check_empty_text_element(value):\n+            setattr(ds, tag_name, value)\n+            elem = ds[tag_name]\n+            assert bool(elem.value) is False\n+\n+        text_vrs = {\n+            'AE': 'Receiver',\n+            'AS': 'PatientAge',\n+            'AT': 'OffendingElement',\n+            'CS': 'QualityControlSubject',\n+            'DA': 'PatientBirthDate',\n+            'DS': 'PatientWeight',\n+            'DT': 'AcquisitionDateTime',\n+            'IS': 'BeamNumber',\n+            'LO': 'DataSetSubtype',\n+            'LT': 'ExtendedCodeMeaning',\n+            'PN': 'PatientName',\n+            'SH': 'CodeValue',\n+            'ST': 'InstitutionAddress',\n+            'TM': 'StudyTime',\n+            'UC': 'LongCodeValue',\n+            'UI': 'SOPClassUID',\n+            'UR': 'CodingSchemeURL',\n+            'UT': 'StrainAdditionalInformation',\n+        }\n+        ds = Dataset()\n+        # set value to new element\n+        for tag_name in text_vrs.values():\n+            check_empty_text_element(None)\n+            del ds[tag_name]\n+            check_empty_text_element(b'')\n+            del ds[tag_name]\n+            check_empty_text_element(u'')\n+            del ds[tag_name]\n+            check_empty_text_element([])\n+            del ds[tag_name]\n+\n+        # set value to existing element\n+        for tag_name in text_vrs.values():\n+            check_empty_text_element(None)\n+            check_empty_text_element(b'')\n+            check_empty_text_element(u'')\n+            check_empty_text_element([])\n+            check_empty_text_element(None)\n+\n+    def test_empty_binary_values(self):\n+        \"\"\"Test that assigning an empty value behaves as expected for\n+        non-text VRs.\"\"\"\n+        def check_empty_binary_element(value):\n+            setattr(ds, tag_name, value)\n+            elem = ds[tag_name]\n+            assert bool(elem.value) is False\n+\n+        non_text_vrs = {\n+            'SL': 'RationalNumeratorValue',\n+            'SS': 'SelectorSSValue',\n+            'UL': 'SimpleFrameList',\n+            'US': 'SourceAcquisitionBeamNumber',\n+            'FD': 'RealWorldValueLUTData',\n+            'FL': 'VectorAccuracy',\n+            'OB': 'FillPattern',\n+            'OD': 'DoubleFloatPixelData',\n+            'OF': 'UValueData',\n+            'OL': 'TrackPointIndexList',\n+            'OW': 'TrianglePointIndexList',\n+            'UN': 'SelectorUNValue',\n+        }\n+        ds = Dataset()\n+        # set value to new element\n+        for tag_name in non_text_vrs.values():\n+            check_empty_binary_element(None)\n+            del ds[tag_name]\n+            check_empty_binary_element([])\n+            del ds[tag_name]\n+            check_empty_binary_element(MultiValue(int, []))\n+            del ds[tag_name]\n+\n+        # set value to existing element\n+        for tag_name in non_text_vrs.values():\n+            check_empty_binary_element(None)\n+            check_empty_binary_element([])\n+            check_empty_binary_element(MultiValue(int, []))\n+            check_empty_binary_element(None)\n+\n \n class RawDataElementTests(unittest.TestCase):\n     def testKeyError(self):\ndiff --git a/pydicom/tests/test_multival.py b/pydicom/tests/test_multival.py\n--- a/pydicom/tests/test_multival.py\n+++ b/pydicom/tests/test_multival.py\n@@ -1,43 +1,46 @@\n # Copyright 2008-2018 pydicom authors. See LICENSE file for details.\n \"\"\"Test suite for MultiValue class\"\"\"\n \n-import unittest\n+import pytest\n+\n from pydicom.multival import MultiValue\n from pydicom.valuerep import DS, DSfloat, DSdecimal, IS\n-from pydicom import config\n+from pydicom import config, compat\n from copy import deepcopy\n \n import sys\n+\n python_version = sys.version_info\n \n \n-class MultiValuetests(unittest.TestCase):\n+class TestMultiValue(object):\n     def testMultiDS(self):\n         \"\"\"MultiValue: Multi-valued data elements can be created........\"\"\"\n         multival = MultiValue(DS, ['11.1', '22.2', '33.3'])\n         for val in multival:\n-            self.assertTrue(isinstance(val, (DSfloat, DSdecimal)),\n-                            \"Multi-value DS item not converted to DS\")\n+            assert isinstance(val, (DSfloat, DSdecimal))\n \n     def testEmptyElements(self):\n         \"\"\"MultiValue: Empty number string elements are not converted...\"\"\"\n         multival = MultiValue(DSfloat, ['1.0', ''])\n-        self.assertEqual(1.0, multival[0])\n-        self.assertEqual('', multival[1])\n+        assert 1.0 == multival[0]\n+        assert '' == multival[1]\n         multival = MultiValue(IS, ['1', ''])\n-        self.assertEqual(1, multival[0])\n-        self.assertEqual('', multival[1])\n+        assert 1 == multival[0]\n+        assert '' == multival[1]\n         multival = MultiValue(DSdecimal, ['1', ''])\n-        self.assertEqual(1, multival[0])\n-        self.assertEqual('', multival[1])\n+        assert 1 == multival[0]\n+        assert '' == multival[1]\n+        multival = MultiValue(IS, [])\n+        assert not multival\n+        assert 0 == len(multival)\n \n     def testLimits(self):\n         \"\"\"MultiValue: Raise error if any item outside DICOM limits....\"\"\"\n         original_flag = config.enforce_valid_values\n         config.enforce_valid_values = True\n-        self.assertRaises(OverflowError,\n-                          MultiValue,\n-                          IS, [1, -2 ** 31 - 1])\n+        with pytest.raises(OverflowError):\n+            MultiValue(IS, [1, -2 ** 31 - 1])\n         # Overflow error not raised for IS out of DICOM valid range\n         config.enforce_valid_values = original_flag\n \n@@ -45,44 +48,39 @@ def testAppend(self):\n         \"\"\"MultiValue: Append of item converts it to required type...\"\"\"\n         multival = MultiValue(IS, [1, 5, 10])\n         multival.append('5')\n-        self.assertTrue(isinstance(multival[-1], IS))\n-        self.assertEqual(multival[-1], 5,\n-                         \"Item set by append is not correct value\")\n+        assert isinstance(multival[-1], IS)\n+        assert multival[-1] == 5\n \n     def testSetIndex(self):\n         \"\"\"MultiValue: Setting list item converts it to required type\"\"\"\n         multival = MultiValue(IS, [1, 5, 10])\n         multival[1] = '7'\n-        self.assertTrue(isinstance(multival[1], IS))\n-        self.assertEqual(multival[1], 7,\n-                         \"Item set by index is not correct value\")\n+        assert isinstance(multival[1], IS)\n+        assert multival[1] == 7\n \n     def testDeleteIndex(self):\n         \"\"\"MultiValue: Deleting item at index behaves as expected...\"\"\"\n         multival = MultiValue(IS, [1, 5, 10])\n         del multival[1]\n-        self.assertEqual(2, len(multival))\n-        self.assertEqual(multival[0], 1)\n-        self.assertEqual(multival[1], 10)\n+        assert 2 == len(multival)\n+        assert multival[0] == 1\n+        assert multival[1] == 10\n \n     def testExtend(self):\n         \"\"\"MultiValue: Extending a list converts all to required type\"\"\"\n         multival = MultiValue(IS, [1, 5, 10])\n         multival.extend(['7', 42])\n-        self.assertTrue(isinstance(multival[-2], IS))\n-        self.assertTrue(isinstance(multival[-1], IS))\n-        self.assertEqual(multival[-2], 7,\n-                         \"Item set by extend not correct value\")\n+        assert isinstance(multival[-2], IS)\n+        assert isinstance(multival[-1], IS)\n+        assert multival[-2], 7\n \n     def testSlice(self):\n         \"\"\"MultiValue: Setting slice converts items to required type.\"\"\"\n         multival = MultiValue(IS, range(7))\n         multival[2:7:2] = [4, 16, 36]\n         for val in multival:\n-            self.assertTrue(isinstance(val, IS),\n-                            \"Slice IS value not correct type\")\n-        self.assertEqual(multival[4], 16,\n-                         \"Set by slice failed for item 4 of list\")\n+            assert isinstance(val, IS)\n+            assert multival[4] == 16\n \n     def testIssue236DeepCopy(self):\n         \"\"\"MultiValue: deepcopy of MultiValue does not generate an error\"\"\"\n@@ -97,38 +95,45 @@ def testSorting(self):\n         \"\"\"MultiValue: allow inline sort.\"\"\"\n         multival = MultiValue(DS, [12, 33, 5, 7, 1])\n         multival.sort()\n-        self.assertEqual([1, 5, 7, 12, 33], multival)\n+        assert [1, 5, 7, 12, 33] == multival\n         multival.sort(reverse=True)\n-        self.assertEqual([33, 12, 7, 5, 1], multival)\n+        assert [33, 12, 7, 5, 1] == multival\n         multival.sort(key=str)\n-        self.assertEqual([1, 12, 33, 5, 7], multival)\n+        assert [1, 12, 33, 5, 7] == multival\n \n     def test_equal(self):\n         \"\"\"MultiValue: test equality operator\"\"\"\n         multival = MultiValue(DS, [12, 33, 5, 7, 1])\n         multival2 = MultiValue(DS, [12, 33, 5, 7, 1])\n         multival3 = MultiValue(DS, [33, 12, 5, 7, 1])\n-        self.assertTrue(multival == multival2)\n-        self.assertFalse(multival == multival3)\n+        assert multival == multival2\n+        assert not (multival == multival3)\n         multival = MultiValue(str, ['a', 'b', 'c'])\n         multival2 = MultiValue(str, ['a', 'b', 'c'])\n         multival3 = MultiValue(str, ['b', 'c', 'a'])\n-        self.assertTrue(multival == multival2)\n-        self.assertFalse(multival == multival3)\n+        assert multival == multival2\n+        assert not (multival == multival3)\n \n     def test_not_equal(self):\n         \"\"\"MultiValue: test equality operator\"\"\"\n         multival = MultiValue(DS, [12, 33, 5, 7, 1])\n         multival2 = MultiValue(DS, [12, 33, 5, 7, 1])\n         multival3 = MultiValue(DS, [33, 12, 5, 7, 1])\n-        self.assertFalse(multival != multival2)\n-        self.assertTrue(multival != multival3)\n+        assert not multival != multival2\n+        assert multival != multival3\n         multival = MultiValue(str, ['a', 'b', 'c'])\n         multival2 = MultiValue(str, ['a', 'b', 'c'])\n         multival3 = MultiValue(str, ['b', 'c', 'a'])\n-        self.assertFalse(multival != multival2)\n-        self.assertTrue(multival != multival3)\n-\n-\n-if __name__ == \"__main__\":\n-    unittest.main()\n+        assert not (multival != multival2)\n+        assert multival != multival3\n+\n+    def test_str_rep(self):\n+        \"\"\"MultiValue: test print output\"\"\"\n+        multival = MultiValue(IS, [])\n+        assert '' == str(multival)\n+        multival = MultiValue(compat.text_type, [1, 2, 3])\n+        assert \"['1', '2', '3']\" == str(multival)\n+        multival = MultiValue(int, [1, 2, 3])\n+        assert '[1, 2, 3]' == str(multival)\n+        multival = MultiValue(float, [1.1, 2.2, 3.3])\n+        assert '[1.1, 2.2, 3.3]' == str(multival)\n", "problem_statement": "Inconsistencies in value testing for PersonName3\n```python\r\nfrom pydicom.dataset import Dataset\r\n\r\nds = Dataset()\r\nds.PatientName = None  # or ''\r\nif ds.PatientName:\r\n    print('Has a value')\r\nelse:\r\n    print('Has no value')\r\n\r\nif None:  # or ''\r\n    print('Evaluates as True')\r\nelse:\r\n    print('Evaluates as False')\r\n```\r\nPrints `Has a value` then `Evaluates as False`. Should print `Has no value` instead (encoded dataset will have a zero-length element).\r\n\r\nCurrent master, python 3.6.\n", "hints_text": "Should that `if None` read `if ds.PatientName is None`?\r\n\r\nI can see the inconsistency if assigning `None` - which results in an empty value instead of `None`. Assigning an empty string works as expected (has an empty string afterwards).\r\nFor other string tags `None` or `''` can be assigned, though I now wonder if this is the correct behavior, as there is no difference between an empty value and no value in DICOM for string VRs. IMHO, the correct behavior should be to assign no value, e.g. the following should be equivalent:\r\n```python\r\nds.PatientName = ''\r\nds.PatientName = None\r\nds.PatientName = []\r\n```\r\nCurrently, the first 2 result in VM=1, the last in VM=0 (which is correct).\r\n@darcymason, @scaramallion - what do you think?\nThe `if None` was just to demonstrate how python behaves and hence how I naively expect an empty pydicom element value to behave.\r\n\r\nI just want a simple consistent \"does this element have a value\" test, which means an element with `None` and `''` should both fail the test. Essentially if the element will end up encoded with zero-length it should fail. It works as I expect for simple text element values, but `PersonName3` and maybe some of the other VR classes might need `__bool__()` or `__len__()`?\r\n\r\n```python\r\n# I'd like this to work cleanly\r\nfor elem in ds:\r\n   if elem.value:\r\n       print(elem.value)\r\n   else:\r\n       print('(no value available)')\r\n```\r\n\r\nI've been playing around with printing out pretty datasets for pynetdicom and have run into a some problems with consistency in how element values are evaluated and printed.\r\n\r\nI think you're right, they should be VM = 0. I think the last case gives you a `MultiValue` rather than a `PersonName3` which is probably why.\n```python\r\n#!/usr/bin/env python\r\n\r\nfrom pydicom.dataset import Dataset\r\nfrom pydicom.datadict import tag_for_keyword\r\n\r\nVRs = {\r\n    'AE' : 'Receiver',\r\n    'AS' : 'PatientAge',\r\n    'AT' : 'OffendingElement',\r\n    'CS' : 'QualityControlSubject',\r\n    'DA' : 'PatientBirthDate',\r\n    'DS' : 'PatientWeight',\r\n    'DT' : 'AcquisitionDateTime',\r\n    'IS' : 'BeamNumber',\r\n    'LO' : 'DataSetSubtype',\r\n    'LT' : 'ExtendedCodeMeaning',\r\n    'PN' : 'PatientName',\r\n    'SH' : 'CodeValue',\r\n    'ST' : 'InstitutionAddress',\r\n    'TM' : 'StudyTime',\r\n    'UC' : 'LongCodeValue',\r\n    'UI' : 'SOPClassUID',\r\n    'UR' : 'CodingSchemeURL',\r\n    'UT' : 'StrainAdditionalInformation',\r\n    'SL' : 'RationalNumeratorValue',\r\n    'SS' : 'SelectorSSValue',\r\n    'UL' : 'SimpleFrameList',\r\n    'US' : 'SourceAcquisitionBeamNumber',\r\n    'FD' : 'RealWorldValueLUTData',\r\n    'FL' : 'VectorAccuracy',\r\n    'OB' : 'FillPattern',\r\n    'OD' : 'DoubleFloatPixelData',\r\n    'OF' : 'UValueData',\r\n    'OL' : 'TrackPointIndexList',\r\n    'OW' : 'TrianglePointIndexList',\r\n    #'OV' : '',\r\n    'UN' : 'SelectorUNValue',\r\n}\r\n\r\nds = Dataset()\r\nfor vr, kw in VRs.items():\r\n    try:\r\n        setattr(ds, kw, None)\r\n    except:\r\n        print('Failed assigning None', vr)\r\n        continue\r\n\r\n    elem = ds[tag_for_keyword(kw)]\r\n\r\n    try:\r\n        assert bool(elem.value) is False\r\n    except:\r\n        print('Failed empty value test', vr)\r\n\r\n    try:\r\n        assert elem.VM == 0\r\n    except Exception as exc:\r\n        print('Failed VM == 0', vr, elem.VM)\r\n```\r\nUI fails assignment using `None`, they all fail the VM == 0 test, PN fails the value test in Python 3.6, passes in 2.7.\nPN in Python 2 is derived from `unicode`, so this is to be expected. So you agree that VM shall be 0 after assigning `None` or `''`?\n> So you agree that VM shall be 0 after assigning None or ''?\r\n\r\nAs far as I can see the standard doesn't explicitly say so, but I think that's reasonable. DCMTK's `dcmdump` uses that convention.\r\n\r\nWhat do you think @darcymason?\n> I think you're right, they should be VM = 0. I think the last case gives you a MultiValue rather than a PersonName3 which is probably why.\r\n\r\nSorry, I overlooked that post. Yes, and that may be the easiest way to change this - assign an empty `MultiValue` (aka an empty list) in these cases instead of `None` or `''`.\nThere is also the question of how to represent empty values. Currently we get:\r\n```\r\n>>> ds.CodeValue = None\r\n>>> ds.CodeValue\r\n>>> print(ds.CodeValue)\r\nNone\r\n>>> ds['CodeValue'].VM\r\n1\r\n>>> ds.CodeValue = []\r\n>>> ds.CodeValue\r\n['']\r\n>>> ds['CodeValue'].VM\r\n0\r\n```\r\nwhich is quite inconsistent. `None` for an empty value may make sense, or an empty list (other than a list with an empty string), not sure about this. Any ideas?\nI think as long as `bool(value)` would normally evaluate as `False` then it should be considered empty.\r\n```python\r\nvalue = '' # or None or [] or b'' or whatever\r\nif not value:\r\n    print('Empty')\r\n```\r\nFor `MultiValue` it should probably be an OR on `[bool(val) for val in MultiValue]`\n> > So you agree that VM shall be 0 after assigning None or ''?\r\n> \r\n> As far as I can see the standard doesn't explicitly say so, but I think that's reasonable. DCMTK's `dcmdump` uses that convention.\r\n> \r\n> What do you think @darcymason?\r\n\r\nHmmm... just read quickly through, so apologies if I've repeated (or missed) something already said.  Haven't looked at any code yet.\r\n\r\nOne thought is that DICOM type 2 data elements allow blank values, which to me is a real valid value, different than a None.\r\n\r\nSecond thought is we've normally handled a non-existent value by just not having the attribute set, i.e. just don't set the attribute, or if it already exists, delete it.  You check for existence of values by `if keyword in ds`.\r\n\r\nThird thought (the last in this comment I think) is that when we drop support for earlier python versions and can start typing pydicom, do we cause trouble by encouraging use of None?  We have not forbidden setting None, or even incorrect types in many cases, but this is in line with python's duck typing philosophy.  It may be that at some point more control over this could be automatically introduced via typing. Yes, we could Union everything with None, but I would foresee a whole lot of `if` statements to handle those cases.\r\n\r\n\nI frequently use `None` for type 2 elements that are required to be present but may or may not have a value. It just seems to me that this is the 'obvious' thing to do when I want an element with no value. Does it make sense that you need to set an integer VR to `''` if you want an empty value?\r\n\r\n`if keyword in ds` tests for the presence of the element in the dataset, not that the element has a value. \nWell, thinking this through some more, I may be coming around to your way of thinking...I suppose blank type 2 can be considered a zero-length (no value) data element.  Using a null (python None) value for those does have a certain logic to it, similar to Null in databases and so on.  And in that case, VM should be zero, although VM is a weak concept to me anyway, since it exists in dictionaries but not in actual DICOM files, so I'm not too invested in that one way or the other.\nThe other thing is that some elements don't encode properly with a value set to `None`, while others work as I expect.\r\n\r\n```python\r\n>>> from pydicom.dataset import Dataset\r\n>>> from pynetdicom.dsutils import encode  # wrapper for filewriter.write_dataset\r\n>>> ds = Dataset()\r\n>>> ds.Receiver = None  # VR of AE\r\n>>> type(ds.Receiver)\r\nNoneType\r\n>>> encode(ds, True, True)  # Works OK\r\nb'\\x00\\x00\\x00\\x03\\x00\\x00\\x00\\x00'\r\n>>> ds = Dataset()\r\n>>> ds.BitsStored = None  # VR of US\r\n>>> type(ds.BitsStored)\r\nNoneType\r\n>>> encode(ds, True, True)  # Fails to encode\r\n>>>\r\n```\nSo I think this has elevated to a larger conversation on \"`None` support\" in pydicom, make it consistent across all types, or raise exceptions if there are any types or situations where it should not be allowed.\r\n\r\nI'll sleep on it for now, pick it up again tomorrow...\r\n\nMy current instinct is to:\r\n- handle assigning `None`, `''` or `[]` to an element with a text VR as assigning an empty string (as they are all equivalent, and an empty string is most convienient for the user, and probably expected), return `False` as truth value, and have a VM of 0 (this is as implemented in the PR) \r\n- handle assigning `None` or `[]` to an element with a binary VR as assigning `None` (I think this better meets the expections for an element without a value), return `False` as truth value, and have a VM of 0 (in the PR I use a `MultiValue` object, which I would change)\r\n- in the case of IS and DS (these are kind of in-between), we have to make sure they handle the empty string case by returning `None` instead of 0 if getting the numerical value (haven't checked the current behavior) \r\n\r\nUsing `None` for text values, apart from not being upwards compatible, is a thing I have been burned with before. I used to write DICOM elements into a database and had used database `null` to represent elements without values. This turned out to be a big mistake, as there is actually no use case where you have to differentiate no value from an empty value for text values (and indeed there is no such distinction in DICOM itself for text VRs), and led to a lot of unnecessary checking for null values in the user code (and exceptions if they have been missing).\nI've started a separate issue for the empty values discussion #896", "created_at": "2019-07-24T18:43:18Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_dataelem.py::DataElementTests::test_empty_text_values", "pydicom/tests/test_multival.py::TestMultiValue::test_str_rep"], "PASS_TO_PASS": ["pydicom/tests/test_dataelem.py::DataElementTests::testBackslash", "pydicom/tests/test_dataelem.py::DataElementTests::testDSFloatConversion", "pydicom/tests/test_dataelem.py::DataElementTests::testEqualityInheritance", "pydicom/tests/test_dataelem.py::DataElementTests::testEqualityNotElement", "pydicom/tests/test_dataelem.py::DataElementTests::testEqualityPrivateElement", "pydicom/tests/test_dataelem.py::DataElementTests::testEqualitySequenceElement", "pydicom/tests/test_dataelem.py::DataElementTests::testEqualityStandardElement", "pydicom/tests/test_dataelem.py::DataElementTests::testHash", "pydicom/tests/test_dataelem.py::DataElementTests::testKeyword", "pydicom/tests/test_dataelem.py::DataElementTests::testRetired", "pydicom/tests/test_dataelem.py::DataElementTests::testUID", "pydicom/tests/test_dataelem.py::DataElementTests::testVM1", "pydicom/tests/test_dataelem.py::DataElementTests::testVM2", "pydicom/tests/test_dataelem.py::DataElementTests::test_description_group_length", "pydicom/tests/test_dataelem.py::DataElementTests::test_description_unknown", "pydicom/tests/test_dataelem.py::DataElementTests::test_description_unknown_private", "pydicom/tests/test_dataelem.py::DataElementTests::test_empty_binary_values", "pydicom/tests/test_dataelem.py::DataElementTests::test_equality_class_members", "pydicom/tests/test_dataelem.py::DataElementTests::test_getitem_raises", "pydicom/tests/test_dataelem.py::DataElementTests::test_inequality_sequence", "pydicom/tests/test_dataelem.py::DataElementTests::test_inequality_standard", "pydicom/tests/test_dataelem.py::DataElementTests::test_private_repeater_tag", "pydicom/tests/test_dataelem.py::DataElementTests::test_private_tag_in_repeater_range", "pydicom/tests/test_dataelem.py::DataElementTests::test_repeater_str", "pydicom/tests/test_dataelem.py::DataElementTests::test_repr_seq", "pydicom/tests/test_dataelem.py::DataElementTests::test_repval_large_elem", "pydicom/tests/test_dataelem.py::DataElementTests::test_repval_large_vm", "pydicom/tests/test_dataelem.py::DataElementTests::test_repval_strange_type", "pydicom/tests/test_dataelem.py::DataElementTests::test_str_no_vr", "pydicom/tests/test_dataelem.py::RawDataElementTests::testKeyError", "pydicom/tests/test_dataelem.py::RawDataElementTests::testTagWithoutEncodingPython3", "pydicom/tests/test_dataelem.py::RawDataElementTests::testValidTag", "pydicom/tests/test_dataelem.py::RawDataElementTests::test_unknown_vr", "pydicom/tests/test_multival.py::TestMultiValue::testMultiDS", "pydicom/tests/test_multival.py::TestMultiValue::testEmptyElements", "pydicom/tests/test_multival.py::TestMultiValue::testLimits", "pydicom/tests/test_multival.py::TestMultiValue::testAppend", "pydicom/tests/test_multival.py::TestMultiValue::testSetIndex", "pydicom/tests/test_multival.py::TestMultiValue::testDeleteIndex", "pydicom/tests/test_multival.py::TestMultiValue::testExtend", "pydicom/tests/test_multival.py::TestMultiValue::testSlice", "pydicom/tests/test_multival.py::TestMultiValue::testIssue236DeepCopy", "pydicom/tests/test_multival.py::TestMultiValue::test_equal", "pydicom/tests/test_multival.py::TestMultiValue::test_not_equal"], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1739", "base_commit": "909f86dc67eddc88154c9e7bff73fd9d6bfe2e4d", "patch": "diff --git a/pvlib/iotools/pvgis.py b/pvlib/iotools/pvgis.py\n--- a/pvlib/iotools/pvgis.py\n+++ b/pvlib/iotools/pvgis.py\n@@ -45,7 +45,7 @@\n \n def get_pvgis_hourly(latitude, longitude, start=None, end=None,\n                      raddatabase=None, components=True,\n-                     surface_tilt=0, surface_azimuth=0,\n+                     surface_tilt=0, surface_azimuth=180,\n                      outputformat='json',\n                      usehorizon=True, userhorizon=None,\n                      pvcalculation=False,\n@@ -76,9 +76,15 @@ def get_pvgis_hourly(latitude, longitude, start=None, end=None,\n         Otherwise only global irradiance is returned.\n     surface_tilt: float, default: 0\n         Tilt angle from horizontal plane. Ignored for two-axis tracking.\n-    surface_azimuth: float, default: 0\n-        Orientation (azimuth angle) of the (fixed) plane. 0=south, 90=west,\n-        -90: east. Ignored for tracking systems.\n+    surface_azimuth: float, default: 180\n+        Orientation (azimuth angle) of the (fixed) plane. Counter-clockwise\n+        from north (north=0, south=180). This is offset 180 degrees from\n+        the convention used by PVGIS. Ignored for tracking systems.\n+\n+        .. versionchanged:: 0.10.0\n+           The `surface_azimuth` parameter now follows the pvlib convention, which\n+           is counterclockwise from north. However, the convention used by the\n+           PVGIS website and pvlib<=0.9.5 is offset by 180 degrees.\n     usehorizon: bool, default: True\n         Include effects of horizon\n     userhorizon: list of float, default: None\n@@ -144,6 +150,13 @@ def get_pvgis_hourly(latitude, longitude, start=None, end=None,\n     time stamp convention, e.g., SARAH and SARAH2 provide instantaneous values,\n     whereas values from ERA5 are averages for the hour.\n \n+    Warning\n+    -------\n+    The azimuth orientation specified in the output metadata does not\n+    correspond to the pvlib convention, but is offset 180 degrees. This is\n+    despite the fact that the input parameter `surface_tilt` has to be\n+    specified according to the pvlib convention.\n+\n     Notes\n     -----\n     data includes the following fields:\n@@ -191,7 +204,7 @@ def get_pvgis_hourly(latitude, longitude, start=None, end=None,\n     \"\"\"  # noqa: E501\n     # use requests to format the query string by passing params dictionary\n     params = {'lat': latitude, 'lon': longitude, 'outputformat': outputformat,\n-              'angle': surface_tilt, 'aspect': surface_azimuth,\n+              'angle': surface_tilt, 'aspect': surface_azimuth-180,\n               'pvcalculation': int(pvcalculation),\n               'pvtechchoice': pvtechchoice, 'mountingplace': mountingplace,\n               'trackingtype': trackingtype, 'components': int(components),\n@@ -315,6 +328,11 @@ def read_pvgis_hourly(filename, pvgis_format=None, map_variables=True):\n     metadata : dict\n         metadata\n \n+    Warning\n+    -------\n+    The azimuth orientation specified in the output metadata does not\n+    correspond to the pvlib convention, but is offset 180 degrees.\n+\n     Raises\n     ------\n     ValueError\n", "test_patch": "diff --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@@ -206,14 +206,14 @@ def test_read_pvgis_hourly_bad_extension():\n \n \n args_radiation_csv = {\n-    'surface_tilt': 30, 'surface_azimuth': 0, 'outputformat': 'csv',\n+    'surface_tilt': 30, 'surface_azimuth': 180, 'outputformat': 'csv',\n     'usehorizon': False, 'userhorizon': None, 'raddatabase': 'PVGIS-SARAH',\n     'start': 2016, 'end': 2016, 'pvcalculation': False, 'components': True}\n \n url_hourly_radiation_csv = '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'  # noqa: E501\n \n args_pv_json = {\n-    'surface_tilt': 30, 'surface_azimuth': 0, 'outputformat': 'json',\n+    'surface_tilt': 30, 'surface_azimuth': 180, 'outputformat': 'json',\n     'usehorizon': True, 'userhorizon': None, 'raddatabase': 'PVGIS-SARAH2',\n     'start': pd.Timestamp(2013, 1, 1), 'end': pd.Timestamp(2014, 5, 1),\n     'pvcalculation': True, 'peakpower': 10, 'pvtechchoice': 'CIS', 'loss': 5,\n", "problem_statement": "`pvlib.iotools.get_pvgis_hourly`'s `surface_azimuth` parameter doesn't use pvlib's azimuth convention\nNearly everything in pvlib represents azimuth angles as values in [0, 360) clockwise from north, except `pvlib.iotools.get_pvgis_hourly`:\r\n\r\nhttps://github.com/pvlib/pvlib-python/blob/3def7e3375002ee3a5492b7bc609d3fb63a8edb1/pvlib/iotools/pvgis.py#L79-L81\r\n\r\nThis inconsistency is a shame.  However, I don't see any way to switch it to pvlib's convention without a hard break, which is also a shame.  I wonder how others view the cost/benefit analysis here.\r\n\r\nSee also https://github.com/pvlib/pvlib-python/pull/1395#discussion_r1181853794\r\n\r\n\n", "hints_text": "Since `get_pvgis_hourly` is only using `surface_azimuth` to pass a value to a pvgis keyword `aspect`, we could add `aspect` with the South=0 convention and deprecate `surface_azimuth` as is. Then later, we can deprecate `aspect` and add `surface_azimuth` back with its pvlib meaning. Now that I write that out, seems better just to make the breaking change.\r\n", "created_at": "2023-05-16T23:56:19Z", "version": "0.9", "FAIL_TO_PASS": ["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"], "PASS_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_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"], "environment_setup_commit": "6072e0982c3c0236f532ddfa48fbf461180d834e"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1623", "base_commit": "30c62e368529df01faa609d6b38456a7b0db9b53", "patch": "diff --git a/docs/examples/adr-pvarray/plot_simulate_system.py b/docs/examples/adr-pvarray/plot_simulate_system.py\n--- a/docs/examples/adr-pvarray/plot_simulate_system.py\n+++ b/docs/examples/adr-pvarray/plot_simulate_system.py\n@@ -29,10 +29,12 @@\n PVLIB_DIR = pvlib.__path__[0]\n DATA_FILE = os.path.join(PVLIB_DIR, 'data', '723170TYA.CSV')\n \n-tmy, metadata = iotools.read_tmy3(DATA_FILE, coerce_year=1990)\n+tmy, metadata = iotools.read_tmy3(DATA_FILE, coerce_year=1990,\n+                                  map_variables=True)\n \n-df = pd.DataFrame({'ghi': tmy['GHI'], 'dhi': tmy['DHI'], 'dni': tmy['DNI'],\n-                   'temp_air': tmy['DryBulb'], 'wind_speed': tmy['Wspd'],\n+df = pd.DataFrame({'ghi': tmy['ghi'], 'dhi': tmy['dhi'], 'dni': tmy['dni'],\n+                   'temp_air': tmy['temp_air'],\n+                   'wind_speed': tmy['wind_speed'],\n                    })\n \n # %%\ndiff --git a/docs/examples/irradiance-decomposition/plot_diffuse_fraction.py b/docs/examples/irradiance-decomposition/plot_diffuse_fraction.py\n--- a/docs/examples/irradiance-decomposition/plot_diffuse_fraction.py\n+++ b/docs/examples/irradiance-decomposition/plot_diffuse_fraction.py\n@@ -27,7 +27,8 @@\n # of data measured from 1990 to 2010. Therefore we change the timestamps to a\n # common year, 1990.\n DATA_DIR = pathlib.Path(pvlib.__file__).parent / 'data'\n-greensboro, metadata = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990)\n+greensboro, metadata = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990,\n+                                 map_variables=True)\n \n # Many of the diffuse fraction estimation methods require the \"true\" zenith, so\n # we calculate the solar positions for the 1990 at Greensboro, NC.\n@@ -36,8 +37,8 @@\n solpos = get_solarposition(\n     greensboro.index.shift(freq=\"-30T\"), latitude=metadata['latitude'],\n     longitude=metadata['longitude'], altitude=metadata['altitude'],\n-    pressure=greensboro.Pressure*100,  # convert from millibar to Pa\n-    temperature=greensboro.DryBulb)\n+    pressure=greensboro.pressure*100,  # convert from millibar to Pa\n+    temperature=greensboro.temp_air)\n solpos.index = greensboro.index  # reset index to end of the hour\n \n # %%\n@@ -56,10 +57,10 @@\n # an exponential relation with airmass.\n \n out_disc = irradiance.disc(\n-    greensboro.GHI, solpos.zenith, greensboro.index, greensboro.Pressure*100)\n+    greensboro.ghi, solpos.zenith, greensboro.index, greensboro.pressure*100)\n # use \"complete sum\" AKA \"closure\" equations: DHI = GHI - DNI * cos(zenith)\n df_disc = irradiance.complete_irradiance(\n-    solar_zenith=solpos.apparent_zenith, ghi=greensboro.GHI, dni=out_disc.dni,\n+    solar_zenith=solpos.apparent_zenith, ghi=greensboro.ghi, dni=out_disc.dni,\n     dhi=None)\n out_disc = out_disc.rename(columns={'dni': 'dni_disc'})\n out_disc['dhi_disc'] = df_disc.dhi\n@@ -72,11 +73,11 @@\n # developed by Richard Perez and Pierre Ineichen in 1992.\n \n dni_dirint = irradiance.dirint(\n-    greensboro.GHI, solpos.zenith, greensboro.index, greensboro.Pressure*100,\n-    temp_dew=greensboro.DewPoint)\n+    greensboro.ghi, solpos.zenith, greensboro.index, greensboro.pressure*100,\n+    temp_dew=greensboro.temp_dew)\n # use \"complete sum\" AKA \"closure\" equation: DHI = GHI - DNI * cos(zenith)\n df_dirint = irradiance.complete_irradiance(\n-    solar_zenith=solpos.apparent_zenith, ghi=greensboro.GHI, dni=dni_dirint,\n+    solar_zenith=solpos.apparent_zenith, ghi=greensboro.ghi, dni=dni_dirint,\n     dhi=None)\n out_dirint = pd.DataFrame(\n     {'dni_dirint': dni_dirint, 'dhi_dirint': df_dirint.dhi},\n@@ -91,7 +92,7 @@\n # splits kt into 3 regions: linear for kt <= 0.22, a 4th order polynomial\n # between 0.22 < kt <= 0.8, and a horizontal line for kt > 0.8.\n \n-out_erbs = irradiance.erbs(greensboro.GHI, solpos.zenith, greensboro.index)\n+out_erbs = irradiance.erbs(greensboro.ghi, solpos.zenith, greensboro.index)\n out_erbs = out_erbs.rename(columns={'dni': 'dni_erbs', 'dhi': 'dhi_erbs'})\n \n # %%\n@@ -102,7 +103,7 @@\n # exponential correlation that is continuously differentiable and bounded\n # between zero and one.\n \n-out_boland = irradiance.boland(greensboro.GHI, solpos.zenith, greensboro.index)\n+out_boland = irradiance.boland(greensboro.ghi, solpos.zenith, greensboro.index)\n out_boland = out_boland.rename(\n     columns={'dni': 'dni_boland', 'dhi': 'dhi_boland'})\n \n@@ -118,20 +119,20 @@\n # file together to make plotting easier.\n \n dni_renames = {\n-    'DNI': 'TMY3', 'dni_disc': 'DISC', 'dni_dirint': 'DIRINT',\n+    'dni': 'TMY3', 'dni_disc': 'DISC', 'dni_dirint': 'DIRINT',\n     'dni_erbs': 'Erbs', 'dni_boland': 'Boland'}\n dni = [\n-    greensboro.DNI, out_disc.dni_disc, out_dirint.dni_dirint,\n+    greensboro.dni, out_disc.dni_disc, out_dirint.dni_dirint,\n     out_erbs.dni_erbs, out_boland.dni_boland]\n dni = pd.concat(dni, axis=1).rename(columns=dni_renames)\n dhi_renames = {\n-    'DHI': 'TMY3', 'dhi_disc': 'DISC', 'dhi_dirint': 'DIRINT',\n+    'dhi': 'TMY3', 'dhi_disc': 'DISC', 'dhi_dirint': 'DIRINT',\n     'dhi_erbs': 'Erbs', 'dhi_boland': 'Boland'}\n dhi = [\n-    greensboro.DHI, out_disc.dhi_disc, out_dirint.dhi_dirint,\n+    greensboro.dhi, out_disc.dhi_disc, out_dirint.dhi_dirint,\n     out_erbs.dhi_erbs, out_boland.dhi_boland]\n dhi = pd.concat(dhi, axis=1).rename(columns=dhi_renames)\n-ghi_kt = pd.concat([greensboro.GHI/1000.0, out_erbs.kt], axis=1)\n+ghi_kt = pd.concat([greensboro.ghi/1000.0, out_erbs.kt], axis=1)\n \n # %%\n # Winter\ndiff --git a/docs/examples/irradiance-transposition/plot_seasonal_tilt.py b/docs/examples/irradiance-transposition/plot_seasonal_tilt.py\n--- a/docs/examples/irradiance-transposition/plot_seasonal_tilt.py\n+++ b/docs/examples/irradiance-transposition/plot_seasonal_tilt.py\n@@ -44,12 +44,13 @@ def get_orientation(self, solar_zenith, solar_azimuth):\n # like we expect:\n \n DATA_DIR = pathlib.Path(pvlib.__file__).parent / 'data'\n-tmy, metadata = iotools.read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990)\n+tmy, metadata = iotools.read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990,\n+                                  map_variables=True)\n # shift from TMY3 right-labeled index to left-labeled index:\n tmy.index = tmy.index - pd.Timedelta(hours=1)\n weather = pd.DataFrame({\n-    'ghi': tmy['GHI'], 'dhi': tmy['DHI'], 'dni': tmy['DNI'],\n-    'temp_air': tmy['DryBulb'], 'wind_speed': tmy['Wspd'],\n+    'ghi': tmy['ghi'], 'dhi': tmy['dhi'], 'dni': tmy['dni'],\n+    'temp_air': tmy['temp_air'], 'wind_speed': tmy['wind_speed'],\n })\n loc = location.Location.from_tmy(metadata)\n solpos = loc.get_solarposition(weather.index)\ndiff --git a/docs/examples/irradiance-transposition/plot_transposition_gain.py b/docs/examples/irradiance-transposition/plot_transposition_gain.py\n--- a/docs/examples/irradiance-transposition/plot_transposition_gain.py\n+++ b/docs/examples/irradiance-transposition/plot_transposition_gain.py\n@@ -32,7 +32,8 @@\n DATA_DIR = pathlib.Path(pvlib.__file__).parent / 'data'\n \n # get TMY3 dataset\n-tmy, metadata = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990)\n+tmy, metadata = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990,\n+                          map_variables=True)\n # TMY3 datasets are right-labeled (AKA \"end of interval\") which means the last\n # interval of Dec 31, 23:00 to Jan 1 00:00 is labeled Jan 1 00:00. When rolling\n # up hourly irradiance to monthly insolation, a spurious January value is\n@@ -60,9 +61,9 @@ def calculate_poa(tmy, solar_position, surface_tilt, surface_azimuth):\n     poa = irradiance.get_total_irradiance(\n         surface_tilt=surface_tilt,\n         surface_azimuth=surface_azimuth,\n-        dni=tmy['DNI'],\n-        ghi=tmy['GHI'],\n-        dhi=tmy['DHI'],\n+        dni=tmy['dni'],\n+        ghi=tmy['ghi'],\n+        dhi=tmy['dhi'],\n         solar_zenith=solar_position['apparent_zenith'],\n         solar_azimuth=solar_position['azimuth'],\n         model='isotropic')\n@@ -97,7 +98,7 @@ def calculate_poa(tmy, solar_position, surface_tilt, surface_azimuth):\n df_monthly['SAT-0.4'] = poa_irradiance.resample('m').sum()\n \n # calculate the percent difference from GHI\n-ghi_monthly = tmy['GHI'].resample('m').sum()\n+ghi_monthly = tmy['ghi'].resample('m').sum()\n df_monthly = 100 * (df_monthly.divide(ghi_monthly, axis=0) - 1)\n \n df_monthly.plot()\ndiff --git a/docs/examples/soiling/plot_greensboro_kimber_soiling.py b/docs/examples/soiling/plot_greensboro_kimber_soiling.py\n--- a/docs/examples/soiling/plot_greensboro_kimber_soiling.py\n+++ b/docs/examples/soiling/plot_greensboro_kimber_soiling.py\n@@ -40,9 +40,10 @@\n DATA_DIR = pathlib.Path(pvlib.__file__).parent / 'data'\n \n # get TMY3 data with rain\n-greensboro, _ = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990)\n+greensboro, _ = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990,\n+                          map_variables=True)\n # get the rain data\n-greensboro_rain = greensboro.Lprecipdepth\n+greensboro_rain = greensboro['Lprecip depth (mm)']\n # calculate soiling with no wash dates and cleaning threshold of 25-mm of rain\n THRESHOLD = 25.0\n soiling_no_wash = kimber(greensboro_rain, cleaning_threshold=THRESHOLD)\ndiff --git a/pvlib/iotools/tmy.py b/pvlib/iotools/tmy.py\n--- a/pvlib/iotools/tmy.py\n+++ b/pvlib/iotools/tmy.py\n@@ -3,9 +3,28 @@\n import datetime\n import re\n import pandas as pd\n-\n-\n-def read_tmy3(filename, coerce_year=None, recolumn=True):\n+import warnings\n+from pvlib._deprecation import pvlibDeprecationWarning\n+\n+# Dictionary mapping TMY3 names to pvlib names\n+VARIABLE_MAP = {\n+    'GHI (W/m^2)': 'ghi',\n+    'ETR (W/m^2)': 'ghi_extra',\n+    'DNI (W/m^2)': 'dni',\n+    'ETRN (W/m^2)': 'dni_extra',\n+    'DHI (W/m^2)': 'dhi',\n+    'Pressure (mbar)': 'pressure',\n+    'Wdir (degrees)': 'wind_direction',\n+    'Wspd (m/s)': 'wind_speed',\n+    'Dry-bulb (C)': 'temp_air',\n+    'Dew-point (C)': 'temp_dew',\n+    'RHum (%)': 'relative_humidity',\n+    'Alb (unitless)': 'albedo',\n+    'Pwat (cm)': 'precipitable_water'\n+}\n+\n+\n+def read_tmy3(filename, coerce_year=None, map_variables=None, recolumn=None):\n     \"\"\"Read a TMY3 file into a pandas dataframe.\n \n     Note that values contained in the metadata dictionary are unchanged\n@@ -24,9 +43,13 @@ def read_tmy3(filename, coerce_year=None, recolumn=True):\n         If supplied, the year of the index will be set to `coerce_year`, except\n         for the last index value which will be set to the *next* year so that\n         the index increases monotonically.\n-    recolumn : bool, default True\n+    map_variables : bool, default None\n+        When True, renames columns of the DataFrame to pvlib variable names\n+        where applicable. See variable :const:`VARIABLE_MAP`.\n+    recolumn : bool (deprecated, use map_variables instead)\n         If ``True``, apply standard names to TMY3 columns. Typically this\n         results in stripping the units from the column name.\n+        Cannot be used in combination with ``map_variables``.\n \n     Returns\n     -------\n@@ -57,80 +80,83 @@ def read_tmy3(filename, coerce_year=None, recolumn=True):\n     USAF              Int     USAF identifier\n     ===============   ======  ===================\n \n-    =====================       ======================================================================================================================================================\n-    field                       description\n-    =====================       ======================================================================================================================================================\n-    Index                       A pandas datetime index. NOTE, the index is timezone aware, and times are set to local standard time (daylight savings is not included)\n-    ETR                         Extraterrestrial horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n-    ETRN                        Extraterrestrial normal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n-    GHI                         Direct and diffuse horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n-    GHISource                   See [1]_, Table 1-4\n-    GHIUncertainty              Uncertainty based on random and bias error estimates see [2]_\n-    DNI                         Amount of direct normal radiation (modeled) recv'd during 60 mintues prior to timestamp, Wh/m^2\n-    DNISource                   See [1]_, Table 1-4\n-    DNIUncertainty              Uncertainty based on random and bias error estimates see [2]_\n-    DHI                         Amount of diffuse horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n-    DHISource                   See [1]_, Table 1-4\n-    DHIUncertainty              Uncertainty based on random and bias error estimates see [2]_\n-    GHillum                     Avg. total horizontal illuminance recv'd during the 60 minutes prior to timestamp, lx\n-    GHillumSource               See [1]_, Table 1-4\n-    GHillumUncertainty          Uncertainty based on random and bias error estimates see [2]_\n-    DNillum                     Avg. direct normal illuminance recv'd during the 60 minutes prior to timestamp, lx\n-    DNillumSource               See [1]_, Table 1-4\n-    DNillumUncertainty          Uncertainty based on random and bias error estimates see [2]_\n-    DHillum                     Avg. horizontal diffuse illuminance recv'd during the 60 minutes prior to timestamp, lx\n-    DHillumSource               See [1]_, Table 1-4\n-    DHillumUncertainty          Uncertainty based on random and bias error estimates see [2]_\n-    Zenithlum                   Avg. luminance at the sky's zenith during the 60 minutes prior to timestamp, cd/m^2\n-    ZenithlumSource             See [1]_, Table 1-4\n-    ZenithlumUncertainty        Uncertainty based on random and bias error estimates see [1]_ section 2.10\n-    TotCld                      Amount of sky dome covered by clouds or obscuring phenonema at time stamp, tenths of sky\n-    TotCldSource                See [1]_, Table 1-5\n-    TotCldUncertainty           See [1]_, Table 1-6\n-    OpqCld                      Amount of sky dome covered by clouds or obscuring phenonema that prevent observing the sky at time stamp, tenths of sky\n-    OpqCldSource                See [1]_, Table 1-5\n-    OpqCldUncertainty           See [1]_, Table 1-6\n-    DryBulb                     Dry bulb temperature at the time indicated, deg C\n-    DryBulbSource               See [1]_, Table 1-5\n-    DryBulbUncertainty          See [1]_, Table 1-6\n-    DewPoint                    Dew-point temperature at the time indicated, deg C\n-    DewPointSource              See [1]_, Table 1-5\n-    DewPointUncertainty         See [1]_, Table 1-6\n-    RHum                        Relatitudeive humidity at the time indicated, percent\n-    RHumSource                  See [1]_, Table 1-5\n-    RHumUncertainty             See [1]_, Table 1-6\n-    Pressure                    Station pressure at the time indicated, 1 mbar\n-    PressureSource              See [1]_, Table 1-5\n-    PressureUncertainty         See [1]_, Table 1-6\n-    Wdir                        Wind direction at time indicated, degrees from north (360 = north; 0 = undefined,calm)\n-    WdirSource                  See [1]_, Table 1-5\n-    WdirUncertainty             See [1]_, Table 1-6\n-    Wspd                        Wind speed at the time indicated, meter/second\n-    WspdSource                  See [1]_, Table 1-5\n-    WspdUncertainty             See [1]_, Table 1-6\n-    Hvis                        Distance to discernable remote objects at time indicated (7777=unlimited), meter\n-    HvisSource                  See [1]_, Table 1-5\n-    HvisUncertainty             See [1]_, Table 1-6\n-    CeilHgt                     Height of cloud base above local terrain (7777=unlimited), meter\n-    CeilHgtSource               See [1]_, Table 1-5\n-    CeilHgtUncertainty          See [1]_, Table 1-6\n-    Pwat                        Total precipitable water contained in a column of unit cross section from earth to top of atmosphere, cm\n-    PwatSource                  See [1]_, Table 1-5\n-    PwatUncertainty             See [1]_, Table 1-6\n-    AOD                         The broadband aerosol optical depth per unit of air mass due to extinction by aerosol component of atmosphere, unitless\n-    AODSource                   See [1]_, Table 1-5\n-    AODUncertainty              See [1]_, Table 1-6\n-    Alb                         The ratio of reflected solar irradiance to global horizontal irradiance, unitless\n-    AlbSource                   See [1]_, Table 1-5\n-    AlbUncertainty              See [1]_, Table 1-6\n-    Lprecipdepth                The amount of liquid precipitation observed at indicated time for the period indicated in the liquid precipitation quantity field, millimeter\n-    Lprecipquantity             The period of accumulatitudeion for the liquid precipitation depth field, hour\n-    LprecipSource               See [1]_, Table 1-5\n-    LprecipUncertainty          See [1]_, Table 1-6\n-    PresWth                     Present weather code, see [2]_.\n-    PresWthSource               Present weather code source, see [2]_.\n-    PresWthUncertainty          Present weather code uncertainty, see [2]_.\n-    =====================       ======================================================================================================================================================\n+\n+    ========================       ======================================================================================================================================================\n+    field                          description\n+    ========================       ======================================================================================================================================================\n+    **\u2020 denotes variables that are mapped when `map_variables` is True**\n+    -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------\n+    Index                          A pandas datetime index. NOTE, the index is timezone aware, and times are set to local standard time (daylight savings is not included)\n+    ghi_extra\u2020                     Extraterrestrial horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n+    dni_extra\u2020                     Extraterrestrial normal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n+    ghi\u2020                           Direct and diffuse horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n+    GHI source                     See [1]_, Table 1-4\n+    GHI uncert (%)                 Uncertainty based on random and bias error estimates see [2]_\n+    dni\u2020                           Amount of direct normal radiation (modeled) recv'd during 60 mintues prior to timestamp, Wh/m^2\n+    DNI source                     See [1]_, Table 1-4\n+    DNI uncert (%)                 Uncertainty based on random and bias error estimates see [2]_\n+    dhi\u2020                           Amount of diffuse horizontal radiation recv'd during 60 minutes prior to timestamp, Wh/m^2\n+    DHI source                     See [1]_, Table 1-4\n+    DHI uncert (%)                 Uncertainty based on random and bias error estimates see [2]_\n+    GH illum (lx)                  Avg. total horizontal illuminance recv'd during the 60 minutes prior to timestamp, lx\n+    GH illum source                See [1]_, Table 1-4\n+    GH illum uncert (%)            Uncertainty based on random and bias error estimates see [2]_\n+    DN illum (lx)                  Avg. direct normal illuminance recv'd during the 60 minutes prior to timestamp, lx\n+    DN illum source                See [1]_, Table 1-4\n+    DN illum uncert (%)            Uncertainty based on random and bias error estimates see [2]_\n+    DH illum (lx)                  Avg. horizontal diffuse illuminance recv'd during the 60 minutes prior to timestamp, lx\n+    DH illum source                See [1]_, Table 1-4\n+    DH illum uncert (%)            Uncertainty based on random and bias error estimates see [2]_\n+    Zenith lum (cd/m^2)            Avg. luminance at the sky's zenith during the 60 minutes prior to timestamp, cd/m^2\n+    Zenith lum source              See [1]_, Table 1-4\n+    Zenith lum uncert (%)          Uncertainty based on random and bias error estimates see [1]_ section 2.10\n+    TotCld (tenths)                Amount of sky dome covered by clouds or obscuring phenonema at time stamp, tenths of sky\n+    TotCld source                  See [1]_, Table 1-5\n+    TotCld uncert (code)           See [1]_, Table 1-6\n+    OpqCld (tenths)                Amount of sky dome covered by clouds or obscuring phenonema that prevent observing the sky at time stamp, tenths of sky\n+    OpqCld source                  See [1]_, Table 1-5\n+    OpqCld uncert (code)           See [1]_, Table 1-6\n+    temp_air\u2020                      Dry bulb temperature at the time indicated, deg C\n+    Dry-bulb source                See [1]_, Table 1-5\n+    Dry-bulb uncert (code)         See [1]_, Table 1-6\n+    temp_dew\u2020                      Dew-point temperature at the time indicated, deg C\n+    Dew-point source               See [1]_, Table 1-5\n+    Dew-point uncert (code)        See [1]_, Table 1-6\n+    relative_humidity\u2020             Relatitudeive humidity at the time indicated, percent\n+    RHum source                    See [1]_, Table 1-5\n+    RHum uncert (code)             See [1]_, Table 1-6\n+    pressure\u2020                      Station pressure at the time indicated, 1 mbar\n+    Pressure source                See [1]_, Table 1-5\n+    Pressure uncert (code)         See [1]_, Table 1-6\n+    wind_direction\u2020                Wind direction at time indicated, degrees from north (360 = north; 0 = undefined,calm)\n+    Wdir source                    See [1]_, Table 1-5\n+    Wdir uncert (code)             See [1]_, Table 1-6\n+    wind_speed\u2020                    Wind speed at the time indicated, meter/second\n+    Wspd source                    See [1]_, Table 1-5\n+    Wspd uncert (code)             See [1]_, Table 1-6\n+    Hvis (m)                       Distance to discernable remote objects at time indicated (7777=unlimited), meter\n+    Hvis source                    See [1]_, Table 1-5\n+    Hvis uncert (coe)              See [1]_, Table 1-6\n+    CeilHgt (m)                    Height of cloud base above local terrain (7777=unlimited), meter\n+    CeilHgt source                 See [1]_, Table 1-5\n+    CeilHgt uncert (code)          See [1]_, Table 1-6\n+    precipitable_water\u2020            Total precipitable water contained in a column of unit cross section from earth to top of atmosphere, cm\n+    Pwat source                    See [1]_, Table 1-5\n+    Pwat uncert (code)             See [1]_, Table 1-6\n+    AOD                            The broadband aerosol optical depth per unit of air mass due to extinction by aerosol component of atmosphere, unitless\n+    AOD source                     See [1]_, Table 1-5\n+    AOD uncert (code)              See [1]_, Table 1-6\n+    albedo\u2020                        The ratio of reflected solar irradiance to global horizontal irradiance, unitless\n+    Alb source                     See [1]_, Table 1-5\n+    Alb uncert (code)              See [1]_, Table 1-6\n+    Lprecip depth (mm)             The amount of liquid precipitation observed at indicated time for the period indicated in the liquid precipitation quantity field, millimeter\n+    Lprecip quantity (hr)          The period of accumulatitudeion for the liquid precipitation depth field, hour\n+    Lprecip source                 See [1]_, Table 1-5\n+    Lprecip uncert (code)          See [1]_, Table 1-6\n+    PresWth (METAR code)           Present weather code, see [2]_.\n+    PresWth source                 Present weather code source, see [2]_.\n+    PresWth uncert (code)          Present weather code uncertainty, see [2]_.\n+    ========================       ======================================================================================================================================================\n \n     .. admonition:: Midnight representation\n \n@@ -152,8 +178,10 @@ def read_tmy3(filename, coerce_year=None, recolumn=True):\n     ----------\n     .. [1] Wilcox, S and Marion, W. \"Users Manual for TMY3 Data Sets\".\n        NREL/TP-581-43156, Revised May 2008.\n+       :doi:`10.2172/928611`\n     .. [2] Wilcox, S. (2007). National Solar Radiation Database 1991 2005\n        Update: Users Manual. 472 pp.; NREL Report No. TP-581-41364.\n+       :doi:`10.2172/901864`\n     .. [3] `SolarAnywhere file formats\n        <https://www.solaranywhere.com/support/historical-data/file-formats/>`_\n     \"\"\"  # noqa: E501\n@@ -198,9 +226,26 @@ def read_tmy3(filename, coerce_year=None, recolumn=True):\n     # NOTE: as of pvlib-0.6.3, min req is pandas-0.18.1, so pd.to_timedelta\n     # unit must be in (D,h,m,s,ms,us,ns), but pandas>=0.24 allows unit='hour'\n     data.index = data_ymd + pd.to_timedelta(shifted_hour, unit='h')\n-\n-    if recolumn:\n-        data = _recolumn(data)  # rename to standard column names\n+    # shouldnt' specify both recolumn and map_variables\n+    if recolumn is not None and map_variables is not None:\n+        msg = \"`map_variables` and `recolumn` cannot both be specified\"\n+        raise ValueError(msg)\n+    elif map_variables is None and recolumn is not None:\n+        warnings.warn(\n+            'The recolumn parameter is deprecated and will be removed in '\n+            'pvlib 0.11.0. Use `map_variables` instead, although note that '\n+            'its behavior is different from `recolumn`.',\n+            pvlibDeprecationWarning)\n+    elif map_variables is None and recolumn is None:\n+        warnings.warn(\n+            'TMY3 variable names will be renamed to pvlib conventions by '\n+            'default starting in pvlib 0.11.0. Specify map_variables=True '\n+            'to enable that behavior now, or specify map_variables=False '\n+            'to hide this warning.', pvlibDeprecationWarning)\n+    if map_variables:\n+        data = data.rename(columns=VARIABLE_MAP)\n+    elif recolumn or (recolumn is None and map_variables is None):\n+        data = _recolumn(data)\n \n     data = data.tz_localize(int(meta['TZ'] * 3600))\n \n", "test_patch": "diff --git a/pvlib/tests/iotools/test_tmy.py b/pvlib/tests/iotools/test_tmy.py\n--- a/pvlib/tests/iotools/test_tmy.py\n+++ b/pvlib/tests/iotools/test_tmy.py\n@@ -1,8 +1,10 @@\n import numpy as np\n import pandas as pd\n from pvlib.iotools import tmy\n+from pvlib._deprecation import pvlibDeprecationWarning\n from ..conftest import DATA_DIR\n import pytest\n+import warnings\n \n # test the API works\n from pvlib.iotools import read_tmy3\n@@ -16,29 +18,60 @@\n \n \n def test_read_tmy3():\n-    tmy.read_tmy3(TMY3_TESTFILE)\n+    tmy.read_tmy3(TMY3_TESTFILE, map_variables=False)\n \n \n def test_read_tmy3_recolumn():\n-    data, meta = tmy.read_tmy3(TMY3_TESTFILE)\n+    with warnings.catch_warnings():\n+        warnings.simplefilter(\"ignore\")\n+        data, meta = tmy.read_tmy3(TMY3_TESTFILE, recolumn=True)\n     assert 'GHISource' in data.columns\n \n \n def test_read_tmy3_norecolumn():\n-    data, _ = tmy.read_tmy3(TMY3_TESTFILE, recolumn=False)\n+    data, _ = tmy.read_tmy3(TMY3_TESTFILE, map_variables=False)\n     assert 'GHI source' in data.columns\n \n \n+def test_read_tmy3_raise_valueerror():\n+    with pytest.raises(ValueError, match='`map_variables` and `recolumn`'):\n+        _ = tmy.read_tmy3(TMY3_TESTFILE, recolumn=True, map_variables=True)\n+\n+\n+def test_read_tmy3_map_variables():\n+    data, meta = tmy.read_tmy3(TMY3_TESTFILE, map_variables=True)\n+    assert 'ghi' in data.columns\n+    assert 'dni' in data.columns\n+    assert 'dhi' in data.columns\n+    assert 'pressure' in data.columns\n+    assert 'wind_direction' in data.columns\n+    assert 'wind_speed' in data.columns\n+    assert 'temp_air' in data.columns\n+    assert 'temp_dew' in data.columns\n+    assert 'relative_humidity' in data.columns\n+    assert 'albedo' in data.columns\n+    assert 'ghi_extra' in data.columns\n+    assert 'dni_extra' in data.columns\n+    assert 'precipitable_water' in data.columns\n+\n+\n+def test_read_tmy3_map_variables_deprecating_warning():\n+    with pytest.warns(pvlibDeprecationWarning, match='names will be renamed'):\n+        data, meta = tmy.read_tmy3(TMY3_TESTFILE)\n+\n+\n def test_read_tmy3_coerce_year():\n     coerce_year = 1987\n-    data, _ = tmy.read_tmy3(TMY3_TESTFILE, coerce_year=coerce_year)\n+    data, _ = tmy.read_tmy3(TMY3_TESTFILE, coerce_year=coerce_year,\n+                            map_variables=False)\n     assert (data.index[:-1].year == 1987).all()\n     assert data.index[-1].year == 1988\n \n \n def test_read_tmy3_no_coerce_year():\n     coerce_year = None\n-    data, _ = tmy.read_tmy3(TMY3_TESTFILE, coerce_year=coerce_year)\n+    data, _ = tmy.read_tmy3(TMY3_TESTFILE, coerce_year=coerce_year,\n+                            map_variables=False)\n     assert 1997 and 1999 in data.index.year\n     assert data.index[-2] == pd.Timestamp('1998-12-31 23:00:00-09:00')\n     assert data.index[-1] == pd.Timestamp('1999-01-01 00:00:00-09:00')\n@@ -50,7 +83,7 @@ def test_read_tmy2():\n \n def test_gh865_read_tmy3_feb_leapyear_hr24():\n     \"\"\"correctly parse the 24th hour if the tmy3 file has a leap year in feb\"\"\"\n-    data, meta = read_tmy3(TMY3_FEB_LEAPYEAR)\n+    data, meta = read_tmy3(TMY3_FEB_LEAPYEAR, map_variables=False)\n     # just to be safe, make sure this _IS_ the Greensboro file\n     greensboro = {\n         'USAF': 723170,\n@@ -66,7 +99,8 @@ def test_gh865_read_tmy3_feb_leapyear_hr24():\n     assert data.index[1414] == pd.Timestamp('1996-02-28 23:00:00-0500')\n     assert data.index[1415] == pd.Timestamp('1996-03-01 00:00:00-0500')\n     # now check if it parses correctly when we try to coerce the year\n-    data, _ = read_tmy3(TMY3_FEB_LEAPYEAR, coerce_year=1990)\n+    data, _ = read_tmy3(TMY3_FEB_LEAPYEAR, coerce_year=1990,\n+                        map_variables=False)\n     # if get's here w/o an error, then gh865 is fixed, but let's check anyway\n     assert all(data.index[:-1].year == 1990)\n     assert data.index[-1].year == 1991\n@@ -87,7 +121,7 @@ def test_solaranywhere_tmy3(solaranywhere_index):\n     # The SolarAnywhere TMY3 format specifies midnight as 00:00 whereas the\n     # NREL TMY3 format utilizes 24:00. The SolarAnywhere file is therefore\n     # included to test files with  00:00 timestamps are parsed correctly\n-    data, meta = tmy.read_tmy3(TMY3_SOLARANYWHERE)\n+    data, meta = tmy.read_tmy3(TMY3_SOLARANYWHERE, map_variables=False)\n     pd.testing.assert_index_equal(data.index, solaranywhere_index)\n     assert meta['USAF'] == 0\n     assert meta['Name'] == 'Burlington  United States'\ndiff --git a/pvlib/tests/test_location.py b/pvlib/tests/test_location.py\n--- a/pvlib/tests/test_location.py\n+++ b/pvlib/tests/test_location.py\n@@ -212,7 +212,7 @@ def test_get_clearsky_valueerror(times):\n def test_from_tmy_3():\n     from pvlib.tests.iotools.test_tmy import TMY3_TESTFILE\n     from pvlib.iotools import read_tmy3\n-    data, meta = read_tmy3(TMY3_TESTFILE)\n+    data, meta = read_tmy3(TMY3_TESTFILE, map_variables=True)\n     loc = Location.from_tmy(meta, data)\n     assert loc.name is not None\n     assert loc.altitude != 0\ndiff --git a/pvlib/tests/test_soiling.py b/pvlib/tests/test_soiling.py\n--- a/pvlib/tests/test_soiling.py\n+++ b/pvlib/tests/test_soiling.py\n@@ -147,8 +147,9 @@ def test_hsu_variable_time_intervals(rainfall_input, expected_output_3):\n @pytest.fixture\n def greensboro_rain():\n     # get TMY3 data with rain\n-    greensboro, _ = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990)\n-    return greensboro.Lprecipdepth\n+    greensboro, _ = read_tmy3(DATA_DIR / '723170TYA.CSV', coerce_year=1990,\n+                              map_variables=True)\n+    return greensboro['Lprecip depth (mm)']\n \n \n @pytest.fixture\n", "problem_statement": "Add variable mapping of read_tmy3\n**Is your feature request related to a problem? Please describe.**\r\nThis PR proposes that a `map_variables` parameter be added to the `read_tmy3` function. Additionally, the current `rename_columns` parameter (which removes the units from the column names) should be deprecated. See #714 for a discussion on the topic.\r\n\r\n**Describe the solution you'd like**\r\nA `map_variables` parameter should be added (defaulting to None), and if specified as True then it should override the `rename_columns` parameter and map the column names to standard pvlib names. A deperecation warning should be added stating that the `rename_columns` parameter will be retired starting in pvlib 0.11.0 - the deprecation warning should be silenced if `map_variables` is specified as either True or False.\r\n\n", "hints_text": "", "created_at": "2022-12-21T16:44:19Z", "version": "0.8", "FAIL_TO_PASS": ["pvlib/tests/iotools/test_tmy.py::test_read_tmy3", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_norecolumn", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_raise_valueerror", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_map_variables", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_map_variables_deprecating_warning", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_coerce_year", "pvlib/tests/iotools/test_tmy.py::test_read_tmy3_no_coerce_year", "pvlib/tests/iotools/test_tmy.py::test_gh865_read_tmy3_feb_leapyear_hr24", "pvlib/tests/iotools/test_tmy.py::test_solaranywhere_tmy3", "pvlib/tests/test_location.py::test_from_tmy_3", "pvlib/tests/test_soiling.py::test_kimber_nowash", "pvlib/tests/test_soiling.py::test_kimber_manwash", "pvlib/tests/test_soiling.py::test_kimber_norain", "pvlib/tests/test_soiling.py::test_kimber_initial_soil"], "PASS_TO_PASS": ["pvlib/tests/iotools/test_tmy.py::test_read_tmy3_recolumn", "pvlib/tests/iotools/test_tmy.py::test_read_tmy2", "pvlib/tests/test_location.py::test_location_required", "pvlib/tests/test_location.py::test_location_all", "pvlib/tests/test_location.py::test_location_tz[tz0]", "pvlib/tests/test_location.py::test_location_tz[America/Phoenix]", "pvlib/tests/test_location.py::test_location_tz[-7]", "pvlib/tests/test_location.py::test_location_tz[-7.0]", "pvlib/tests/test_location.py::test_location_tz[tz4]", "pvlib/tests/test_location.py::test_location_invalid_tz", "pvlib/tests/test_location.py::test_location_invalid_tz_type", "pvlib/tests/test_location.py::test_location_print_all", "pvlib/tests/test_location.py::test_location_print_pytz", "pvlib/tests/test_location.py::test_get_clearsky", "pvlib/tests/test_location.py::test_get_clearsky_ineichen_supply_linke", "pvlib/tests/test_location.py::test_get_clearsky_haurwitz", "pvlib/tests/test_location.py::test_get_clearsky_simplified_solis", "pvlib/tests/test_location.py::test_get_clearsky_simplified_solis_apparent_elevation", "pvlib/tests/test_location.py::test_get_clearsky_simplified_solis_dni_extra", "pvlib/tests/test_location.py::test_get_clearsky_simplified_solis_pressure", "pvlib/tests/test_location.py::test_get_clearsky_simplified_solis_aod_pw", "pvlib/tests/test_location.py::test_get_clearsky_valueerror", "pvlib/tests/test_location.py::test_from_tmy_2", "pvlib/tests/test_location.py::test_from_epw", "pvlib/tests/test_location.py::test_get_solarposition", "pvlib/tests/test_location.py::test_get_airmass", "pvlib/tests/test_location.py::test_get_airmass_valueerror", "pvlib/tests/test_location.py::test_Location___repr__", "pvlib/tests/test_location.py::test_get_sun_rise_set_transit", "pvlib/tests/test_location.py::test_get_sun_rise_set_transit_valueerror", "pvlib/tests/test_location.py::test_extra_kwargs", "pvlib/tests/test_location.py::test_lookup_altitude", "pvlib/tests/test_soiling.py::test_hsu_no_cleaning", "pvlib/tests/test_soiling.py::test_hsu", "pvlib/tests/test_soiling.py::test_hsu_defaults", "pvlib/tests/test_soiling.py::test_hsu_variable_time_intervals"], "environment_setup_commit": "ef8ad2fee9840a77d14b0dfd17fc489dd85c9b91"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-944", "base_commit": "a0300a69a1da1626caef0d9738cff29b17ce79cc", "patch": "diff --git a/pydicom/valuerep.py b/pydicom/valuerep.py\n--- a/pydicom/valuerep.py\n+++ b/pydicom/valuerep.py\n@@ -545,7 +545,7 @@ def MultiString(val, valtype=str):\n     # 2005.05.25: also check for trailing 0, error made\n     # in PET files we are converting\n \n-    if val and (val.endswith(' ') or val.endswith('\\x00')):\n+    while val and (val.endswith(' ') or val.endswith('\\x00')):\n         val = val[:-1]\n     splitup = val.split(\"\\\\\")\n \ndiff --git a/pydicom/values.py b/pydicom/values.py\n--- a/pydicom/values.py\n+++ b/pydicom/values.py\n@@ -407,7 +407,7 @@ def convert_single_string(byte_string, encodings=None):\n     \"\"\"\n     encodings = encodings or [default_encoding]\n     value = decode_string(byte_string, encodings, TEXT_VR_DELIMS)\n-    if value and value.endswith(' '):\n+    while value and (value.endswith(' ') or value.endswith('\\0')):\n         value = value[:-1]\n     return value\n \n", "test_patch": "diff --git a/pydicom/tests/test_values.py b/pydicom/tests/test_values.py\n--- a/pydicom/tests/test_values.py\n+++ b/pydicom/tests/test_values.py\n@@ -86,6 +86,21 @@ def test_single_value_with_delimiters(self):\n         expected = u'\u0394\u03b9\u03bf\u03bd\u03c5\u03c3\u03b9\u03bf\u03c2\\r\\nJ\u00e9r\u00f4me/\u041b\u044e\u043ace\u043c\u0431yp\u0433\\tJ\u00e9r\u00f4me'\n         assert expected == convert_single_string(bytestring, encodings)\n \n+    def test_value_ending_with_padding(self):\n+        bytestring = b'Value ending with spaces   '\n+        assert 'Value ending with spaces' == convert_single_string(bytestring)\n+        assert 'Value ending with spaces' == convert_text(bytestring)\n+\n+        bytestring = b'Values  \\\\with spaces   '\n+        assert ['Values', 'with spaces'] == convert_text(bytestring)\n+\n+        bytestring = b'Value ending with zeros\\0\\0\\0'\n+        assert 'Value ending with zeros' == convert_single_string(bytestring)\n+        assert 'Value ending with zeros' == convert_text(bytestring)\n+\n+        bytestring = b'Values\\0\\0\\\\with zeros\\0'\n+        assert ['Values', 'with zeros'] == convert_text(bytestring)\n+\n \n class TestConvertAT(object):\n     def test_big_endian(self):\n", "problem_statement": "Embedded Null character\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\nKeyError                                  Traceback (most recent call last)\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/charset.py in convert_encodings(encodings)\r\n    624         try:\r\n--> 625             py_encodings.append(python_encoding[encoding])\r\n    626         except KeyError:\r\n\r\nKeyError: 'ISO_IR 100\\x00'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nValueError                                Traceback (most recent call last)\r\n<ipython-input-12-605c3c3edcf4> in <module>\r\n      4 print(filename)\r\n      5 dcm = pydicom.dcmread(filename,force=True)\r\n----> 6 dcm = pydicom.dcmread('/home/zhuzhemin/XrayKeyPoints/data/10-31-13_11H18M20_3674972_FACE_0_SC.dcm',force=True)\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/filereader.py in dcmread(fp, defer_size, stop_before_pixels, force, specific_tags)\r\n    848     try:\r\n    849         dataset = read_partial(fp, stop_when, defer_size=defer_size,\r\n--> 850                                force=force, specific_tags=specific_tags)\r\n    851     finally:\r\n    852         if not caller_owns_file:\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/filereader.py in read_partial(fileobj, stop_when, defer_size, force, specific_tags)\r\n    726         dataset = read_dataset(fileobj, is_implicit_VR, is_little_endian,\r\n    727                                stop_when=stop_when, defer_size=defer_size,\r\n--> 728                                specific_tags=specific_tags)\r\n    729     except EOFError:\r\n    730         pass  # error already logged in read_dataset\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/filereader.py in read_dataset(fp, is_implicit_VR, is_little_endian, bytelength, stop_when, defer_size, parent_encoding, specific_tags)\r\n    361     try:\r\n    362         while (bytelength is None) or (fp.tell() - fp_start < bytelength):\r\n--> 363             raw_data_element = next(de_gen)\r\n    364             # Read data elements. Stop on some errors, but return what was read\r\n    365             tag = raw_data_element.tag\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/filereader.py in data_element_generator(fp, is_implicit_VR, is_little_endian, stop_when, defer_size, encoding, specific_tags)\r\n    203                 # Store the encoding value in the generator\r\n    204                 # for use with future elements (SQs)\r\n--> 205                 encoding = convert_encodings(encoding)\r\n    206 \r\n    207             yield RawDataElement(tag, VR, length, value, value_tell,\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/charset.py in convert_encodings(encodings)\r\n    626         except KeyError:\r\n    627             py_encodings.append(\r\n--> 628                 _python_encoding_for_corrected_encoding(encoding))\r\n    629 \r\n    630     if len(encodings) > 1:\r\n\r\n~/anaconda3/lib/python3.7/site-packages/pydicom/charset.py in _python_encoding_for_corrected_encoding(encoding)\r\n    664     # fallback: assume that it is already a python encoding\r\n    665     try:\r\n--> 666         codecs.lookup(encoding)\r\n    667         return encoding\r\n    668     except LookupError:\r\n\r\nValueError: embedded null character\r\n#### Steps/Code to Reproduce\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\nimport pydicom\r\ndcm = pydicom.dcmread('/home/zhuzhemin/XrayKeyPoints/data/10-31-13_11H18M20_3674972_FACE_0_SC.dcm')\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\nNo error\r\nI used dcmread function in matlab to read the same file and it was ok. So it should not be the problem of the file.\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\nError: Embedded Null character\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\n1.3.0\r\n\r\n<!-- Thanks for contributing! -->\r\n\n", "hints_text": "Well, this is invalid DICOM, so strictly speaking this is not a bug, but we can probably just ignore the null character and change the error into a warning (if `config.enforce_valid_values` is not set).\n> Well, this is invalid DICOM, so strictly speaking this is not a bug, but we can probably just ignore the null character and change the error into a warning (if `config.enforce_valid_values` is not set).\r\n\r\nWe could do that, but given that this is the first time seeing an error like this, I don't think it is worth the effort.  Instead, I think using [data_element_callback](https://pydicom.github.io/pydicom/stable/api_ref.html#pydicom.config.data_element_callback) is better suited - example of how to use in my comment [here](https://github.com/pydicom/pydicom/issues/820#issuecomment-473500989).  Not quite the same situation, but could be adapted to replace the bad value.\nI'm in favour of ignoring it with a warning/exception, its not really any different from all the other non-conformant fixes we have.\nI guess a terminating null is probably common enough carry-over from C that we could be tolerant to reading that.  I'm worried about chasing invalid DICOM endlessly, though, when there is an existing facility for people to filter any kind of invalid values out, and every extra check hits performance, if only a small amount.  Perhaps we just need to make that `util.fixer` code easier to use, maybe a `pydicom.config` setting to set characters to strip as a one-liner before reading a file.\nYeah, I see your point... the code gets messier each time we add a workaround for another incarnation of invalid DICOM. \r\nThis concrete exception happens during Python encoding lookup (as a fallback to check if the encoding is already a Python encoding), where we only expect a `LookupError`. The actual fix, if we would add one, would have to happen earlier (like stripping any trailing zero byte from string values), but I'm not sure if that's worth it.\r\nIt may be interesting to understand where this comes from, as I doubt any major DICOM library or modality would have written such a value, and if this may happen elsewhere. I would also check if dcmtk handles this - if they do, I would be more inclined to add a fix. \nOk, dcmdump just ignores it (I replaced the last '0' by a a zero):\r\n```(0008,0005) CS [ISO_IR 10 ]                             #  10, 1 SpecificCharacterSet```\r\n\n> I'm worried about chasing invalid DICOM endlessly, though, when there is an existing facility for people to filter any kind of invalid values out, and every extra check hits performance, if only a small amount.\r\n\r\nFair enough. Perhaps we could update fixer (if needed) and have a 'library' of available fixes instead and make sure its all documented. That way in the future we can just add to the library instead of adding a workaround to the codebase.\nI read your answers but cannot adapt the callback to my own situation. I really do not understand how dicom is organized and parsed.  All I want is to grab the pixel array and do something. Would Appreciate if you could update the version and fix it.\nAs @mrbean-bremen said, this can't currently be fixed using `config.data_element_callback` because that gets called after reading while the exception is raised during reading (because character set is special). We'd need an earlier hook if we want to go the fixer route.\nLet's try this again... quick workaround.\r\n```python\r\nimport codecs\r\nimport re\r\n\r\nfrom pydicom import charset\r\nfrom pydicom import dcmread\r\n\r\ndef _python_encoding_for_corrected_encoding(encoding):\r\n    encoding = encoding.strip(' \\r\\t\\n\\0')\r\n\r\n    # standard encodings\r\n    patched = None\r\n    if re.match('^ISO[^_]IR', encoding) is not None:\r\n        patched = 'ISO_IR' + encoding[6:]\r\n    # encodings with code extensions\r\n    elif re.match('^(?=ISO.2022.IR.)(?!ISO 2022 IR )',\r\n                  encoding) is not None:\r\n        patched = 'ISO 2022 IR ' + encoding[12:]\r\n\r\n    if patched:\r\n        # handle encoding patched for common spelling errors\r\n        try:\r\n            py_encoding = python_encoding[patched]\r\n            charset._warn_about_invalid_encoding(encoding, patched)\r\n            return py_encoding\r\n        except KeyError:\r\n            charset._warn_about_invalid_encoding(encoding)\r\n            return default_encoding\r\n\r\n    # fallback: assume that it is already a python encoding\r\n    try:\r\n        codecs.lookup(encoding)\r\n        return encoding\r\n    except LookupError:\r\n        charset._warn_about_invalid_encoding(encoding)\r\n        return default_encoding\r\n\r\ncharset._python_encoding_for_corrected_encoding = _python_encoding_for_corrected_encoding\r\n\r\nds = dcmread(...)\r\n```\nInterestingly we actually do handle charset values that have trailing padding `\\x00`, through `valuerep.MultiString`, what we don't do is handle values that end in more than one null (i.e. `\\x00\\x00`) which is the case here.\r\n\r\nWe could change [this line](https://github.com/pydicom/pydicom/blob/a0300a69a1da1626caef0d9738cff29b17ce79cc/pydicom/valuerep.py#L548)\r\n```python\r\nwhile val.endswith(' ') or val.endswith('\\x00'):\r\n    val = val[:-1]\r\n```\nHm, that looks like an easy fix without impact - I wasn't aware of this. The only question is - should we warn in this case?\r\nThat being said, I still think that your proposal to add a repository of available fixes is a good one, even if not applicable to this issue.", "created_at": "2019-09-05T18:25:50Z", "version": "1.3", "FAIL_TO_PASS": ["pydicom/tests/test_values.py::TestConvertText::test_value_ending_with_padding"], "PASS_TO_PASS": ["pydicom/tests/test_values.py::TestConvertTag::test_big_endian", "pydicom/tests/test_values.py::TestConvertTag::test_little_endian", "pydicom/tests/test_values.py::TestConvertTag::test_offset", "pydicom/tests/test_values.py::TestConvertAE::test_strip_blanks", "pydicom/tests/test_values.py::TestConvertText::test_single_value", "pydicom/tests/test_values.py::TestConvertText::test_multi_value", "pydicom/tests/test_values.py::TestConvertText::test_single_value_with_backslash", "pydicom/tests/test_values.py::TestConvertText::test_single_value_with_delimiters", "pydicom/tests/test_values.py::TestConvertAT::test_big_endian", "pydicom/tests/test_values.py::TestConvertAT::test_little_endian", "pydicom/tests/test_values.py::TestConvertAT::test_empty_bytestring", "pydicom/tests/test_values.py::TestConvertDA::test_big_endian", "pydicom/tests/test_values.py::TestConvertDA::test_little_endian", "pydicom/tests/test_values.py::TestConvertDA::test_empty_bytestring", "pydicom/tests/test_values.py::TestConvertValue::test_convert_value_raises"], "environment_setup_commit": "7241f5d9db0de589b230bb84212fbb643a7c86c3"}, {"repo": "pydicom/pydicom", "instance_id": "pydicom__pydicom-809", "base_commit": "356a51ab4bc54fd18950041ebc44dbfa1a425a10", "patch": "diff --git a/pydicom/dataset.py b/pydicom/dataset.py\n--- a/pydicom/dataset.py\n+++ b/pydicom/dataset.py\n@@ -197,6 +197,9 @@ def __init__(self, *args, **kwargs):\n         self.is_little_endian = None\n         self.is_implicit_VR = None\n \n+        # the parent data set, if this dataset is a sequence item\n+        self.parent = None\n+\n     def __enter__(self):\n         \"\"\"Method invoked on entry to a with statement.\"\"\"\n         return self\n@@ -555,7 +558,13 @@ def __getattr__(self, name):\n             # Try the base class attribute getter (fix for issue 332)\n             return super(Dataset, self).__getattribute__(name)\n         else:\n-            return self[tag].value\n+            data_elem = self[tag]\n+            value = data_elem.value\n+            if data_elem.VR == 'SQ':\n+                # let a sequence know its parent dataset, as sequence items\n+                # may need parent dataset tags to resolve ambiguous tags\n+                value.parent = self\n+            return value\n \n     @property\n     def _character_set(self):\n@@ -1165,6 +1174,11 @@ def __setattr__(self, name, value):\n                 # don't have this tag yet->create the data_element instance\n                 VR = dictionary_VR(tag)\n                 data_element = DataElement(tag, VR, value)\n+                if VR == 'SQ':\n+                    # let a sequence know its parent dataset to pass it\n+                    # to its items, who may need parent dataset tags\n+                    # to resolve ambiguous tags\n+                    data_element.parent = self\n             else:\n                 # already have this data_element, just changing its value\n                 data_element = self[tag]\ndiff --git a/pydicom/filewriter.py b/pydicom/filewriter.py\n--- a/pydicom/filewriter.py\n+++ b/pydicom/filewriter.py\n@@ -69,6 +69,9 @@ def _correct_ambiguous_vr_element(elem, ds, is_little_endian):\n         # US if PixelRepresentation value is 0x0000, else SS\n         #   For references, see the list at\n         #   https://github.com/darcymason/pydicom/pull/298\n+        # PixelRepresentation is usually set in the root dataset\n+        while 'PixelRepresentation' not in ds and ds.parent:\n+            ds = ds.parent\n         if ds.PixelRepresentation == 0:\n             elem.VR = 'US'\n             byte_type = 'H'\ndiff --git a/pydicom/sequence.py b/pydicom/sequence.py\n--- a/pydicom/sequence.py\n+++ b/pydicom/sequence.py\n@@ -43,6 +43,9 @@ def __init__(self, iterable=None):\n         if isinstance(iterable, Dataset):\n             raise TypeError('The Sequence constructor requires an iterable')\n \n+        # the parent dataset\n+        self._parent = None\n+\n         # If no inputs are provided, we create an empty Sequence\n         if not iterable:\n             iterable = list()\n@@ -50,6 +53,24 @@ def __init__(self, iterable=None):\n         # validate_dataset is used as a pseudo type_constructor\n         super(Sequence, self).__init__(validate_dataset, iterable)\n \n+    @property\n+    def parent(self):\n+        \"\"\"Return the parent dataset.\"\"\"\n+        return self._parent\n+\n+    @parent.setter\n+    def parent(self, value):\n+        \"\"\"Set the parent dataset and pass it to all items.\"\"\"\n+        if value != self._parent:\n+            self._parent = value\n+            for item in self._list:\n+                item.parent = self._parent\n+\n+    def __setitem__(self, i, val):\n+        \"\"\"Set the parent dataset to the new sequence item\"\"\"\n+        super(Sequence, self).__setitem__(i, val)\n+        val.parent = self._parent\n+\n     def __str__(self):\n         \"\"\"String description of the Sequence.\"\"\"\n         lines = [str(x) for x in self]\n", "test_patch": "diff --git a/pydicom/tests/test_filewriter.py b/pydicom/tests/test_filewriter.py\n--- a/pydicom/tests/test_filewriter.py\n+++ b/pydicom/tests/test_filewriter.py\n@@ -864,6 +864,47 @@ def test_write_new_ambiguous(self):\n         assert ds[0x00283002].VR == 'US'\n         assert ds.LUTDescriptor == [1, 0]\n \n+    def test_ambiguous_element_in_sequence_explicit(self):\n+        \"\"\"Test that writing a sequence with an ambiguous element\n+        as explicit transfer syntax works.\"\"\"\n+        # regression test for #804\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.ModalityLUTSequence = [Dataset()]\n+        ds.ModalityLUTSequence[0].LUTDescriptor = [0, 0, 16]\n+        ds.ModalityLUTSequence[0].LUTExplanation = None\n+        ds.ModalityLUTSequence[0].ModalityLUTType = 'US'  # US = unspecified\n+        ds.ModalityLUTSequence[0].LUTData = b'\\x0000\\x149a\\x1f1c\\xc2637'\n+\n+        ds.is_little_endian = True\n+        ds.is_implicit_VR = False\n+        fp = BytesIO()\n+        ds.save_as(fp, write_like_original=True)\n+\n+        ds = dcmread(fp, force=True)\n+        assert 'US' == ds.ModalityLUTSequence[0][0x00283002].VR\n+\n+    def test_ambiguous_element_in_sequence_implicit(self):\n+        \"\"\"Test that reading a sequence with an ambiguous element\n+        from a file with implicit transfer syntax works.\"\"\"\n+        # regression test for #804\n+        ds = Dataset()\n+        ds.PixelRepresentation = 0\n+        ds.ModalityLUTSequence = [Dataset()]\n+        ds.ModalityLUTSequence[0].LUTDescriptor = [0, 0, 16]\n+        ds.ModalityLUTSequence[0].LUTExplanation = None\n+        ds.ModalityLUTSequence[0].ModalityLUTType = 'US'  # US = unspecified\n+        ds.ModalityLUTSequence[0].LUTData = b'\\x0000\\x149a\\x1f1c\\xc2637'\n+\n+        ds.is_little_endian = True\n+        ds.is_implicit_VR = True\n+        fp = BytesIO()\n+        ds.save_as(fp, write_like_original=True)\n+        ds = dcmread(fp, force=True)\n+        # we first have to access the value to trigger correcting the VR\n+        assert 16 == ds.ModalityLUTSequence[0].LUTDescriptor[2]\n+        assert 'US' == ds.ModalityLUTSequence[0][0x00283002].VR\n+\n \n class TestCorrectAmbiguousVRElement(object):\n     \"\"\"Test filewriter.correct_ambiguous_vr_element\"\"\"\n", "problem_statement": "\"Printing\" of certain dicom files fails once, but works the second time\n<!-- Instructions For Filing a Bug: https://github.com/pydicom/pydicom/blob/master/CONTRIBUTING.md#filing-bugs -->\r\n\r\n#### Description\r\n\"Printing\" of certain dicom files (see [example](https://github.com/pydicom/pydicom/files/2865551/dicom_exception.zip)) fails once, but not the second time\r\n\r\n#### Steps/Code to Reproduce\r\n```python\r\nfrom pydicom import read_file\r\n\r\na = read_file('...')\r\nprint(a)\r\n# triggers exception: AttributeError: With tag (0028, 3000) got exception: Failed to resolve ambiguous VR for tag (0028, 3002): 'Dataset' object has no attribute 'PixelRepresentation'\r\n\r\n# try same thing again...\r\nprint(a)\r\n# just works...\r\n```\r\n\r\n#### Versions\r\nBehaviour as described above at least on:\r\n```\r\nLinux-4.18.0-15-generic-x86_64-with-Ubuntu-18.10-cosmic\r\n('Python', '2.7.15+ (default, Oct  2 2018, 22:12:08) \\n[GCC 8.2.0]')\r\n('numpy', '1.14.5')\r\n('pydicom', '1.3.0.dev0')\r\n```\r\nand\r\n\r\n\r\n```\r\n('pydicom', '1.2.2')\r\n```\r\n\r\nWorks as expected on:\r\n```\r\nLinux-4.18.0-15-generic-x86_64-with-Ubuntu-18.10-cosmic\r\n('Python', '2.7.15+ (default, Oct  2 2018, 22:12:08) \\n[GCC 8.2.0]')\r\n('pydicom', '1.0.1')\r\n```\n", "hints_text": "Occurs because Pixel Representation is in the top level-dataset while the ambiguous element is in a sequence.\r\n\r\nRegression test:\r\n```python\r\nfrom pydicom.dataset import Dataset\r\n\r\nds = Dataset()\r\nds.PixelRepresentation = 0\r\nds.ModalityLUTSequence = [Dataset()]\r\nds.ModalityLUTSequence[0].LUTDescriptor = [0, 0, 16]\r\nds.ModalityLUTSequence[0].LUTExplanation = None\r\nds.ModalityLUTSequence[0].ModalityLUTType = 'US'  # US = unspecified\r\nds.ModalityLUTSequence[0].LUTData = b'\\x0000\\x149a\\x1f1c\\c2637'\r\n\r\nds.is_little_endian= True\r\nds.is_implicit_VR = False\r\nds.save_as('test.dcm')\r\n```\r\n\r\nThe reason it works the second time is the ambiguous VR correction only gets used during the initial decoding (pydicom uses deferred decoding which is triggered by the first `print()`).\r\n\r\nThis might be a bit tricky to fix elegantly...\nOne thing we should probably change is to warn rather than raise if ambiguous correction fails during decoding. Should still raise if occurs during encoding\n> This might be a bit tricky to fix elegantly...\r\n\r\nYes... we have to support the cases where the tag needed to resolve the ambiguity is in the sequence item, or in any parent dataset (for nested sequences). Having the parent dataset as a member in the dataset would allow this, but this would also mean, that it has always to be set on creating a sequence item... not sure if this is a good idea. \nAnother option is to pass the function a dict/Dataset containing elements required for resolving ambiguity (if they're present)", "created_at": "2019-03-04T20:14:54Z", "version": "1.2", "FAIL_TO_PASS": ["pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_explicit", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_ambiguous_element_in_sequence_implicit"], "PASS_TO_PASS": ["pydicom/tests/test_filewriter.py::TestWriteFile::testRTPlan", "pydicom/tests/test_filewriter.py::TestWriteFile::testRTDose", "pydicom/tests/test_filewriter.py::TestWriteFile::testCT", "pydicom/tests/test_filewriter.py::TestWriteFile::testMR", "pydicom/tests/test_filewriter.py::TestWriteFile::testUnicode", "pydicom/tests/test_filewriter.py::TestWriteFile::testMultiPN", "pydicom/tests/test_filewriter.py::TestWriteFile::testJPEG2000", "pydicom/tests/test_filewriter.py::TestWriteFile::testwrite_short_uid", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_no_ts", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_double_filemeta", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_ffff_ffff", "pydicom/tests/test_filewriter.py::TestWriteFile::test_write_removes_grouplength", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testRTPlan", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testRTDose", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testCT", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testMR", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testUnicode", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testMultiPN", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testJPEG2000", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::testwrite_short_uid", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_no_ts", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_double_filemeta", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_ffff_ffff", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_write_removes_grouplength", "pydicom/tests/test_filewriter.py::TestScratchWriteDateTime::test_multivalue_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_empty_AT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_empty_LO", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_DA", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_TM", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_TM", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_DT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_multi_DT", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_ascii_vr_with_padding", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OD_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OD_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OL_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_OL_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UC_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UC_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UR_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UR_explicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_UN_implicit_little", "pydicom/tests/test_filewriter.py::TestWriteDataElement::test_write_unknown_vr_raises", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_representation_vm_one", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_representation_vm_three", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_pixel_data", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_waveform_bits_allocated", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_lut_descriptor", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_overlay", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_sequence", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVR::test_write_new_ambiguous", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_not_ambiguous", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_not_ambiguous_raw_data_element", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_correct_ambiguous_data_element", "pydicom/tests/test_filewriter.py::TestCorrectAmbiguousVRElement::test_correct_ambiguous_raw_data_element", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_raises", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_little_endian", "pydicom/tests/test_filewriter.py::TestWriteAmbiguousVR::test_write_explicit_vr_big_endian", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_preamble_default", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_preamble_custom", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_no_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_none_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_bad_preamble", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_prefix", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_prefix_none", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_ds_changed", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raw_elements_preserved_implicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raw_elements_preserved_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_implicit_to_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_dataset", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_dataset_with_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_implicit_to_explicit_vr_using_destination", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_explicit_to_implicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_big_to_little_endian", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_little_to_big_endian", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_changed_character_set", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_private_tag_vr_from_implicit_data", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_convert_rgb_from_implicit_to_explicit_vr", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_not_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_transfer_syntax_raises", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_media_storage_sop_class_uid_added", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_write_no_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_raise_no_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_add_file_meta", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_standard", "pydicom/tests/test_filewriter.py::TestWriteToStandard::test_commandset_no_written", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_bad_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_missing_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_group_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_group_length_updated", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_version", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_implementation_version_name_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_implementation_class_uid_length", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoToStandard::test_filelike_position", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_default", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_custom", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_no_preamble", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_ds_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_file_meta_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_filemeta_dataset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_commandset_filemeta", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_preamble_commandset_filemeta", "pydicom/tests/test_filewriter.py::TestWriteNonStandard::test_read_write_identical", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_transfer_syntax_not_added", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_bad_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_missing_elements", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_group_length_updated", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_filelike_position", "pydicom/tests/test_filewriter.py::TestWriteFileMetaInfoNonStandard::test_meta_unchanged", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_empty_value", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_list", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_singleton", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_exception", "pydicom/tests/test_filewriter.py::TestWriteNumbers::test_write_big_endian", "pydicom/tests/test_filewriter.py::TestWritePN::test_no_encoding", "pydicom/tests/test_filewriter.py::TestWritePN::test_single_byte_multi_charset_groups", "pydicom/tests/test_filewriter.py::TestWritePN::test_single_byte_multi_charset_values", "pydicom/tests/test_filewriter.py::TestWriteText::test_no_encoding", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_byte_multi_charset_text", "pydicom/tests/test_filewriter.py::TestWriteText::test_encode_mixed_charsets_text", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_byte_multi_charset_text_multivalue", "pydicom/tests/test_filewriter.py::TestWriteText::test_invalid_encoding", "pydicom/tests/test_filewriter.py::TestWriteText::test_invalid_encoding_enforce_standard", "pydicom/tests/test_filewriter.py::TestWriteText::test_single_value_with_delimiters", "pydicom/tests/test_filewriter.py::TestWriteDT::test_format_dt", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_little_endian_correct_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_big_endian_correct_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_little_endian_incorrect_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_big_endian_incorrect_data", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_writing_to_gzip", "pydicom/tests/test_filewriter.py::TestWriteUndefinedLengthPixelData::test_writing_too_big_data_in_explicit_encoding"], "environment_setup_commit": "b4b44acbf1ddcaf03df16210aac46cb3a8acd6b9"}, {"repo": "pvlib/pvlib-python", "instance_id": "pvlib__pvlib-python-1073", "base_commit": "b105021f7c1a47f888363af5585083fc27aefd4c", "patch": "diff --git a/pvlib/modelchain.py b/pvlib/modelchain.py\n--- a/pvlib/modelchain.py\n+++ b/pvlib/modelchain.py\n@@ -323,9 +323,9 @@ class ModelChain:\n         as the first argument to a user-defined function.\n \n     temperature_model: None, str or function, default None\n-        Valid strings are 'sapm', 'pvsyst', and 'faiman'. The ModelChain\n-        instance will be passed as the first argument to a user-defined\n-        function.\n+        Valid strings are 'sapm', 'pvsyst', 'faiman', and 'fuentes'.\n+        The ModelChain instance will be passed as the first argument to a\n+        user-defined function.\n \n     losses_model: str or function, default 'no_loss'\n         Valid strings are 'pvwatts', 'no_loss'. The ModelChain instance\n@@ -866,6 +866,8 @@ def temperature_model(self, model):\n                 self._temperature_model = self.pvsyst_temp\n             elif model == 'faiman':\n                 self._temperature_model = self.faiman_temp\n+            elif model == 'fuentes':\n+                self._temperature_model = self.fuentes_temp\n             else:\n                 raise ValueError(model + ' is not a valid temperature model')\n             # check system.temperature_model_parameters for consistency\n@@ -891,6 +893,8 @@ def infer_temperature_model(self):\n             return self.pvsyst_temp\n         elif {'u0', 'u1'} <= params:\n             return self.faiman_temp\n+        elif {'noct_installed'} <= params:\n+            return self.fuentes_temp\n         else:\n             raise ValueError('could not infer temperature model from '\n                              'system.temperature_module_parameters {}.'\n@@ -914,6 +918,12 @@ def faiman_temp(self):\n             self.weather['wind_speed'])\n         return self\n \n+    def fuentes_temp(self):\n+        self.cell_temperature = self.system.fuentes_celltemp(\n+            self.total_irrad['poa_global'], self.weather['temp_air'],\n+            self.weather['wind_speed'])\n+        return self\n+\n     @property\n     def losses_model(self):\n         return self._losses_model\ndiff --git a/pvlib/pvsystem.py b/pvlib/pvsystem.py\n--- a/pvlib/pvsystem.py\n+++ b/pvlib/pvsystem.py\n@@ -609,6 +609,46 @@ def faiman_celltemp(self, poa_global, temp_air, wind_speed=1.0):\n         return temperature.faiman(poa_global, temp_air, wind_speed,\n                                   **kwargs)\n \n+    def fuentes_celltemp(self, poa_global, temp_air, wind_speed):\n+        \"\"\"\n+        Use :py:func:`temperature.fuentes` to calculate cell temperature.\n+\n+        Parameters\n+        ----------\n+        poa_global : pandas Series\n+            Total incident irradiance [W/m^2]\n+\n+        temp_air : pandas Series\n+            Ambient dry bulb temperature [C]\n+\n+        wind_speed : pandas Series\n+            Wind speed [m/s]\n+\n+        Returns\n+        -------\n+        temperature_cell : pandas Series\n+            The modeled cell temperature [C]\n+\n+        Notes\n+        -----\n+        The Fuentes thermal model uses the module surface tilt for convection\n+        modeling. The SAM implementation of PVWatts hardcodes the surface tilt\n+        value at 30 degrees, ignoring whatever value is used for irradiance\n+        transposition. This method defaults to using ``self.surface_tilt``, but\n+        if you want to match the PVWatts behavior, you can override it by\n+        including a ``surface_tilt`` value in ``temperature_model_parameters``.\n+        \"\"\"\n+        # default to using the PVSystem attribute, but allow user to\n+        # override with a custom surface_tilt value\n+        kwargs = {'surface_tilt': self.surface_tilt}\n+        temp_model_kwargs = _build_kwargs([\n+            'noct_installed', 'module_height', 'wind_height', 'emissivity',\n+            'absorption', 'surface_tilt', 'module_width', 'module_length'],\n+            self.temperature_model_parameters)\n+        kwargs.update(temp_model_kwargs)\n+        return temperature.fuentes(poa_global, temp_air, wind_speed,\n+                                   **kwargs)\n+\n     def first_solar_spectral_loss(self, pw, airmass_absolute):\n \n         \"\"\"\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@@ -138,6 +138,18 @@ def pvwatts_dc_pvwatts_ac_pvsyst_temp_system():\n     return system\n \n \n+@pytest.fixture(scope=\"function\")\n+def pvwatts_dc_pvwatts_ac_fuentes_temp_system():\n+    module_parameters = {'pdc0': 220, 'gamma_pdc': -0.003}\n+    temp_model_params = {'noct_installed': 45}\n+    inverter_parameters = {'pdc0': 220, 'eta_inv_nom': 0.95}\n+    system = PVSystem(surface_tilt=32.2, surface_azimuth=180,\n+                      module_parameters=module_parameters,\n+                      temperature_model_parameters=temp_model_params,\n+                      inverter_parameters=inverter_parameters)\n+    return system\n+\n+\n @pytest.fixture(scope=\"function\")\n def system_no_aoi(cec_module_cs5p_220m, sapm_temperature_cs5p_220m,\n                   cec_inverter_parameters):\n@@ -317,6 +329,23 @@ def test_run_model_with_weather_faiman_temp(sapm_dc_snl_ac_system, location,\n     assert not mc.ac.empty\n \n \n+def test_run_model_with_weather_fuentes_temp(sapm_dc_snl_ac_system, location,\n+                                             weather, mocker):\n+    weather['wind_speed'] = 5\n+    weather['temp_air'] = 10\n+    sapm_dc_snl_ac_system.temperature_model_parameters = {\n+        'noct_installed': 45\n+    }\n+    mc = ModelChain(sapm_dc_snl_ac_system, location)\n+    mc.temperature_model = 'fuentes'\n+    m_fuentes = mocker.spy(sapm_dc_snl_ac_system, 'fuentes_celltemp')\n+    mc.run_model(weather)\n+    assert m_fuentes.call_count == 1\n+    assert_series_equal(m_fuentes.call_args[0][1], weather['temp_air'])\n+    assert_series_equal(m_fuentes.call_args[0][2], weather['wind_speed'])\n+    assert not mc.ac.empty\n+\n+\n def test_run_model_tracker(sapm_dc_snl_ac_system, location, weather, mocker):\n     system = SingleAxisTracker(\n         module_parameters=sapm_dc_snl_ac_system.module_parameters,\n@@ -479,14 +508,16 @@ def test_infer_spectral_model(location, sapm_dc_snl_ac_system,\n \n \n @pytest.mark.parametrize('temp_model', [\n-    'sapm_temp', 'faiman_temp', 'pvsyst_temp'])\n+    'sapm_temp', 'faiman_temp', 'pvsyst_temp', 'fuentes_temp'])\n def test_infer_temp_model(location, sapm_dc_snl_ac_system,\n                           pvwatts_dc_pvwatts_ac_pvsyst_temp_system,\n                           pvwatts_dc_pvwatts_ac_faiman_temp_system,\n+                          pvwatts_dc_pvwatts_ac_fuentes_temp_system,\n                           temp_model):\n     dc_systems = {'sapm_temp': sapm_dc_snl_ac_system,\n                   'pvsyst_temp': pvwatts_dc_pvwatts_ac_pvsyst_temp_system,\n-                  'faiman_temp': pvwatts_dc_pvwatts_ac_faiman_temp_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',\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@@ -360,6 +360,50 @@ def test_PVSystem_faiman_celltemp(mocker):\n     assert_allclose(out, 56.4, atol=1)\n \n \n+def test_PVSystem_fuentes_celltemp(mocker):\n+    noct_installed = 45\n+    temp_model_params = {'noct_installed': noct_installed}\n+    system = pvsystem.PVSystem(temperature_model_parameters=temp_model_params)\n+    spy = mocker.spy(temperature, 'fuentes')\n+    index = pd.date_range('2019-01-01 11:00', freq='h', periods=3)\n+    temps = pd.Series(25, index)\n+    irrads = pd.Series(1000, index)\n+    winds = pd.Series(1, index)\n+    out = system.fuentes_celltemp(irrads, temps, winds)\n+    assert_series_equal(spy.call_args[0][0], irrads)\n+    assert_series_equal(spy.call_args[0][1], temps)\n+    assert_series_equal(spy.call_args[0][2], winds)\n+    assert spy.call_args[1]['noct_installed'] == noct_installed\n+    assert_series_equal(out, pd.Series([52.85, 55.85, 55.85], index,\n+                                       name='tmod'))\n+\n+\n+def test_PVSystem_fuentes_celltemp_override(mocker):\n+    # test that the surface_tilt value in the cell temp calculation can be\n+    # overridden but defaults to the surface_tilt attribute of the PVSystem\n+    spy = mocker.spy(temperature, 'fuentes')\n+\n+    noct_installed = 45\n+    index = pd.date_range('2019-01-01 11:00', freq='h', periods=3)\n+    temps = pd.Series(25, index)\n+    irrads = pd.Series(1000, index)\n+    winds = pd.Series(1, index)\n+\n+    # uses default value\n+    temp_model_params = {'noct_installed': noct_installed}\n+    system = pvsystem.PVSystem(temperature_model_parameters=temp_model_params,\n+                               surface_tilt=20)\n+    system.fuentes_celltemp(irrads, temps, winds)\n+    assert spy.call_args[1]['surface_tilt'] == 20\n+\n+    # can be overridden\n+    temp_model_params = {'noct_installed': noct_installed, 'surface_tilt': 30}\n+    system = pvsystem.PVSystem(temperature_model_parameters=temp_model_params,\n+                               surface_tilt=20)\n+    system.fuentes_celltemp(irrads, temps, winds)\n+    assert spy.call_args[1]['surface_tilt'] == 30\n+\n+\n def test__infer_temperature_model_params():\n     system = pvsystem.PVSystem(module_parameters={},\n                                racking_model='open_rack',\n", "problem_statement": "expose pvlib.temperature.fuentes in PVSystem and ModelChain\nFollow up to #1032 and #1037 \n", "hints_text": "", "created_at": "2020-10-01T01:40:51Z", "version": "0.7", "FAIL_TO_PASS": ["pvlib/tests/test_modelchain.py::test_run_model_with_weather_fuentes_temp", "pvlib/tests/test_modelchain.py::test_infer_temp_model[fuentes_temp]", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_fuentes_celltemp_override"], "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_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_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_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", "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_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_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_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_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_sapm_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_sapm_celltemp_kwargs", "pvlib/tests/test_pvsystem.py::test_PVSystem_pvsyst_celltemp", "pvlib/tests/test_pvsystem.py::test_PVSystem_faiman_celltemp", "pvlib/tests/test_pvsystem.py::test__infer_temperature_model_params", "pvlib/tests/test_pvsystem.py::test_calcparams_desoto", "pvlib/tests/test_pvsystem.py::test_calcparams_cec", "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_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_snlinverter", "pvlib/tests/test_pvsystem.py::test_PVSystem_creation", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_aoi", "pvlib/tests/test_pvsystem.py::test_PVSystem_get_irradiance", "pvlib/tests/test_pvsystem.py::test_PVSystem_localize_with_location", "pvlib/tests/test_pvsystem.py::test_PVSystem_localize_with_latlon", "pvlib/tests/test_pvsystem.py::test_PVSystem___repr__", "pvlib/tests/test_pvsystem.py::test_PVSystem_localize___repr__", "pvlib/tests/test_pvsystem.py::test_LocalizedPVSystem_creation", "pvlib/tests/test_pvsystem.py::test_LocalizedPVSystem___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_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_combine_loss_factors", "pvlib/tests/test_pvsystem.py::test_deprecated_09"], "environment_setup_commit": "6e5148f59c5050e8f7a0084b7ae39e93b80f72e6"}]