{
  "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\": \"9af9f1d2-f7d7-4b9e-a290-25fcdfc693e8\", \"workflow_run_id\": \"e2f8ec67-e15f-4bc8-a7d3-d822c1dbbd20\", \"data\": {\"id\": \"e2f8ec67-e15f-4bc8-a7d3-d822c1dbbd20\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -250,20 +250,21 @@ class QDP(basic.Basic):\\n         for line in lines:\\n             line = line.strip()\\n             if not line:\\n                 continue\\n+            line_upper = line.upper()\\n             \\n-            if line.startswith('READ SERR'):\\n+            if line_upper.startswith('READ SERR'):\\n                 # Handle error column specification\\n                 pass\\n-            elif line.startswith('READ TERR'):\\n+            elif line_upper.startswith('READ TERR'):\\n                 # Handle time error specification\\n                 pass\\n-            elif line.startswith('READ'):\\n+            elif line_upper.startswith('READ'):\\n                 # Handle other READ commands\\n                 pass\\n-            elif line.startswith('!'):\\n+            elif line_upper.startswith('!'):\\n                 # Comment line\\n                 continue\\n             else:\\n                 # Data line or unrecognized command\\n-                raise ValueError(f'Unrecognized QDP line: {line}')\\n+                raise ValueError(f'Unrecognized QDP line: {line}')\", \"--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -248,22 +248,23 @@ class QDP(basic.Basic):\\n         \\n         for line in lines:\\n             line = line.strip()\\n             if not line:\\n                 continue\\n+            # Normalize to uppercase for case-insensitive command parsing\\n+            line = line.upper()\\n             \\n             if line.startswith('READ SERR'):\\n                 # Handle error column specification\\n                 pass\\n             elif line.startswith('READ TERR'):\\n                 # Handle time error specification\\n                 pass\\n             elif line.startswith('READ'):\\n                 # Handle other READ commands\\n                 pass\\n             elif line.startswith('!'):\\n                 # Comment line\\n                 continue\\n             else:\\n                 # Data line or unrecognized command\\n                 raise ValueError(f'Unrecognized QDP line: {line}')\", \"--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -1,6 +1,7 @@\\n # Licensed under a 3-clause BSD style license - see LICENSE.rst\\n \\\"\\\"\\\"An extensible ASCII table reader and writer.\\\"\\\"\\\"\\n \\n+import re\\n from astropy.table import Table\\n from . import basic\\n from . import core\\n@@ -248,17 +249,17 @@ class QDP(basic.Basic):\\n         \\n         for line in lines:\\n             line = line.strip()\\n             if not line:\\n                 continue\\n             \\n-            if line.startswith('READ SERR'):\\n+            if re.match(r'^read\\\\s+serr', line, re.IGNORECASE):\\n                 # Handle error column specification\\n                 pass\\n-            elif line.startswith('READ TERR'):\\n+            elif re.match(r'^read\\\\s+terr', line, re.IGNORECASE):\\n                 # Handle time error specification\\n                 pass\\n-            elif line.startswith('READ'):\\n+            elif re.match(r'^read\\\\b', line, re.IGNORECASE):\\n                 # Handle other READ commands\\n                 pass\\n-            elif line.startswith('!'):\\n+            elif line.startswith('!'):\\n                 # Comment line\\n                 continue\\n             else:\", \"--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -240,6 +240,10 @@ class QDP(basic.Basic):\\n         \\n         return header\\n     \\n+    def _is_command(self, line, command):\\n+        \\\"\\\"\\\"Check if line starts with command (case-insensitive).\\\"\\\"\\\"\\n+        return line.upper().startswith(command.upper())\\n+    \\n     def _read_header(self, lines):\\n         \\\"\\\"\\\"Read header information from QDP file.\\\"\\\"\\\"\\n         header = {}\\n@@ -250,17 +254,17 @@ class QDP(basic.Basic):\\n             if not line:\\n                 continue\\n             \\n-            if line.startswith('READ SERR'):\\n+            if self._is_command(line, 'READ SERR'):\\n                 # Handle error column specification\\n                 pass\\n-            elif line.startswith('READ TERR'):\\n+            elif self._is_command(line, 'READ TERR'):\\n                 # Handle time error specification\\n                 pass\\n-            elif line.startswith('READ'):\\n+            elif self._is_command(line, 'READ'):\\n                 # Handle other READ commands\\n                 pass\\n             elif line.startswith('!'):\\n                 # Comment line\\n                 continue\\n             else:\\n                 # Data line or unrecognized command\\n                 raise ValueError(f'Unrecognized QDP line: {line}')\", \"--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -248,17 +248,18 @@ class QDP(basic.Basic):\\n         \\n         for line in lines:\\n             line = line.strip()\\n             if not line:\\n                 continue\\n+            # Convert to uppercase for case-insensitive command parsing\\n+            line_cmd = line.upper()\\n             \\n-            if line.startswith('READ SERR'):\\n+            if line_cmd.startswith('READ SERR'):\\n                 # Handle error column specification\\n                 pass\\n-            elif line.startswith('READ TERR'):\\n+            elif line_cmd.startswith('READ TERR'):\\n                 # Handle time error specification\\n                 pass\\n-            elif line.startswith('READ'):\\n+            elif line_cmd.startswith('READ'):\\n                 # Handle other READ commands\\n                 pass\\n             elif line.startswith('!'):\\n                 # Comment line\\n                 continue\\n@@ -300,10 +301,11 @@ class QDP(basic.Basic):\\n         for line in lines:\\n             line = line.strip()\\n             if not line or line.startswith('!'):\\n                 continue\\n+            # Convert to uppercase for case-insensitive command parsing\\n+            line_cmd = line.upper()\\n             \\n-            if line.startswith('READ'):\\n+            if line_cmd.startswith('READ'):\\n                 # Parse READ command\\n-                parts = line.split()\\n+                parts = line_cmd.split()\\n                 if len(parts) >= 3 and parts[1] in ['SERR', 'TERR']:\\n                     # Handle error specifications\\n                     pass\", \"diff\\n--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -240,6 +240,10 @@ class QDP(basic.Basic):\\n         \\n         return header\\n     \\n+    def _is_command(self, line, command):\\n+        \\\"\\\"\\\"Check if line starts with command (case-insensitive).\\\"\\\"\\\"\\n+        return line.upper().startswith(command.upper())\\n+    \\n     def _read_header(self, lines):\\n         \\\"\\\"\\\"Read header information from QDP file.\\\"\\\"\\\"\\n         header = {}\\n@@ -250,17 +254,17 @@ class QDP(basic.Basic):\\n             if not line:\\n                 continue\\n             \\n-            if line.startswith('READ SERR'):\\n+            if self._is_command(line, 'READ SERR'):\\n                 # Handle error column specification\\n                 pass\\n-            elif line.startswith('READ TERR'):\\n+            elif self._is_command(line, 'READ TERR'):\\n                 # Handle time error specification\\n                 pass\\n-            elif line.startswith('READ'):\\n+            elif self._is_command(line, 'READ'):\\n                 # Handle other READ commands\\n                 pass\\n             elif line.startswith('!'):\\n                 # Comment line\\n                 continue\\n             else:\\n                 # Data line or unrecognized command\\n@@ -300,10 +304,10 @@ class QDP(basic.Basic):\\n         for line in lines:\\n             line = line.strip()\\n             if not line or line.startswith('!'):\\n                 continue\\n             \\n-            if line.startswith('READ'):\\n+            if self._is_command(line, 'READ'):\\n                 # Parse READ command\\n-                parts = line.split()\\n+                parts = line.upper().split()\\n                 if len(parts) >= 3 and parts[1] in ['SERR', 'TERR']:\\n                     # Handle error specifications\\n                     pass\\n\", \"diff\\n--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -140,6 +140,10 @@ def _line_type(line, delimiter=None):\\n     return _QDPLineType.DATA, None\\n \\n \\n+def _command_matches(line, command):\\n+    \\\"\\\"\\\"Check if line starts with command (case-insensitive).\\\"\\\"\\\"\\n+    return line.upper().startswith(command.upper())\\n+\\n def _get_type_from_list_of_lines(lines, delimiter=None):\\n     \\\"\\\"\\\"\\n     Scan the lines to determine table type.\\n@@ -179,17 +183,17 @@ def _get_type_from_list_of_lines(lines, delimiter=None):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ SERR\\\"):\\n+        if _command_matches(line, \\\"READ SERR\\\"):\\n             serr_line = line\\n-        elif line.startswith(\\\"READ TERR\\\"):\\n+        elif _command_matches(line, \\\"READ TERR\\\"):\\n             terr_line = line\\n-        elif line.startswith(\\\"READ\\\"):\\n+        elif _command_matches(line, \\\"READ\\\"):\\n             # This is a read command but not SERR or TERR\\n             pass\\n         elif line.startswith(\\\"!\\\"):\\n             # Comment line\\n             continue\\n-        elif line.startswith((\\\"SKIP\\\", \\\"NO\\\", \\\"PLOT\\\", \\\"TIME\\\", \\\"LST\\\", \\\"LAB\\\")):\\n+        elif _command_matches(line, \\\"SKIP\\\") or _command_matches(line, \\\"NO\\\") or _command_matches(line, \\\"PLOT\\\") or _command_matches(line, \\\"TIME\\\") or _command_matches(line, \\\"LST\\\") or _command_matches(line, \\\"LAB\\\"):\\n             # Other QDP commands\\n             continue\\n         else:\\n@@ -248,7 +252,7 @@ def _get_lines_from_file(qdp_file):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ\\\"):\\n+        if _command_matches(line, \\\"READ\\\"):\\n             command_lines.append(line)\\n         elif line.startswith(\\\"!\\\"):\\n             # Comment line\\n@@ -318,7 +322,7 @@ def _get_tables_from_qdp_file(qdp_file, input_colnames=None, delimiter=None):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ\\\"):\\n+        if _command_matches(line, \\\"READ\\\"):\\n             # Process READ command\\n             continue\\n         elif line.startswith(\\\"!\\\"):\\n\", \"diff\\n--- a/astropy/io/ascii/qdp.py\\n+++ b/astropy/io/ascii/qdp.py\\n@@ -140,6 +140,11 @@ def _line_type(line, delimiter=None):\\n     return _QDPLineType.DATA, None\\n \\n \\n+def _command_matches(line, command):\\n+    \\\"\\\"\\\"Check if line starts with command (case-insensitive).\\\"\\\"\\\"\\n+    return line.upper().startswith(command.upper())\\n+\\n+\\n def _get_type_from_list_of_lines(lines, delimiter=None):\\n     \\\"\\\"\\\"\\n     Scan the lines to determine table type.\\n@@ -179,17 +184,17 @@ def _get_type_from_list_of_lines(lines, delimiter=None):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ SERR\\\"):\\n+        if _command_matches(line, \\\"READ SERR\\\"):\\n             serr_line = line\\n-        elif line.startswith(\\\"READ TERR\\\"):\\n+        elif _command_matches(line, \\\"READ TERR\\\"):\\n             terr_line = line\\n-        elif line.startswith(\\\"READ\\\"):\\n+        elif _command_matches(line, \\\"READ\\\"):\\n             # This is a read command but not SERR or TERR\\n             pass\\n         elif line.startswith(\\\"!\\\"):\\n             # Comment line\\n             continue\\n-        elif line.startswith((\\\"SKIP\\\", \\\"NO\\\", \\\"PLOT\\\", \\\"TIME\\\", \\\"LST\\\", \\\"LAB\\\")):\\n+        elif any(_command_matches(line, cmd) for cmd in [\\\"SKIP\\\", \\\"NO\\\", \\\"PLOT\\\", \\\"TIME\\\", \\\"LST\\\", \\\"LAB\\\"]):\\n             # Other QDP commands\\n             continue\\n         else:\\n@@ -248,7 +253,7 @@ def _get_lines_from_file(qdp_file):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ\\\"):\\n+        if _command_matches(line, \\\"READ\\\"):\\n             command_lines.append(line)\\n         elif line.startswith(\\\"!\\\"):\\n             # Comment line\\n@@ -318,7 +323,7 @@ def _get_tables_from_qdp_file(qdp_file, input_colnames=None, delimiter=None):\\n         if not line:\\n             continue\\n \\n-        if line.startswith(\\\"READ\\\"):\\n+        if _command_matches(line, \\\"READ\\\"):\\n             # Process READ command\\n             continue\\n         elif line.startswith(\\\"!\\\"):\\n\"]}, \"error\": \"\", \"elapsed_time\": 136.248821, \"total_tokens\": 45732, \"total_steps\": 30, \"created_at\": 1754645069, \"finished_at\": 1754645205}}"
}