{
  "id": "astropy__astropy-14365",
  "question": "ascii.qdp Table format assumes QDP commands are upper case\n### Description\n\nascii.qdp assumes that commands in a QDP file are upper case, for example, for errors they must be \"READ SERR 1 2\" whereas QDP itself is not case sensitive and case use \"read serr 1 2\". \r\n\r\nAs many QDP files are created by hand, the expectation that all commands be all-caps should be removed.\n\n### Expected behavior\n\nThe following qdp file should read into a `Table` with errors, rather than crashing.\r\n```\r\nread serr 1 2 \r\n1 0.5 1 0.5\r\n```\n\n### How to Reproduce\n\nCreate a QDP file:\r\n```\r\n> cat > test.qdp\r\nread serr 1 2 \r\n1 0.5 1 0.5\r\n<EOF>\r\n\r\n > python\r\nPython 3.10.9 (main, Dec  7 2022, 02:03:23) [Clang 13.0.0 (clang-1300.0.29.30)] on darwin\r\nType \"help\", \"copyright\", \"credits\" or \"license\" for more information.\r\n>>> from astropy.table import Table\r\n>>> Table.read('test.qdp',format='ascii.qdp')\r\nWARNING: table_id not specified. Reading the first available table [astropy.io.ascii.qdp]\r\nTraceback (most recent call last):\r\n...\r\n    raise ValueError(f'Unrecognized QDP line: {line}')\r\nValueError: Unrecognized QDP line: read serr 1 2\r\n```\r\n\r\nRunning \"qdp test.qdp\" works just fine.\r\n\n\n### Versions\n\nPython 3.10.9 (main, Dec  7 2022, 02:03:23) [Clang 13.0.0 (clang-1300.0.29.30)]\r\nastropy 5.1\r\nNumpy 1.24.1\r\npyerfa 2.0.0.1\r\nScipy 1.10.0\r\nMatplotlib 3.6.3\r\n\n",
  "answer": "{\"task_id\": \"ddd23d94-5c12-478c-b408-00d998ce1e36\", \"workflow_run_id\": \"3fca2082-8a85-42e4-b6e7-0975bab209e7\", \"data\": {\"id\": \"3fca2082-8a85-42e4-b6e7-0975bab209e7\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/astropy/io/ascii/qdp.py b/astropy/io/ascii/qdp.py\\\\nindex 1234567..4567890 100644\\\\n--- a/astropy/io/ascii/qdp.py\\\\n+++ b/astropy/io/ascii/qdp.py\\\\n@@ -45,8 +45,10 @@ def _read_qdp(filename):\\\\n     # read the file and handle commands\\\\n     with open(filename, 'r') as f:\\\\n         for line in f:\\\\n-            line_tokens = line.strip().split()\\\\n+            stripped_line = line.strip()\\\\n+            line_tokens = stripped_line.split()\\\\n             if not line_tokens:\\\\n                 continue\\\\n-            if line_tokens[0] == 'READ':\\\\n+\\\\n+            command = line_tokens[0].upper()\\\\n+            if command == 'READ':\\\\n                 # Process the read command\\\\n\\\\n                 # Check for error specification\\\\n@@ -50,7 +52,7 @@ def _read_qdp(filename):\\\\n\\\\n                     if line_tokens[1].upper() == 'SERR':\\\\n                         error_columns = list(map(int, line_tokens[2:]))\\\\n                         continue\\\\n-            raise ValueError(f'Unrecognized QDP line: {line}')\\\\n+            raise ValueError(f'Unrecognized QDP line: {stripped_line}')\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly addresses the issue by converting only the command part of the line to uppercase, ensuring that the rest of the line remains unchanged. This minimizes the risk of altering data unintentionally and aligns well with coding best practices.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/astropy/io/ascii/qdp.py b/astropy/io/ascii/qdp.py\\\\nindex 1234567..abcdef0 100644\\\\n--- a/astropy/io/ascii/qdp.py\\\\n+++ b/astropy/io/ascii/qdp.py\\\\n@@ -30,7 +30,7 @@ class QDP:\\\\n\\\\n     def read(self, filename):\\\\n         # Delegate to _read_qdp\\\\n-        data, meta = _read_qdp(filename)\\\\n+        data, meta = _read_qdp(filename.lower())\\\\n         return data, meta\\\\n\\\\n\\\\ndef _read_qdp(filename):\\\\n@@ -45,8 +45,13 @@ def _read_qdp(filename):\\\\n     # read the file and handle commands\\\\n     with open(filename, 'r') as f:\\\\n         for line in f:\\\\n-            line_tokens = line.strip().split()\\\\n-            \\\\n+            stripped_line = line.strip()\\\\n+            line_upper = stripped_line.upper()\\\\n+            line_tokens = line_upper.split()\\\\n+\\\\n             if not line_tokens:\\\\n                 continue\\\\n\\\\n-            if line_tokens[0] == 'READ':\\\\n\\\\n+            # Convert only the command to uppercase\\\\n+            command, *args = line_tokens\\\\n+            command = command.upper()\\\\n+            if command == 'READ':\\\\n                 # Process read...\\\\n@@ -50,7 +55,7 @@ def _read_qdp(filename):\\\\n                 if args[0].upper() == 'SERR':\\\\n                     error_columns = list(map(int, args[1:]))\\\\n                     continue\\\\n\\\\n-            raise ValueError(f'Unrecognized QDP line: {line}')\\\\n+            raise ValueError(f'Unrecognized QDP line: {stripped_line}')\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue by converting the command to uppercase, but it introduces unnecessary complexity by converting the entire line to uppercase initially. This could potentially affect data processing if not handled carefully.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/astropy/io/ascii/qdp.py b/astropy/io/ascii/qdp.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/astropy/io/ascii/qdp.py\\\\n+++ b/astropy/io/ascii/qdp.py\\\\n@@ -45,7 +45,7 @@ def _read_qdp(filename):\\\\n     # read the file and handle commands\\\\n     with open(filename, 'r') as f:\\\\n         for line in f:\\\\n-            line_tokens = line.strip().split()\\\\n+            line_tokens = line.strip().upper().split()\\\\n             if not line_tokens:\\\\n                 continue\\\\n             if line_tokens[0] == 'READ':\\\\n@@ -50,7 +50,7 @@ def _read_qdp(filename):\\\\n\\\\n                     # Check for error specification\\\\n                     if line_tokens[1] == 'SERR':\\\\n-                        error_columns = list(map(int, line_tokens[2:]))\\\\n+                        error_columns = list(map(int, line.upper().split()[2:]))\\\\n                         continue\\\\n-            raise ValueError(f'Unrecognized QDP line: {line}')\\\\n+            raise ValueError(f'Unrecognized QDP line: {line.upper()}')\\\\n\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch converts the entire line to uppercase, which could lead to unintended consequences if the line contains data that should remain case-sensitive. This approach is less precise and could introduce errors in data processing.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the best choice as it correctly addresses the issue by converting only the command part of the line to uppercase, ensuring minimal risk and alignment with best practices. Patch 3 is a viable alternative but introduces unnecessary complexity. Patch 1 is not recommended due to its potential to alter data unintentionally.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    qdp_input = 'read serr 1 2\\\\\\\\n1 0.5 1 0.5\\\\\\\\n'\\\\n    with open('test.qdp', 'w') as f:\\\\n        f.write(qdp_input)\\\\n    try:\\\\n        from astropy.table import Table\\\\n        Table.read('test.qdp',format='ascii.qdp')\\\\n    except ValueError as e:\\\\n        assert 'Unrecognized QDP line: read serr 1 2' in str(e)\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by creating a QDP file with lowercase commands and attempting to read it with ascii.qdp format\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to raise a ValueError with the message 'Unrecognized QDP line: read serr 1 2'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_lowercase_commands\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_lowercase_commands():\\\\n    lowercase_qdp = 'read serr 1 2\\\\\\\\n1 0.5 1 0.5\\\\\\\\n'\\\\n    from astropy.table import Table\\\\n    table = Table.read(lowercase_qdp, format='ascii.qdp')\\\\n    assert len(table) > 0\\\\n\\\",\\n      \\\"description\\\": \\\"This test handles an edge case of QDP file with all lowercase commands\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to pass without any errors\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_mixedcase_commands\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_mixedcase_commands():\\\\n    mixedcase_qdp = 'ReAd SeRr 1 2\\\\\\\\n1 0.5 1 0.5\\\\\\\\n'\\\\n    from astropy.table import Table\\\\n    table = Table.read(mixedcase_qdp, format='ascii.qdp')\\\\n    assert len(table) > 0\\\\n\\\",\\n      \\\"description\\\": \\\"This test handles an edge case of QDP file with mixed case commands\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to pass without any errors\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    qdp_input = 'TeStInG cAsE SeNsItIvItY 1 2\\\\\\\\n1.0 2.0 3.0 4.0\\\\\\\\n'\\\\n    with open('test.qdp', 'w') as f:\\\\n        f.write(qdp_input)\\\\n    try:\\\\n        from astropy.table import Table\\\\n        table = Table.read('test.qdp', format='ascii.qdp')\\\\n        assert len(table) > 0\\\\n    except ValueError as e:\\\\n        assert 'Unrecognized QDP line: Testing Case Sensitivity 1 2' in str(e)\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates that the patch correctly handles case sensitivity in QDP commands\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to raise a ValueError with the message 'Unrecognized QDP line: Testing Case Sensitivity 1 2' after applying the patch\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to cover reproducing the original issue, testing edge cases with lowercase and mixed case commands, and validating the patch's effectiveness\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 331.883786, \"total_tokens\": 23630, \"total_steps\": 9, \"created_at\": 1753279804, \"finished_at\": 1753280136}}"
}