{
  "id": "astropy__astropy-14182",
  "question": "Please support header rows in RestructuredText output\n### Description\r\n\r\nIt would be great if the following would work:\r\n\r\n```Python\r\n>>> from astropy.table import QTable\r\n>>> import astropy.units as u\r\n>>> import sys\r\n>>> tbl = QTable({'wave': [350,950]*u.nm, 'response': [0.7, 1.2]*u.count})\r\n>>> tbl.write(sys.stdout,  format=\"ascii.rst\")\r\n===== ========\r\n wave response\r\n===== ========\r\n350.0      0.7\r\n950.0      1.2\r\n===== ========\r\n>>> tbl.write(sys.stdout,  format=\"ascii.fixed_width\", header_rows=[\"name\", \"unit\"])\r\n|  wave | response |\r\n|    nm |       ct |\r\n| 350.0 |      0.7 |\r\n| 950.0 |      1.2 |\r\n>>> tbl.write(sys.stdout,  format=\"ascii.rst\", header_rows=[\"name\", \"unit\"])\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/usr/lib/python3/dist-packages/astropy/table/connect.py\", line 129, in __call__\r\n    self.registry.write(instance, *args, **kwargs)\r\n  File \"/usr/lib/python3/dist-packages/astropy/io/registry/core.py\", line 369, in write\r\n    return writer(data, *args, **kwargs)\r\n  File \"/usr/lib/python3/dist-packages/astropy/io/ascii/connect.py\", line 26, in io_write\r\n    return write(table, filename, **kwargs)\r\n  File \"/usr/lib/python3/dist-packages/astropy/io/ascii/ui.py\", line 856, in write\r\n    writer = get_writer(Writer=Writer, fast_writer=fast_writer, **kwargs)\r\n  File \"/usr/lib/python3/dist-packages/astropy/io/ascii/ui.py\", line 800, in get_writer\r\n    writer = core._get_writer(Writer, fast_writer, **kwargs)\r\n  File \"/usr/lib/python3/dist-packages/astropy/io/ascii/core.py\", line 1719, in _get_writer\r\n    writer = Writer(**writer_kwargs)\r\nTypeError: RST.__init__() got an unexpected keyword argument 'header_rows'\r\n```\r\n\r\n\r\n### Additional context\r\n\r\nRestructuredText output is a great way to fill autogenerated documentation with content, so having this flexible makes the life easier `:-)`\r\n\r\n\n",
  "answer": "{\"task_id\": \"39cba3d9-c3b5-4a82-9753-95a154387703\", \"workflow_run_id\": \"b2432213-ceca-4829-9545-6b0f620907f0\", \"data\": {\"id\": \"b2432213-ceca-4829-9545-6b0f620907f0\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,25 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n-        else:\\n-            # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n-            for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n-\\n-        return lines\\n+        header_lines = []\\n+        \\n+        # Generate lines for each header row type\\n+        for header_row in self.parent_table.header_rows:\\n+            line = []\\n+            for col in self.cols:\\n+                if header_row == 'name':\\n+                    line.append(col.info.name)\\n+                elif header_row == 'unit':\\n+                    unit = getattr(col.info, 'unit', None)\\n+                    if unit is not None:\\n+                        line.append(str(unit))\\n+                    else:\\n+                        line.append('')\\n+                else:\\n+                    line.append('')\\n+            header_lines.append(line)\\n+        \\n+        return header_lines\\n \\n \\n class RSTData(FixedWidthData):\", \"--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,11 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        # Pass header_rows to parent class if provided\\n+        if header_rows is not None:\\n+            kwargs['header_rows'] = header_rows\\n+        super().__init__(**kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n-        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +87,20 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Check if header_rows is specified and use parent class logic\\n+        if hasattr(self.parent_table, 'header_rows') and self.parent_table.header_rows:\\n+            # Use parent class implementation which handles header_rows\\n+            return super()._header_lines(lines)\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n-        return lines\\n+        return lines\\n \\n \\n class RSTData(FixedWidthData):\", \"--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self._header_rows = header_rows\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,32 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Handle multiple header rows if specified\\n+        if hasattr(self.parent_table, '_header_rows') and self.parent_table._header_rows:\\n+            header_lines = []\\n+            for header_row in self.parent_table._header_rows:\\n+                line = []\\n+                for col in self.cols:\\n+                    if header_row == 'name':\\n+                        line.append(col.info.name)\\n+                    elif header_row == 'unit':\\n+                        unit = getattr(col.info, 'unit', None)\\n+                        line.append(str(unit) if unit is not None else '')\\n+                    else:\\n+                        line.append('')\\n+                header_lines.append(line)\\n+            return header_lines\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n-        return lines\\n+        return lines\\n \\n \\n class RSTData(FixedWidthData):\", \"--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,7 +30,7 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\", \"--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,29 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Handle header_rows parameter\\n+        if hasattr(self.parent_table, 'header_rows') and len(self.parent_table.header_rows) > 1:\\n+            # Multiple header rows requested\\n+            header_lines = []\\n+            for header_row in self.parent_table.header_rows:\\n+                line = []\\n+                for col in self.cols:\\n+                    if header_row == 'name':\\n+                        line.append(col.info.name)\\n+                    elif header_row == 'unit':\\n+                        unit = getattr(col.info, 'unit', None)\\n+                        line.append(str(unit) if unit is not None else '')\\n+                    else:\\n+                        line.append('')\\n+                header_lines.append(line)\\n+            return header_lines\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n-\\n-        return lines\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n \\n class RSTData(FixedWidthData):\", \"diff\\n--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,29 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Handle header_rows parameter\\n+        if hasattr(self.parent_table, 'header_rows') and len(self.parent_table.header_rows) > 1:\\n+            # Multiple header rows requested\\n+            header_lines = []\\n+            for header_row in self.parent_table.header_rows:\\n+                line = []\\n+                for col in self.cols:\\n+                    if header_row == 'name':\\n+                        line.append(col.info.name)\\n+                    elif header_row == 'unit':\\n+                        unit = getattr(col.info, 'unit', None)\\n+                        line.append(str(unit) if unit is not None else '')\\n+                    else:\\n+                        line.append('')\\n+                header_lines.append(line)\\n+            return header_lines\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n-\\n-        return lines\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n \\n class RSTData(FixedWidthData):\\n\", \"diff\\n--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,29 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Handle header_rows parameter\\n+        if hasattr(self.parent_table, 'header_rows') and len(self.parent_table.header_rows) > 1:\\n+            # Multiple header rows requested\\n+            header_lines = []\\n+            for header_row in self.parent_table.header_rows:\\n+                line = []\\n+                for col in self.cols:\\n+                    if header_row == 'name':\\n+                        line.append(col.info.name)\\n+                    elif header_row == 'unit':\\n+                        unit = getattr(col.info, 'unit', None)\\n+                        line.append(str(unit) if unit is not None else '')\\n+                    else:\\n+                        line.append('')\\n+                header_lines.append(line)\\n+            return header_lines\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n-\\n-        return lines\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n \\n class RSTData(FixedWidthData):\\n\", \"diff\\n--- a/astropy/io/ascii/rst.py\\n+++ b/astropy/io/ascii/rst.py\\n@@ -30,9 +30,10 @@ class RST(FixedWidth):\\n     \\\"\\\"\\\"\\n \\n     def __init__(self, header_rows=None, **kwargs):\\n-        super().__init__(**kwargs)\\n+        super().__init__(header_rows=header_rows, **kwargs)\\n         self.header = RSTHeader()\\n         self.data = RSTData()\\n+        self.header_rows = header_rows or ['name']\\n \\n \\n class RSTHeader(FixedWidthHeader):\\n@@ -85,15 +86,29 @@ class RSTHeader(FixedWidthHeader):\\n         return lines\\n \\n     def _header_lines(self, lines):\\n-        if self.parent_table.pformat_all:\\n-            lines = super()._header_lines(lines)\\n+        # Handle header_rows parameter\\n+        if hasattr(self.parent_table, 'header_rows') and len(self.parent_table.header_rows) > 1:\\n+            # Multiple header rows requested\\n+            header_lines = []\\n+            for header_row in self.parent_table.header_rows:\\n+                line = []\\n+                for col in self.cols:\\n+                    if header_row == 'name':\\n+                        line.append(col.info.name)\\n+                    elif header_row == 'unit':\\n+                        unit = getattr(col.info, 'unit', None)\\n+                        line.append(str(unit) if unit is not None else '')\\n+                    else:\\n+                        line.append('')\\n+                header_lines.append(line)\\n+            return header_lines\\n+        elif self.parent_table.pformat_all:\\n+            return super()._header_lines(lines)\\n         else:\\n             # Use only col names, not the col<name> format that parent class uses\\n-            lines = []\\n+            header_lines = []\\n             for i, col in enumerate(self.cols):\\n-                lines.append(col.info.name)\\n-\\n-        return lines\\n+                header_lines.append(col.info.name)\\n+            return [header_lines]\\n \\n \\n class RSTData(FixedWidthData):\\n\"]}, \"error\": \"\", \"elapsed_time\": 146.004817, \"total_tokens\": 51479, \"total_steps\": 30, \"created_at\": 1754644829, \"finished_at\": 1754644975}}"
}