{
  "instance_id": "astropy__astropy-14182",
  "repo": "astropy/astropy",
  "created_at": "2022-12-16T11:13:37Z",
  "problem_statement": "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",
  "patch": "diff --git a/astropy/io/ascii/rst.py b/astropy/io/ascii/rst.py\n--- a/astropy/io/ascii/rst.py\n+++ b/astropy/io/ascii/rst.py\n@@ -27,7 +27,6 @@ def get_fixedwidth_params(self, line):\n \n \n class SimpleRSTData(FixedWidthData):\n-    start_line = 3\n     end_line = -1\n     splitter_class = FixedWidthTwoLineDataSplitter\n \n@@ -39,12 +38,29 @@ class RST(FixedWidth):\n \n     Example::\n \n-        ==== ===== ======\n-        Col1  Col2  Col3\n-        ==== ===== ======\n-          1    2.3  Hello\n-          2    4.5  Worlds\n-        ==== ===== ======\n+      >>> from astropy.table import QTable\n+      >>> import astropy.units as u\n+      >>> import sys\n+      >>> tbl = QTable({\"wave\": [350, 950] * u.nm, \"response\": [0.7, 1.2] * u.count})\n+      >>> tbl.write(sys.stdout,  format=\"ascii.rst\")\n+      ===== ========\n+       wave response\n+      ===== ========\n+      350.0      0.7\n+      950.0      1.2\n+      ===== ========\n+\n+    Like other fixed-width formats, when writing a table you can provide ``header_rows``\n+    to specify a list of table rows to output as the header.  For example::\n+\n+      >>> tbl.write(sys.stdout,  format=\"ascii.rst\", header_rows=['name', 'unit'])\n+      ===== ========\n+       wave response\n+         nm       ct\n+      ===== ========\n+      350.0      0.7\n+      950.0      1.2\n+      ===== ========\n \n     Currently there is no support for reading tables which utilize continuation lines,\n     or for ones which define column spans through the use of an additional\n@@ -57,10 +73,15 @@ class RST(FixedWidth):\n     data_class = SimpleRSTData\n     header_class = SimpleRSTHeader\n \n-    def __init__(self):\n-        super().__init__(delimiter_pad=None, bookend=False)\n+    def __init__(self, header_rows=None):\n+        super().__init__(delimiter_pad=None, bookend=False, header_rows=header_rows)\n \n     def write(self, lines):\n         lines = super().write(lines)\n-        lines = [lines[1]] + lines + [lines[1]]\n+        idx = len(self.header.header_rows)\n+        lines = [lines[idx]] + lines + [lines[idx]]\n         return lines\n+\n+    def read(self, table):\n+        self.data.start_line = 2 + len(self.header.header_rows)\n+        return super().read(table)\n",
  "similar_bug_items": [
    {
      "pr_number": 2750,
      "pr_title": ".fz fits files KeyError: \"Keyword 'ZSIMPLE' not found in 0.3.2 versus 0.3",
      "pr_body": "I have some .fz fits files that are giving the error:\n\nKeyError: \"Keyword 'ZSIMPLE' not found.\"\n\nwhen I use astropy 0.3.2 and also 0.4rc3.dev9443 but 0.3 reads the header OK\n\nThe header looks like\n.\n.\n.\nZVAL2   =                    4 / bytes per pixel (1, 2, 4, or 8)  \nEXTNAME = 'COMPRESSED_IMAGE'  \nZSIMPLE =                    T / file does conform to FITS standard  \nZBITPIX =                  -32 / number of bits per data pixel  \n.\n.\n.\n",
      "issue_id": 2750,
      "issue_title": ".fz fits files KeyError: \"Keyword 'ZSIMPLE' not found in 0.3.2 versus 0.3",
      "issue_body": "I have some .fz fits files that are giving the error:\n\nKeyError: \"Keyword 'ZSIMPLE' not found.\"\n\nwhen I use astropy 0.3.2 and also 0.4rc3.dev9443 but 0.3 reads the header OK\n\nThe header looks like\n.\n.\n.\nZVAL2   =                    4 / bytes per pixel (1, 2, 4, or 8)  \nEXTNAME = 'COMPRESSED_IMAGE'  \nZSIMPLE =                    T / file does conform to FITS standard  \nZBITPIX =                  -32 / number of bits per data pixel  \n.\n.\n.\n",
      "issue_closed_at": "2014-09-16T14:07:45Z",
      "base_commit": "c027ce8414ab4eef47b2f5821960dc75b331d611",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/compressed.py",
          "type": "function",
          "name": "header",
          "class_name": "CompImageHDU",
          "code": "def header(self):\n        # The header attribute is the header for the image data.  It\n        # is not actually stored in the object dictionary.  Instead,\n        # the _image_header is stored.  If the _image_header attribute\n        # has already been defined we just return it.  If not, we nust\n        # create it from the table header (the _header attribute).\n        if hasattr(self, '_image_header'):\n            return self._image_header\n\n        # Start with a copy of the table header.\n        image_header = self._header.copy()\n\n        # Delete cards that are related to the table.  And move\n        # the values of those cards that relate to the image from\n        # their corresponding table cards.  These include\n        # ZBITPIX -> BITPIX, ZNAXIS -> NAXIS, and ZNAXISn -> NAXISn.\n        for keyword in list(image_header):\n            if CompImageHeader._is_reserved_keyword(keyword, warn=False):\n                del image_header[keyword]\n\n        if 'ZSIMPLE' in self._header:\n            image_header.set('SIMPLE', self._header['ZSIMPLE'],\n                             self._header.comments['ZSIMPLE'], before=0)\n        elif 'ZTENSION' in self._header:\n            if self._header['ZTENSION'] != 'IMAGE':\n                warnings.warn(\"ZTENSION keyword in compressed \"\n                              \"extension != 'IMAGE'\", AstropyUserWarning)\n            image_header.set('XTENSION', 'IMAGE',\n                             self._header.comments['ZTENSION'], before=0)\n        else:\n            image_header.set('XTENSION', 'IMAGE', before=0)\n\n\n        image_header.set('BITPIX', self._header['ZBITPIX'],\n                         self._header.comments['ZBITPIX'], before=1)\n\n        image_header.set('NAXIS', self._header['ZNAXIS'],\n                         self._header.comments['ZNAXIS'], before=2)\n\n        last_naxis = 'NAXIS'\n        for idx in range(image_header['NAXIS']):\n            znaxis = 'ZNAXIS' + str(idx + 1)\n            naxis = znaxis[1:]\n            image_header.set(naxis, self._header[znaxis],\n                             self._header.comments[znaxis],\n                             after=last_naxis)\n            last_naxis = naxis\n\n        # Delete any other spurious NAXISn keywords:\n        naxis = image_header['NAXIS']\n        for keyword in list(image_header['NAXIS?*']):\n            try:\n                n = int(keyword[5:])\n            except:\n                continue\n\n            if n > naxis:\n                del image_header[keyword]\n\n        # Although PCOUNT and GCOUNT are considered mandatory for IMAGE HDUs,\n        # ZPCOUNT and ZGCOUNT are optional, probably because for IMAGE HDUs\n        # their values are always 0 and 1 respectively\n        if 'ZPCOUNT' in self._header:\n            image_header.set('PCOUNT', self._header['ZPCOUNT'],\n                             self._header.comments['ZPCOUNT'],\n                             after=last_naxis)\n        else:\n            image_header.set('PCOUNT', 0, after=last_naxis)\n\n        if 'ZGCOUNT' in self._header:\n            image_header.set('GCOUNT', self._header['ZGCOUNT'],\n                             self._header.comments['ZGCOUNT'],\n                             after='PCOUNT')\n        else:\n            image_header.set('GCOUNT', 1, after='PCOUNT')\n\n        if 'ZEXTEND' in self._header:\n            image_header.set('EXTEND', self._header['ZEXTEND'],\n                             self._header.comments['ZEXTEND'])\n\n        if 'ZBLOCKED' in self._header:\n            image_header.set('BLOCKED', self._header['ZBLOCKED'],\n                             self._header.comments['ZBLOCKED'])\n\n        # Move the ZHECKSUM and ZDATASUM cards to the image header\n        # as CHECKSUM and DATASUM\n        if 'ZHECKSUM' in self._header:\n            image_header.set('CHECKSUM', self._header['ZHECKSUM'],\n                             self._header.comments['ZHECKSUM'])\n\n        if 'ZDATASUM' in self._header:\n            image_header.set('DATASUM', self._header['ZDATASUM'],\n                             self._header.comments['ZDATASUM'])\n\n        # Remove the EXTNAME card if the value in the table header\n        # is the default value of COMPRESSED_IMAGE.\n        if ('EXTNAME' in self._header and\n                self._header['EXTNAME'] == 'COMPRESSED_IMAGE'):\n            del image_header['EXTNAME']\n\n        # Look to see if there are any blank cards in the table\n        # header.  If there are, there should be the same number\n        # of blank cards in the image header.  Add blank cards to\n        # the image header to make it so.\n        table_blanks = self._header._countblanks()\n        image_blanks = image_header._countblanks()\n\n        for _ in range(table_blanks - image_blanks):\n            image_header.append()\n\n        # Create the CompImageHeader that syncs with the table header, and save\n        # it off to self._image_header so it can be referenced later\n        # unambiguously\n        self._image_header = CompImageHeader(self._header, image_header)\n\n        return self._image_header"
        }
      ]
    },
    {
      "pr_number": 9489,
      "pr_title": "Add hint for filtering multidim cols in to_pandas and io.ascii ECSV",
      "pr_body": "### Description\r\n\r\nPR #9423 was not accepted by all as a way to deal with handling multidimensional columns in `to_pandas()` and `ascii.write`.  This takes a different approach and provides the 2-liner for users in this case, instead of embedding it as a new Table method.\r\n\r\nCloses #9423",
      "issue_id": 9423,
      "issue_title": "Add Table.drop_array_columns and improve Table.to_pandas",
      "issue_body": "This PR adds a new method `Table.drop_array_colums`. \r\n\r\nThe motivation is to make it easy to convert a table to pandas or write it to an ASCII file, i.e. go to formats that don't support array columns.\r\n\r\nAt first I thought we'd add an option `drop_array_columns=True` with default to do this to `table.to_pandas` (see  https://groups.google.com/d/msg/astropy-dev/iOha8R59PX8/KM1dHbytAQAJ), but probably it's better to have the default be to raise an error and not just silently drop columns, and if so, then the extra method is probably the better API.\r\n\r\nI also changed `Table.to_pandas` slightly, fixing the docstring, and checking and raising an error for array columns earlier.\r\n\r\nAt first I added an `Colum.is_scalar` attribute and wanted to use that, but then there were failing tests because sometimes `table[\"time_col\"]` was a `Time` and not a `Column` object. Not sure if there is a better way to implement this functionality, or if the current implementation is fine. It's from the suggestion by @taldcroft in https://github.com/astropy/astropy/issues/4604#issuecomment-184425733\r\n\r\n@taldcroft or @mhvk - Please review.",
      "issue_closed_at": "2019-10-27T18:47:23Z",
      "base_commit": "297589cd9b3f087795803cd698d8410c46cac0eb",
      "changes": [
        {
          "file": "astropy/io/ascii/ecsv.py",
          "type": "function",
          "name": "write",
          "class_name": "EcsvHeader",
          "code": "def write(self, lines):\n        \"\"\"\n        Write header information in the ECSV ASCII format.  This format\n        starts with a delimiter separated list of the column names in order\n        to make this format readable by humans and simple csv-type readers.\n        It then encodes the full table meta and column attributes and meta\n        as YAML and pretty-prints this in the header.  Finally the delimited\n        column names are repeated again, for humans and readers that look\n        for the *last* comment line as defining the column names.\n        \"\"\"\n        if self.splitter.delimiter not in DELIMITERS:\n            raise ValueError('only space and comma are allowed for delimiter in ECSV format')\n\n        for col in self.cols:\n            if len(getattr(col, 'shape', ())) > 1:\n                raise ValueError(\"ECSV format does not support multidimensional column '{}'\"\n                                 .format(col.info.name))\n\n        # Now assemble the header dict that will be serialized by the YAML dumper\n        header = {'cols': self.cols, 'schema': 'astropy-2.0'}\n\n        if self.table_meta:\n            header['meta'] = self.table_meta\n\n        # Set the delimiter only for the non-default option(s)\n        if self.splitter.delimiter != ' ':\n            header['delimiter'] = self.splitter.delimiter\n\n        header_yaml_lines = ([f'%ECSV {ECSV_VERSION}',\n                              '---']\n                             + meta.get_yaml_from_header(header))\n\n        lines.extend([self.write_comment + line for line in header_yaml_lines])\n        lines.append(self.splitter.join([x.info.name for x in self.cols]))"
        },
        {
          "file": "astropy/table/table.py",
          "type": "function",
          "name": "_encode_mixins",
          "class_name": "Table",
          "code": "def _encode_mixins(tbl):\n            \"\"\"Encode a Table ``tbl`` that may have mixin columns to a Table with only\n            astropy Columns + appropriate meta-data to allow subsequent decoding.\n            \"\"\"\n            from . import serialize\n            from astropy.utils.data_info import MixinInfo, serialize_context_as\n            from astropy.time import Time, TimeDelta\n\n            # Convert any Time or TimeDelta columns and pay attention to masking\n            time_cols = [col for col in tbl.itercols() if isinstance(col, Time)]\n            if time_cols:\n\n                # Make a light copy of table and clear any indices\n                new_cols = []\n                for col in tbl.itercols():\n                    new_col = col_copy(col, copy_indices=False) if col.info.indices else col\n                    new_cols.append(new_col)\n                tbl = tbl.__class__(new_cols, copy=False)\n\n                for col in time_cols:\n                    if isinstance(col, TimeDelta):\n                        # Convert to nanoseconds (matches astropy datetime64 support)\n                        new_col = (col.sec * 1e9).astype('timedelta64[ns]')\n                        nat = np.timedelta64('NaT')\n                    else:\n                        new_col = col.datetime64.copy()\n                        nat = np.datetime64('NaT')\n                    if col.masked:\n                        new_col[col.mask] = nat\n                    tbl[col.info.name] = new_col\n\n            # Convert the table to one with no mixins, only Column objects.\n            encode_tbl = serialize.represent_mixins_as_columns(tbl)\n            return encode_tbl"
        }
      ]
    },
    {
      "pr_number": 4561,
      "pr_title": "Remove new line characters after last row of data in ascii.latex.AASTex",
      "pr_body": "For #3888. Also fixed `AASTex` tests in `test_write.py` to incorporate these changes.\n",
      "issue_id": 3888,
      "issue_title": "ascii.latex.AASTex: don't include new line characters after last row of data",
      "issue_body": "Very minor issue: writing a table in AASTex (deluxetable) format includes new line characters at the end of each line, including the last one. This results in a blank line at the end of each table. The last row of a table before \"\\enddata\" should not include the \"\\\\\" characters.\n\nExample:\n\n```\nfrom astropy.io import ascii\nfrom astropy.table import Table\nimport sys    \nt = Table([[1,2],[1.234e9,2.34e-12]], names = ('a','b'))\nascii.write(t,sys.stdout,Writer=ascii.latex.AASTex)\n```\n\nResult: \n\n```\n\\begin{deluxetable}{cc}\n \\tablehead{\\colhead{a} & \\colhead{b}}\n\\startdata\n1 & 1234000000.0 \\\\\n2 & 2.34e-12 \\\\\n\\enddata\n\\end{deluxetable}\n```\n\nWhich looks like:\n![image](https://cloud.githubusercontent.com/assets/4870555/8362092/c5c2f72a-1b2b-11e5-98e5-d53a2110d9a0.png)\n\nBut if you got rid of the \"\\\\\" after 2.34e-12, you'd get:\n![image](https://cloud.githubusercontent.com/assets/4870555/8362105/e005836e-1b2b-11e5-9d0d-997302f1d083.png)\n",
      "issue_closed_at": "2016-02-08T12:03:53Z",
      "base_commit": "12e9367e6e9aafece435469e06f0048d94519aa6",
      "changes": [
        {
          "file": "astropy/io/ascii/latex.py",
          "type": "function",
          "name": "start_line",
          "class_name": "AASTexData",
          "code": "def start_line(self, lines):\n        return find_latex_line(lines, self.data_start) + 1"
        }
      ]
    },
    {
      "pr_number": 3657,
      "pr_title": "#3564 assume sensible default for col_ends or col_starts if omitted for fixed-width table",
      "pr_body": "This is to resolve #3564. I've changed the structure of an if-statement in io.ascii.fixedwidth.py inside the FixedWidthHeader.get_fixedwidth_params method to allow me to handle the case where only one of col_starts or col_ends is specified. The behavious for other cases is not changed.\n\nI've also added tests, and two examples in the documentation.\n\nI have run\n\n```\npython setup.py test --package=io.ascii\n```\n\nusing Python 2.7.6 and 3.4.0 for this change and received no failures.\n",
      "issue_id": 3564,
      "issue_title": "For fixed with reader, make ``col_ends`` default sensible",
      "issue_body": "For the fixed width reader in io.ascii it would be nice to avoid having to do things like\n\n```\ncol_starts=[0,5,20,32,42,55], col_ends=[5,20,32,42,55,100]\n```\n\nBy default, one could assume that columns end where the next start, and have the last item default to infinity (or whatever means to the end of the line).\n",
      "issue_closed_at": "2015-04-03T15:38:08Z",
      "base_commit": "66d3a10569adb5de046a0ff458038293748228f7",
      "changes": [
        {
          "file": "astropy/io/ascii/fixedwidth.py",
          "type": "function",
          "name": "get_fixedwidth_params",
          "class_name": "FixedWidthHeader",
          "code": "def get_fixedwidth_params(self, line):\n        \"\"\"\n        Split ``line`` on the delimiter and determine column values and\n        column start and end positions.  This might include null columns with\n        zero length (e.g. for ``header row = \"| col1 || col2 | col3 |\"`` or\n        ``header2_row = \"----- ------- -----\"``).  The null columns are\n        stripped out.  Returns the values between delimiters and the\n        corresponding start and end positions.\n\n        Parameters\n        ----------\n        line : str\n            Input line\n\n        Returns\n        -------\n        vals : list\n            List of values.\n        starts : list\n            List of starting indices.\n        ends : list\n            List of ending indices.\n\n        \"\"\"\n\n        # If column positions are already specified then just use those, otherwise\n        # figure out positions between delimiters.\n        if self.col_starts is not None and self.col_ends is not None:\n            starts = list(self.col_starts)  # could be any iterable, e.g. np.array\n            ends = [x + 1 for x in self.col_ends]  # user supplies inclusive endpoint\n            if len(starts) != len(ends):\n                raise ValueError('Fixed width col_starts and col_ends must have the same length')\n            vals = [line[start:end].strip() for start, end in zip(starts, ends)]\n        else:\n            # There might be a cleaner way to do this but it works...\n            vals = line.split(self.splitter.delimiter)\n            starts = [0]\n            ends = []\n            for val in vals:\n                if val:\n                    ends.append(starts[-1] + len(val))\n                    starts.append(ends[-1] + 1)\n                else:\n                    starts[-1] += 1\n            starts = starts[:-1]\n            vals = [x.strip() for x in vals if x]\n            if len(vals) != len(starts) or len(vals) != len(ends):\n                raise InconsistentTableError('Error parsing fixed width header')\n\n        return vals, starts, ends"
        },
        {
          "file": "astropy/io/ascii/fixedwidth.py",
          "type": "function",
          "name": "get_fixedwidth_params",
          "class_name": "FixedWidthHeader",
          "code": "def get_fixedwidth_params(self, line):\n        \"\"\"\n        Split ``line`` on the delimiter and determine column values and\n        column start and end positions.  This might include null columns with\n        zero length (e.g. for ``header row = \"| col1 || col2 | col3 |\"`` or\n        ``header2_row = \"----- ------- -----\"``).  The null columns are\n        stripped out.  Returns the values between delimiters and the\n        corresponding start and end positions.\n\n        Parameters\n        ----------\n        line : str\n            Input line\n\n        Returns\n        -------\n        vals : list\n            List of values.\n        starts : list\n            List of starting indices.\n        ends : list\n            List of ending indices.\n\n        \"\"\"\n\n        # If column positions are already specified then just use those, otherwise\n        # figure out positions between delimiters.\n        if self.col_starts is not None and self.col_ends is not None:\n            starts = list(self.col_starts)  # could be any iterable, e.g. np.array\n            ends = [x + 1 for x in self.col_ends]  # user supplies inclusive endpoint\n            if len(starts) != len(ends):\n                raise ValueError('Fixed width col_starts and col_ends must have the same length')\n            vals = [line[start:end].strip() for start, end in zip(starts, ends)]\n        else:\n            # There might be a cleaner way to do this but it works...\n            vals = line.split(self.splitter.delimiter)\n            starts = [0]\n            ends = []\n            for val in vals:\n                if val:\n                    ends.append(starts[-1] + len(val))\n                    starts.append(ends[-1] + 1)\n                else:\n                    starts[-1] += 1\n            starts = starts[:-1]\n            vals = [x.strip() for x in vals if x]\n            if len(vals) != len(starts) or len(vals) != len(ends):\n                raise InconsistentTableError('Error parsing fixed width header')\n\n        return vals, starts, ends"
        }
      ]
    },
    {
      "pr_number": 11982,
      "pr_title": "Accept UCD C atoms when parsing UCDs",
      "pr_body": "### Description\r\n\r\nThis pull request is to address the rejection of phot.color (and\r\npossibly other C atoms, if these ever came to exist) UCDs by astropy.\r\n\r\nFixes #11981\r\n\r\n### Checklist for package maintainer(s)\r\n<!-- This section is to be filled by package maintainer(s) who will\r\nreview this pull request. -->\r\n\r\nThis checklist is meant to remind the package maintainer(s) who will review this pull request of some common things to look for. This list is not exhaustive.\r\n\r\n- [x] Do the proposed changes actually accomplish desired goals?\r\n- [x] Do the proposed changes follow the [Astropy coding guidelines](https://docs.astropy.org/en/latest/development/codeguide.html)?\r\n- [x] Are tests added/updated as required? If so, do they follow the [Astropy testing guidelines](https://docs.astropy.org/en/latest/development/testguide.html)?\r\n- [x] Are docs added/updated as required? If so, do they follow the [Astropy documentation guidelines](https://docs.astropy.org/en/latest/development/docguide.html#astropy-documentation-rules-and-guidelines)?\r\n- [x] Is rebase and/or squash necessary? If so, please provide the author with appropriate instructions. Also see [\"When to rebase and squash commits\"](https://docs.astropy.org/en/latest/development/when_to_rebase.html).\r\n- [x] Did the CI pass? If no, are the failures related? If you need to run daily and weekly cron jobs as part of the PR, please apply the `Extra CI` label.\r\n- [x] Is a change log needed? If yes, did the change log check pass? If no, add the `no-changelog-entry-needed` label.\r\n- [x] Is a milestone set? Milestone must be set but `astropy-bot` check might be missing; do not let the green checkmark fool you.\r\n- [x] At the time of adding the milestone, if the milestone set requires a backport to release branch(es), apply the appropriate `backport-X.Y.x` label(s) *before* merge.",
      "issue_id": 11981,
      "issue_title": "phot.color UCDs not accepted",
      "issue_body": "<!-- This comments are hidden when you submit the issue,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/main/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/main/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- Please have a search on our GitHub repository to see if a similar\r\nissue has already been posted.\r\nIf a similar issue is closed, have a quick look to see if you are satisfied\r\nby the resolution.\r\nIf not please go ahead and open an issue! -->\r\n\r\n<!-- Please check that the development version still produces the same bug.\r\nYou can install development version with\r\npip install git+https://github.com/astropy/astropy\r\ncommand. -->\r\n\r\n### Description\r\n<!-- Provide a general description of the bug. -->\r\n\r\nUCDs containing the phot.color atom are not accepted.\r\n\r\n### Expected behavior\r\n<!-- What did you expect to happen. -->\r\n\r\n```\r\nfrom astropy.io.votable.ucd import check_ucd\r\ncheck_ucd(\"phot.color;em.opt.B;em.opt.V\", True)\r\n```\r\n\r\nshould return True, as should\r\n\r\n```\r\ncheck_ucd(\"stat.error;phot.color;em.opt.B;em.opt.V\", True)\r\n```\r\n\r\n### Actual behavior\r\n<!-- What actually happened. -->\r\n<!-- Was the output confusing or poorly described? -->\r\n\r\nFalse is returned in both cases.\r\n\r\n\r\n### System Details\r\n\r\nThis is astropy 4.2 on Debian, but the problem persists in git HEAD.\r\n\r\nI'll prepare a PR to fix this.",
      "issue_closed_at": "2021-08-02T14:39:00Z",
      "base_commit": "2216de2614825335cd0e403a04ab718fcfca45cd",
      "changes": [
        {
          "file": "astropy/io/votable/ucd.py",
          "type": "function",
          "name": "__init__",
          "class_name": "UCDWords",
          "code": "def __init__(self):\n        self._primary = set()\n        self._secondary = set()\n        self._descriptions = {}\n        self._capitalization = {}\n\n        with data.get_pkg_data_fileobj(\n                \"data/ucd1p-words.txt\", encoding='ascii') as fd:\n            for line in fd.readlines():\n                type, name, descr = [\n                    x.strip() for x in line.split('|')]\n                name_lower = name.lower()\n                if type in 'QPEV':\n                    self._primary.add(name_lower)\n                if type in 'QSEV':\n                    self._secondary.add(name_lower)\n                self._descriptions[name_lower] = descr\n                self._capitalization[name_lower] = name"
        }
      ]
    }
  ]
}