{
  "instance_id": "astropy__astropy-14365",
  "repo": "astropy/astropy",
  "created_at": "2023-02-06T19:20:34Z",
  "problem_statement": "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",
  "patch": "diff --git a/astropy/io/ascii/qdp.py b/astropy/io/ascii/qdp.py\n--- a/astropy/io/ascii/qdp.py\n+++ b/astropy/io/ascii/qdp.py\n@@ -68,7 +68,7 @@ def _line_type(line, delimiter=None):\n     _new_re = rf\"NO({sep}NO)+\"\n     _data_re = rf\"({_decimal_re}|NO|[-+]?nan)({sep}({_decimal_re}|NO|[-+]?nan))*)\"\n     _type_re = rf\"^\\s*((?P<command>{_command_re})|(?P<new>{_new_re})|(?P<data>{_data_re})?\\s*(\\!(?P<comment>.*))?\\s*$\"\n-    _line_type_re = re.compile(_type_re)\n+    _line_type_re = re.compile(_type_re, re.IGNORECASE)\n     line = line.strip()\n     if not line:\n         return \"comment\"\n@@ -306,7 +306,7 @@ def _get_tables_from_qdp_file(qdp_file, input_colnames=None, delimiter=None):\n \n             values = []\n             for v in line.split(delimiter):\n-                if v == \"NO\":\n+                if v.upper() == \"NO\":\n                     values.append(np.ma.masked)\n                 else:\n                     # Understand if number is int or float\n",
  "similar_bug_items": [
    {
      "pr_number": 5125,
      "pr_title": "Fix to CompImageHDU from 5118 with regression test",
      "pr_body": "I managed to reproduce the failure that #5118 fixes, so I'm starting by adding a regression test to make sure it fails, then I'll cherry-pick the fix from #5118. I think the data passed to `CompImageHDU` had to be integer so that the BSCALE and BZERO still applied.\n\ncc @parejkoj @eteq @embray\n\nEDIT: Closes #5118 \n",
      "issue_id": 5118,
      "issue_title": "Fix CompImageHDU's made from ImageHDUs with BSCALE/BZERO.",
      "issue_body": "If an existing ImageHDU containing BSCALE/BZERO is turned into a CompImageHDU,\nthe TFIELDS keyword don't get placed in the correct location (immediately\nfollowing GCOUNT), resulting in an invalid CompImageHDU. This fixes this behavior in the file I originally had trouble with, but I was not able to come up with a minimal example to turn into a proper test. It passes all the io.fits tests on my local copy, but it appears CompImageHDUs don't have good test coverage, so I can't promise this change is safe (though I believe it more closely matches the FITS standard, which is the important thing).\n\nSaid minimal example should look something like the below, but this doesn't actually produce the failure on the old code:\n\n``` python\nfrom astropy.io import fits\nimport numpy as np\n\nx = np.random.random((100, 100))*100\n\nx0 = fits.PrimaryHDU()\nx1 = fits.ImageHDU(x)\nx2 = fits.ImageHDU(np.array(x-50, dtype=int), uint=True)\nx2.header['BZERO'] = 32768\nx2.header['BSCALE'] = 1\nx3 = fits.ImageHDU(x*3)\nx4 = fits.BinTableHDU()\nhdus = fits.HDUList([x0, x1, x2, x3, x4])\nhdus.writeto('3hdus.fits')\n\n# fitsverify (based on cfitsio) should fail on this file, only seeing the first HDU.\ndata = fits.open('3hdus.fits')\nfor i in [1, 2, 3]:\n    data[i] = fits.CompImageHDU(data=data[i].data, header=data[i].header)\ndata.writeto('3hdus_comp.fits')\n```\n\nFor further details, see my recent emails to astropy and astropy-dev.\n",
      "issue_closed_at": "2016-06-22T14:05:25Z",
      "base_commit": "c2de3ec51a24a0f93e90c7e1c178c2627814c06b",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/base.py",
          "type": "function",
          "name": "_update_uint_scale_keywords",
          "class_name": "_BaseHDU",
          "code": "def _update_uint_scale_keywords(self):\n        \"\"\"\n        If the data is unsigned int 16, 32, or 64 add BSCALE/BZERO cards to\n        header.\n        \"\"\"\n\n        if (self._has_data and self._standard and\n                _is_pseudo_unsigned(self.data.dtype)):\n            if 'GCOUNT' in self._header:\n                self._header.set('BSCALE', 1, after='GCOUNT')\n            else:\n                self._header.set('BSCALE', 1)\n            self._header.set('BZERO', _unsigned_zero(self.data.dtype),\n                             after='BSCALE')"
        },
        {
          "file": "astropy/io/fits/hdu/compressed.py",
          "type": "function",
          "name": "_update_header_data",
          "class_name": "CompImageHDU",
          "code": "def _update_header_data(self, image_header,\n                            name=None,\n                            compression_type=None,\n                            tile_size=None,\n                            hcomp_scale=None,\n                            hcomp_smooth=None,\n                            quantize_level=None,\n                            quantize_method=None,\n                            dither_seed=None):\n        \"\"\"\n        Update the table header (`_header`) to the compressed\n        image format and to match the input data (if any).  Create\n        the image header (`_image_header`) from the input image\n        header (if any) and ensure it matches the input\n        data. Create the initially-empty table data array to hold\n        the compressed data.\n\n        This method is mainly called internally, but a user may wish to\n        call this method after assigning new data to the `CompImageHDU`\n        object that is of a different type.\n\n        Parameters\n        ----------\n        image_header : Header instance\n            header to be associated with the image\n\n        name : str, optional\n            the ``EXTNAME`` value; if this value is `None`, then the name from\n            the input image header will be used; if there is no name in the\n            input image header then the default name 'COMPRESSED_IMAGE' is used\n\n        compression_type : str, optional\n            compression algorithm 'RICE_1', 'PLIO_1', 'GZIP_1', 'HCOMPRESS_1';\n            if this value is `None`, use value already in the header; if no\n            value already in the header, use 'RICE_1'\n\n        tile_size : sequence of int, optional\n            compression tile sizes as a list; if this value is `None`, use\n            value already in the header; if no value already in the header,\n            treat each row of image as a tile\n\n        hcomp_scale : float, optional\n            HCOMPRESS scale parameter; if this value is `None`, use the value\n            already in the header; if no value already in the header, use 1\n\n        hcomp_smooth : float, optional\n            HCOMPRESS smooth parameter; if this value is `None`, use the value\n            already in the header; if no value already in the header, use 0\n\n        quantize_level : float, optional\n            floating point quantization level; if this value is `None`, use the\n            value already in the header; if no value already in header, use 16\n\n        quantize_method : int, optional\n            floating point quantization dithering method; can be either\n            NO_DITHER (-1), SUBTRACTIVE_DITHER_1 (1; default), or\n            SUBTRACTIVE_DITHER_2 (2)\n\n        dither_seed : int, optional\n            random seed to use for dithering; can be either an integer in the\n            range 1 to 1000 (inclusive), DITHER_SEED_CLOCK (0; default), or\n            DITHER_SEED_CHECKSUM (-1)\n        \"\"\"\n\n        image_hdu = ImageHDU(data=self.data, header=self._header)\n        self._image_header = CompImageHeader(self._header, image_hdu.header)\n        self._axes = image_hdu._axes\n        del image_hdu\n\n        # Determine based on the size of the input data whether to use the Q\n        # column format to store compressed data or the P format.\n        # The Q format is used only if the uncompressed data is larger than\n        # 4 GB.  This is not a perfect heuristic, as one can contrive an input\n        # array which, when compressed, the entire binary table representing\n        # the compressed data is larger than 4GB.  That said, this is the same\n        # heuristic used by CFITSIO, so this should give consistent results.\n        # And the cases where this heuristic is insufficient are extreme and\n        # almost entirely contrived corner cases, so it will do for now\n        if self._has_data:\n            huge_hdu = self.data.nbytes > 2 ** 32\n\n            if huge_hdu and not CFITSIO_SUPPORTS_Q_FORMAT:\n                raise IOError(\n                    \"Astropy cannot compress images greater than 4 GB in size \"\n                    \"(%s is %s bytes) without CFITSIO >= 3.35\" %\n                    ((self.name, self.ver), self.data.nbytes))\n        else:\n            huge_hdu = False\n\n        # Update the extension name in the table header\n        if not name and 'EXTNAME' not in self._header:\n            name = 'COMPRESSED_IMAGE'\n\n        if name:\n            self._header.set('EXTNAME', name,\n                             'name of this binary table extension',\n                             after='TFIELDS')\n            self.name = name\n        else:\n            self.name = self._header['EXTNAME']\n\n        # Set the compression type in the table header.\n        if compression_type:\n            if compression_type not in ['RICE_1', 'GZIP_1', 'PLIO_1',\n                                        'HCOMPRESS_1']:\n                warnings.warn('Unknown compression type provided.  Default '\n                              '(%s) compression used.' %\n                              DEFAULT_COMPRESSION_TYPE, AstropyUserWarning)\n                compression_type = DEFAULT_COMPRESSION_TYPE\n\n            self._header.set('ZCMPTYPE', compression_type,\n                             'compression algorithm', after='TFIELDS')\n        else:\n            compression_type = self._header.get('ZCMPTYPE',\n                                                DEFAULT_COMPRESSION_TYPE)\n            compression_type = CMTYPE_ALIASES.get(compression_type,\n                                                  compression_type)\n\n        # If the input image header had BSCALE/BZERO cards, then insert\n        # them in the table header.\n\n        if image_header:\n            bzero = image_header.get('BZERO', 0.0)\n            bscale = image_header.get('BSCALE', 1.0)\n            after_keyword = 'EXTNAME'\n\n            if bscale != 1.0:\n                self._header.set('BSCALE', bscale, after=after_keyword)\n                after_keyword = 'BSCALE'\n\n            if bzero != 0.0:\n                self._header.set('BZERO', bzero, after=after_keyword)\n\n            bitpix_comment = image_header.comments['BITPIX']\n            naxis_comment = image_header.comments['NAXIS']\n        else:\n            bitpix_comment = 'data type of original image'\n            naxis_comment = 'dimension of original image'\n\n        # Set the label for the first column in the table\n\n        self._header.set('TTYPE1', 'COMPRESSED_DATA', 'label for field 1',\n                         after='TFIELDS')\n\n        # Set the data format for the first column.  It is dependent\n        # on the requested compression type.\n\n        if compression_type == 'PLIO_1':\n            tform1 = '1QI' if huge_hdu else '1PI'\n        else:\n            tform1 = '1QB' if huge_hdu else '1PB'\n\n        self._header.set('TFORM1', tform1,\n                         'data format of field: variable length array',\n                         after='TTYPE1')\n\n        # Create the first column for the table.  This column holds the\n        # compressed data.\n        col1 = Column(name=self._header['TTYPE1'], format=tform1)\n\n        # Create the additional columns required for floating point\n        # data and calculate the width of the output table.\n\n        zbitpix = self._image_header['BITPIX']\n\n        if zbitpix < 0 and quantize_level != 0.0:\n            # floating point image has 'COMPRESSED_DATA',\n            # 'UNCOMPRESSED_DATA', 'ZSCALE', and 'ZZERO' columns (unless using\n            # lossless compression, per CFITSIO)\n            ncols = 4\n\n            # CFITSIO 3.28 and up automatically use the GZIP_COMPRESSED_DATA\n            # store floating point data that couldn't be quantized, instead\n            # of the UNCOMPRESSED_DATA column.  There's no way to control\n            # this behavior so the only way to determine which behavior will\n            # be employed is via the CFITSIO version\n\n            if CFITSIO_SUPPORTS_GZIPDATA:\n                ttype2 = 'GZIP_COMPRESSED_DATA'\n                # The required format for the GZIP_COMPRESSED_DATA is actually\n                # missing from the standard docs, but CFITSIO suggests it\n                # should be 1PB, which is logical.\n                tform2 = '1QB' if huge_hdu else '1PB'\n            else:\n                # Q format is not supported for UNCOMPRESSED_DATA columns.\n                ttype2 = 'UNCOMPRESSED_DATA'\n                if zbitpix == 8:\n                    tform2 = '1QB' if huge_hdu else '1PB'\n                elif zbitpix == 16:\n                    tform2 = '1QI' if huge_hdu else '1PI'\n                elif zbitpix == 32:\n                    tform2 = '1QJ' if huge_hdu else '1PJ'\n                elif zbitpix == -32:\n                    tform2 = '1QE' if huge_hdu else '1PE'\n                else:\n                    tform2 = '1QD' if huge_hdu else '1PD'\n\n            # Set up the second column for the table that will hold any\n            # uncompressable data.\n            self._header.set('TTYPE2', ttype2, 'label for field 2',\n                             after='TFORM1')\n\n            self._header.set('TFORM2', tform2,\n                             'data format of field: variable length array',\n                             after='TTYPE2')\n\n            col2 = Column(name=ttype2, format=tform2)\n\n            # Set up the third column for the table that will hold\n            # the scale values for quantized data.\n            self._header.set('TTYPE3', 'ZSCALE', 'label for field 3',\n                             after='TFORM2')\n            self._header.set('TFORM3', '1D',\n                             'data format of field: 8-byte DOUBLE',\n                             after='TTYPE3')\n            col3 = Column(name=self._header['TTYPE3'],\n                          format=self._header['TFORM3'])\n\n            # Set up the fourth column for the table that will hold\n            # the zero values for the quantized data.\n            self._header.set('TTYPE4', 'ZZERO', 'label for field 4',\n                             after='TFORM3')\n            self._header.set('TFORM4', '1D',\n                             'data format of field: 8-byte DOUBLE',\n                             after='TTYPE4')\n            after = 'TFORM4'\n            col4 = Column(name=self._header['TTYPE4'],\n                          format=self._header['TFORM4'])\n\n            # Create the ColDefs object for the table\n            cols = ColDefs([col1, col2, col3, col4])\n        else:\n            # default table has just one 'COMPRESSED_DATA' column\n            ncols = 1\n            after = 'TFORM1'\n\n            # remove any header cards for the additional columns that\n            # may be left over from the previous data\n            to_remove = ['TTYPE2', 'TFORM2', 'TTYPE3', 'TFORM3', 'TTYPE4',\n                         'TFORM4']\n\n            for k in to_remove:\n                try:\n                    del self._header[k]\n                except KeyError:\n                    pass\n\n            # Create the ColDefs object for the table\n            cols = ColDefs([col1])\n\n        # Update the table header with the width of the table, the\n        # number of fields in the table, the indicator for a compressed\n        # image HDU, the data type of the image data and the number of\n        # dimensions in the image data array.\n        self._header.set('NAXIS1', cols.dtype.itemsize,\n                         'width of table in bytes')\n        self._header.set('TFIELDS', ncols, 'number of fields in each row')\n        self._header.set('ZIMAGE', True, 'extension contains compressed image',\n                         after=after)\n        self._header.set('ZBITPIX', zbitpix,\n                         bitpix_comment, after='ZIMAGE')\n        self._header.set('ZNAXIS', self._image_header['NAXIS'], naxis_comment,\n                         after='ZBITPIX')\n\n        # Strip the table header of all the ZNAZISn and ZTILEn keywords\n        # that may be left over from the previous data\n\n        idx = 1\n        while True:\n            try:\n                del self._header['ZNAXIS' + str(idx)]\n                del self._header['ZTILE' + str(idx)]\n                idx += 1\n            except KeyError:\n                break\n\n        # Verify that any input tile size parameter is the appropriate\n        # size to match the HDU's data.\n\n        naxis = self._image_header['NAXIS']\n\n        if not tile_size:\n            tile_size = []\n        elif len(tile_size) != naxis:\n            warnings.warn('Provided tile size not appropriate for the data.  '\n                          'Default tile size will be used.', AstropyUserWarning)\n            tile_size = []\n\n        # Set default tile dimensions for HCOMPRESS_1\n\n        if compression_type == 'HCOMPRESS_1':\n            if (self._image_header['NAXIS1'] < 4 or\n                    self._image_header['NAXIS2'] < 4):\n                raise ValueError('Hcompress minimum image dimension is '\n                                 '4 pixels')\n            elif tile_size:\n                if tile_size[0] < 4 or tile_size[1] < 4:\n                    # user specified tile size is too small\n                    raise ValueError('Hcompress minimum tile dimension is '\n                                     '4 pixels')\n                major_dims = len([ts for ts in tile_size if ts > 1])\n                if major_dims > 2:\n                    raise ValueError(\n                        'HCOMPRESS can only support 2-dimensional tile sizes.'\n                        'All but two of the tile_size dimensions must be set '\n                        'to 1.')\n\n            if tile_size and (tile_size[0] == 0 and tile_size[1] == 0):\n                # compress the whole image as a single tile\n                tile_size[0] = self._image_header['NAXIS1']\n                tile_size[1] = self._image_header['NAXIS2']\n\n                for i in range(2, naxis):\n                    # set all higher tile dimensions = 1\n                    tile_size[i] = 1\n            elif not tile_size:\n                # The Hcompress algorithm is inherently 2D in nature, so the\n                # row by row tiling that is used for other compression\n                # algorithms is not appropriate.  If the image has less than 30\n                # rows, then the entire image will be compressed as a single\n                # tile.  Otherwise the tiles will consist of 16 rows of the\n                # image.  This keeps the tiles to a reasonable size, and it\n                # also includes enough rows to allow good compression\n                # efficiency.  It the last tile of the image happens to contain\n                # less than 4 rows, then find another tile size with between 14\n                # and 30 rows (preferably even), so that the last tile has at\n                # least 4 rows.\n\n                # 1st tile dimension is the row length of the image\n                tile_size.append(self._image_header['NAXIS1'])\n\n                if self._image_header['NAXIS2'] <= 30:\n                    tile_size.append(self._image_header['NAXIS1'])\n                else:\n                    # look for another good tile dimension\n                    naxis2 = self._image_header['NAXIS2']\n                    for dim in [16, 24, 20, 30, 28, 26, 22, 18, 14]:\n                        if naxis2 % dim == 0 or naxis2 % dim > 3:\n                            tile_size.append(dim)\n                            break\n                    else:\n                        tile_size.append(17)\n\n                for i in range(2, naxis):\n                    # set all higher tile dimensions = 1\n                    tile_size.append(1)\n\n            # check if requested tile size causes the last tile to have\n            # less than 4 pixels\n\n            remain = self._image_header['NAXIS1'] % tile_size[0]  # 1st dimen\n\n            if remain > 0 and remain < 4:\n                tile_size[0] += 1  # try increasing tile size by 1\n\n                remain = self._image_header['NAXIS1'] % tile_size[0]\n\n                if remain > 0 and remain < 4:\n                    raise ValueError('Last tile along 1st dimension has '\n                                     'less than 4 pixels')\n\n            remain = self._image_header['NAXIS2'] % tile_size[1]  # 2nd dimen\n\n            if remain > 0 and remain < 4:\n                tile_size[1] += 1  # try increasing tile size by 1\n\n                remain = self._image_header['NAXIS2'] % tile_size[1]\n\n                if remain > 0 and remain < 4:\n                    raise ValueError('Last tile along 2nd dimension has '\n                                     'less than 4 pixels')\n\n        # Set up locations for writing the next cards in the header.\n        last_znaxis = 'ZNAXIS'\n\n        if self._image_header['NAXIS'] > 0:\n            after1 = 'ZNAXIS1'\n        else:\n            after1 = 'ZNAXIS'\n\n        # Calculate the number of rows in the output table and\n        # write the ZNAXISn and ZTILEn cards to the table header.\n        nrows = 0\n\n        for idx, axis in enumerate(self._axes):\n            naxis = 'NAXIS' + str(idx + 1)\n            znaxis = 'ZNAXIS' + str(idx + 1)\n            ztile = 'ZTILE' + str(idx + 1)\n\n            if tile_size and len(tile_size) >= idx + 1:\n                ts = tile_size[idx]\n            else:\n                if ztile not in self._header:\n                    # Default tile size\n                    if not idx:\n                        ts = self._image_header['NAXIS1']\n                    else:\n                        ts = 1\n                else:\n                    ts = self._header[ztile]\n                tile_size.append(ts)\n\n            if not nrows:\n                nrows = (axis - 1) // ts + 1\n            else:\n                nrows *= ((axis - 1) // ts + 1)\n\n            if image_header and naxis in image_header:\n                self._header.set(znaxis, axis, image_header.comments[naxis],\n                                 after=last_znaxis)\n            else:\n                self._header.set(znaxis, axis,\n                                 'length of original image axis',\n                                 after=last_znaxis)\n\n            self._header.set(ztile, ts, 'size of tiles to be compressed',\n                             after=after1)\n            last_znaxis = znaxis\n            after1 = ztile\n\n        # Set the NAXIS2 header card in the table hdu to the number of\n        # rows in the table.\n        self._header.set('NAXIS2', nrows, 'number of rows in table')\n\n        self.columns = cols\n\n        # Set the compression parameters in the table header.\n\n        # First, setup the values to be used for the compression parameters\n        # in case none were passed in.  This will be either the value\n        # already in the table header for that parameter or the default\n        # value.\n        idx = 1\n\n        while True:\n            zname = 'ZNAME' + str(idx)\n            if zname not in self._header:\n                break\n            zval = 'ZVAL' + str(idx)\n            if self._header[zname] == 'NOISEBIT':\n                if quantize_level is None:\n                    quantize_level = self._header[zval]\n            if self._header[zname] == 'SCALE   ':\n                if hcomp_scale is None:\n                    hcomp_scale = self._header[zval]\n            if self._header[zname] == 'SMOOTH  ':\n                if hcomp_smooth is None:\n                    hcomp_smooth = self._header[zval]\n            idx += 1\n\n        if quantize_level is None:\n            quantize_level = DEFAULT_QUANTIZE_LEVEL\n\n        if hcomp_scale is None:\n            hcomp_scale = DEFAULT_HCOMP_SCALE\n\n        if hcomp_smooth is None:\n            hcomp_smooth = DEFAULT_HCOMP_SCALE\n\n        # Next, strip the table header of all the ZNAMEn and ZVALn keywords\n        # that may be left over from the previous data\n\n        idx = 1\n\n        while True:\n            zname = 'ZNAME' + str(idx)\n            if zname not in self._header:\n                break\n            zval = 'ZVAL' + str(idx)\n            del self._header[zname]\n            del self._header[zval]\n            idx += 1\n\n        # Finally, put the appropriate keywords back based on the\n        # compression type.\n\n        after_keyword = 'ZCMPTYPE'\n        idx = 1\n\n        if compression_type == 'RICE_1':\n            self._header.set('ZNAME1', 'BLOCKSIZE', 'compression block size',\n                             after=after_keyword)\n            self._header.set('ZVAL1', DEFAULT_BLOCK_SIZE, 'pixels per block',\n                             after='ZNAME1')\n\n            self._header.set('ZNAME2', 'BYTEPIX',\n                             'bytes per pixel (1, 2, 4, or 8)', after='ZVAL1')\n\n            if self._header['ZBITPIX'] == 8:\n                bytepix = 1\n            elif self._header['ZBITPIX'] == 16:\n                bytepix = 2\n            else:\n                bytepix = DEFAULT_BYTE_PIX\n\n            self._header.set('ZVAL2', bytepix,\n                             'bytes per pixel (1, 2, 4, or 8)',\n                             after='ZNAME2')\n            after_keyword = 'ZVAL2'\n            idx = 3\n        elif compression_type == 'HCOMPRESS_1':\n            self._header.set('ZNAME1', 'SCALE', 'HCOMPRESS scale factor',\n                             after=after_keyword)\n            self._header.set('ZVAL1', hcomp_scale, 'HCOMPRESS scale factor',\n                             after='ZNAME1')\n            self._header.set('ZNAME2', 'SMOOTH', 'HCOMPRESS smooth option',\n                             after='ZVAL1')\n            self._header.set('ZVAL2', hcomp_smooth, 'HCOMPRESS smooth option',\n                             after='ZNAME2')\n            after_keyword = 'ZVAL2'\n            idx = 3\n\n        if self._image_header['BITPIX'] < 0:   # floating point image\n            self._header.set('ZNAME' + str(idx), 'NOISEBIT',\n                             'floating point quantization level',\n                             after=after_keyword)\n            self._header.set('ZVAL' + str(idx), quantize_level,\n                             'floating point quantization level',\n                             after='ZNAME' + str(idx))\n\n            # Add the dither method and seed\n            if quantize_method:\n                if quantize_method not in [NO_DITHER, SUBTRACTIVE_DITHER_1,\n                                           SUBTRACTIVE_DITHER_2]:\n                    name = QUANTIZE_METHOD_NAMES[DEFAULT_QUANTIZE_METHOD]\n                    warnings.warn('Unknown quantization method provided.  '\n                                  'Default method (%s) used.' % name)\n                    quantize_method = DEFAULT_QUANTIZE_METHOD\n\n                if quantize_method == NO_DITHER:\n                    zquantiz_comment = 'No dithering during quantization'\n                else:\n                    zquantiz_comment = 'Pixel Quantization Algorithm'\n\n                self._header.set('ZQUANTIZ',\n                                 QUANTIZE_METHOD_NAMES[quantize_method],\n                                 zquantiz_comment,\n                                 after='ZVAL' + str(idx))\n            else:\n                # If the ZQUANTIZ keyword is missing the default is to assume\n                # no dithering, rather than whatever DEFAULT_QUANTIZE_METHOD\n                # is set to\n                quantize_method = self._header.get('ZQUANTIZ', NO_DITHER)\n\n                if isinstance(quantize_method, string_types):\n                    for k, v in iteritems(QUANTIZE_METHOD_NAMES):\n                        if v.upper() == quantize_method:\n                            quantize_method = k\n                            break\n                    else:\n                        quantize_method = NO_DITHER\n\n            if quantize_method == NO_DITHER:\n                if 'ZDITHER0' in self._header:\n                    # If dithering isn't being used then there's no reason to\n                    # keep the ZDITHER0 keyword\n                    del self._header['ZDITHER0']\n            else:\n                if dither_seed:\n                    dither_seed = self._generate_dither_seed(dither_seed)\n                elif 'ZDITHER0' in self._header:\n                    dither_seed = self._header['ZDITHER0']\n                else:\n                    dither_seed = self._generate_dither_seed(\n                            DEFAULT_DITHER_SEED)\n\n                self._header.set('ZDITHER0', dither_seed,\n                                 'dithering offset when quantizing floats',\n                                 after='ZQUANTIZ')\n\n        if image_header:\n            # Move SIMPLE card from the image header to the\n            # table header as ZSIMPLE card.\n\n            if 'SIMPLE' in image_header:\n                self._header.set('ZSIMPLE', image_header['SIMPLE'],\n                                 image_header.comments['SIMPLE'],\n                                 before='ZBITPIX')\n\n            # Move EXTEND card from the image header to the\n            # table header as ZEXTEND card.\n\n            if 'EXTEND' in image_header:\n                self._header.set('ZEXTEND', image_header['EXTEND'],\n                                 image_header.comments['EXTEND'])\n\n            # Move BLOCKED card from the image header to the\n            # table header as ZBLOCKED card.\n\n            if 'BLOCKED' in image_header:\n                self._header.set('ZBLOCKED', image_header['BLOCKED'],\n                                 image_header.comments['BLOCKED'])\n\n            # Move XTENSION card from the image header to the\n            # table header as ZTENSION card.\n\n            # Since we only handle compressed IMAGEs, ZTENSION should\n            # always be IMAGE, even if the caller has passed in a header\n            # for some other type of extension.\n            if 'XTENSION' in image_header:\n                self._header.set('ZTENSION', 'IMAGE',\n                                 image_header.comments['XTENSION'],\n                                 before='ZBITPIX')\n\n            # Move PCOUNT and GCOUNT cards from image header to the table\n            # header as ZPCOUNT and ZGCOUNT cards.\n\n            if 'PCOUNT' in image_header:\n                self._header.set('ZPCOUNT', image_header['PCOUNT'],\n                                 image_header.comments['PCOUNT'],\n                                 after=last_znaxis)\n\n            if 'GCOUNT' in image_header:\n                self._header.set('ZGCOUNT', image_header['GCOUNT'],\n                                 image_header.comments['GCOUNT'],\n                                 after='ZPCOUNT')\n\n            # Move CHECKSUM and DATASUM cards from the image header to the\n            # table header as XHECKSUM and XDATASUM cards.\n\n            if 'CHECKSUM' in image_header:\n                self._header.set('ZHECKSUM', image_header['CHECKSUM'],\n                                 image_header.comments['CHECKSUM'])\n\n            if 'DATASUM' in image_header:\n                self._header.set('ZDATASUM', image_header['DATASUM'],\n                                 image_header.comments['DATASUM'])\n        else:\n            # Move XTENSION card from the image header to the\n            # table header as ZTENSION card.\n\n            # Since we only handle compressed IMAGEs, ZTENSION should\n            # always be IMAGE, even if the caller has passed in a header\n            # for some other type of extension.\n            if 'XTENSION' in self._image_header:\n                self._header.set('ZTENSION', 'IMAGE',\n                                 self._image_header.comments['XTENSION'],\n                                 before='ZBITPIX')\n\n            # Move PCOUNT and GCOUNT cards from image header to the table\n            # header as ZPCOUNT and ZGCOUNT cards.\n\n            if 'PCOUNT' in self._image_header:\n                self._header.set('ZPCOUNT', self._image_header['PCOUNT'],\n                                 self._image_header.comments['PCOUNT'],\n                                 after=last_znaxis)\n\n            if 'GCOUNT' in self._image_header:\n                self._header.set('ZGCOUNT', self._image_header['GCOUNT'],\n                                 self._image_header.comments['GCOUNT'],\n                                 after='ZPCOUNT')\n\n        # When we have an image checksum we need to ensure that the same\n        # number of blank cards exist in the table header as there were in\n        # the image header.  This allows those blank cards to be carried\n        # over to the image header when the hdu is uncompressed.\n\n        if 'ZHECKSUM' in self._header:\n            required_blanks = image_header._countblanks()\n            image_blanks = self._image_header._countblanks()\n            table_blanks = self._header._countblanks()\n\n            for _ in range(required_blanks - image_blanks):\n                self._image_header.append()\n                table_blanks += 1\n\n            for _ in range(required_blanks - table_blanks):\n                self._header.append()"
        }
      ]
    },
    {
      "pr_number": 2711,
      "pr_title": "FITS: 'BLANK' keyword causes crash when reading data",
      "pr_body": "Hello,\nI already notice since a while an issue while trying to read, e.g., data fits cube (so simply (x,y,z))  see below for the full error which originate from 'hdu/image.py'.\nMy dummy, quickest fix was to add, right before the offending 'if':\n                `if blanks==False: blanks=np.array([False])`\n\nI don't know if it is really appropriate, but it does the trick for me.\n\nregards,\nGilles\n\n```\nIn [120]: array=py.getdata(cube_file)\n---------------------------------------------------------------------------\nAttributeError                            Traceback (most recent call last)\n<ipython-input-120-5d145d1683de> in <module>()\n----> 1 array=py.getdata(cube_file)\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/convenience.pyc in getdata(filename, *args, **kwargs)\n    186     hdulist, extidx = _getext(filename, mode, *args, **kwargs)\n    187     hdu = hdulist[extidx]\n--> 188     data = hdu.data\n    189     if data is None and extidx == 0:\n    190         try:\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/utils/misc.pyc in __get__(self, obj, owner)\n    277         key = self._fget.__name__\n    278         if key not in obj.__dict__:\n--> 279             val = self._fget(obj)\n    280             obj.__dict__[key] = val\n    281             return val\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/hdu/image.pyc in data(self)\n    213             return\n    214 \n--> 215         data = self._get_scaled_image_data(self._data_offset, self.shape)\n    216         self._update_header_scale_info(data.dtype)\n    217 \n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/hdu/image.pyc in _get_scaled_image_data(self, offset, shape)\n    582                 # So if the number of blank items is fewer than\n    583                 # len(raw_data.flat) / 8, using np.where will use less memory\n--> 584                 if blanks.sum() < len(blanks) / 8:\n    585                     blanks = np.where(blanks)\n    586 \n\nAttributeError: 'bool' object has no attribute 'sum'\n```\n",
      "issue_id": 2711,
      "issue_title": "FITS: 'BLANK' keyword causes crash when reading data",
      "issue_body": "Hello,\nI already notice since a while an issue while trying to read, e.g., data fits cube (so simply (x,y,z))  see below for the full error which originate from 'hdu/image.py'.\nMy dummy, quickest fix was to add, right before the offending 'if':\n                `if blanks==False: blanks=np.array([False])`\n\nI don't know if it is really appropriate, but it does the trick for me.\n\nregards,\nGilles\n\n```\nIn [120]: array=py.getdata(cube_file)\n---------------------------------------------------------------------------\nAttributeError                            Traceback (most recent call last)\n<ipython-input-120-5d145d1683de> in <module>()\n----> 1 array=py.getdata(cube_file)\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/convenience.pyc in getdata(filename, *args, **kwargs)\n    186     hdulist, extidx = _getext(filename, mode, *args, **kwargs)\n    187     hdu = hdulist[extidx]\n--> 188     data = hdu.data\n    189     if data is None and extidx == 0:\n    190         try:\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/utils/misc.pyc in __get__(self, obj, owner)\n    277         key = self._fget.__name__\n    278         if key not in obj.__dict__:\n--> 279             val = self._fget(obj)\n    280             obj.__dict__[key] = val\n    281             return val\n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/hdu/image.pyc in data(self)\n    213             return\n    214 \n--> 215         data = self._get_scaled_image_data(self._data_offset, self.shape)\n    216         self._update_header_scale_info(data.dtype)\n    217 \n\n/usr/local/lib/python2.7/dist-packages/astropy-0.3.2-py2.7-linux-x86_64.egg/astropy/io/fits/hdu/image.pyc in _get_scaled_image_data(self, offset, shape)\n    582                 # So if the number of blank items is fewer than\n    583                 # len(raw_data.flat) / 8, using np.where will use less memory\n--> 584                 if blanks.sum() < len(blanks) / 8:\n    585                     blanks = np.where(blanks)\n    586 \n\nAttributeError: 'bool' object has no attribute 'sum'\n```\n",
      "issue_closed_at": "2014-09-16T19:18:56Z",
      "base_commit": "0307f793cf700560673ff482d37de447958db437",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/image.py",
          "type": "line",
          "name": "line 1",
          "code": "# Licensed under a 3-clause BSD style license - see PYFITS.rst\n\nimport sys\nimport numpy as np\n\nfrom .base import DELAYED, _ValidHDU, ExtensionHDU\nfrom ..header import Header\nfrom ..util import (_is_pseudo_unsigned, _unsigned_zero, _is_int,\n                    _normalize_slice)\n\nfrom ....extern.six import string_types\nfrom ....extern.six.moves import xrange"
        },
        {
          "file": "astropy/io/fits/hdu/image.py",
          "type": "function",
          "name": "__init__",
          "class_name": "_KeyType",
          "code": "def __init__(self, npts, offset):\n        self.npts = npts\n        self.offset = offset"
        },
        {
          "file": "astropy/io/fits/hdu/image.py",
          "type": "function",
          "name": "_get_scaled_image_data",
          "class_name": "_ImageBaseHDU",
          "code": "def _get_scaled_image_data(self, offset, shape):\n        \"\"\"\n        Internal function for reading image data from a file and apply scale\n        factors to it.  Normally this is used for the entire image, but it\n        supports alternate offset/shape for Section support.\n        \"\"\"\n\n        code = _ImageBaseHDU.NumCode[self._orig_bitpix]\n\n        raw_data = self._get_raw_data(shape, code, offset)\n        raw_data.dtype = raw_data.dtype.newbyteorder('>')\n\n        if (self._orig_bzero == 0 and self._orig_bscale == 1 and\n                self._blank is None):\n            # No further conversion of the data is necessary\n            return raw_data\n\n        data = None\n        if not (self._orig_bzero == 0 and self._orig_bscale == 1):\n            data = self._convert_pseudo_unsigned(raw_data)\n\n        if data is None:\n            # In these cases, we end up with floating-point arrays and have to\n            # apply bscale and bzero. We may have to handle BLANK and convert\n            # to NaN in the resulting floating-point arrays.\n            if self._blank is not None:\n                blanks = raw_data.flat == self._blank\n                # The size of blanks in bytes is the number of elements in\n                # raw_data.flat.  However, if we use np.where instead we will\n                # only use 8 bytes for each index where the condition is true.\n                # So if the number of blank items is fewer than\n                # len(raw_data.flat) / 8, using np.where will use less memory\n                if blanks.sum() < len(blanks) / 8:\n                    blanks = np.where(blanks)\n\n            new_dtype = self._dtype_for_bitpix()\n            if new_dtype is not None:\n                data = np.array(raw_data, dtype=new_dtype)\n            else:  # floating point cases\n                if self._file.memmap:\n                    data = raw_data.copy()\n                # if not memmap, use the space already in memory\n                else:\n                    data = raw_data\n\n            del raw_data\n\n            if self._orig_bscale != 1:\n                np.multiply(data, self._orig_bscale, data)\n            if self._orig_bzero != 0:\n                data += self._orig_bzero\n\n            if self._blank is not None:\n                data.flat[blanks] = np.nan\n\n        return data"
        }
      ]
    },
    {
      "pr_number": 13842,
      "pr_title": "Ensure we get a new mixin instance even when copy=False",
      "pr_body": "<!-- This comments are hidden when you submit the pull request,\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<!-- If you are new or need to be re-acquainted with Astropy\r\ncontributing workflow, please see\r\nhttp://docs.astropy.org/en/latest/development/workflow/development_workflow.html .\r\nThere is even a practical example at\r\nhttps://docs.astropy.org/en/latest/development/workflow/git_edit_workflow_examples.html#astropy-fix-example . -->\r\n\r\n<!-- Astropy coding style guidelines can be found here:\r\nhttps://docs.astropy.org/en/latest/development/codeguide.html#coding-style-conventions\r\nOur testing infrastructure enforces to follow a subset of the PEP8 to be\r\nfollowed. You can check locally whether your changes have followed these by\r\nrunning the following command:\r\n\r\ntox -e codestyle\r\n\r\n-->\r\n\r\n<!-- Please just have a quick search on GitHub to see if a similar\r\npull request has already been posted.\r\nWe have old closed pull requests that might provide useful code or ideas\r\nthat directly tie in with your pull request. -->\r\n\r\n<!-- We have several automatic features that run when a pull request is open.\r\nThey can appear daunting but do not worry because maintainers will help\r\nyou navigate them, if necessary. -->\r\n\r\n### Description\r\n<!-- Provide a general description of what your pull request does.\r\nComplete the following sentence and add relevant details as you see fit. -->\r\n\r\n<!-- In addition please ensure that the pull request title is descriptive\r\nand allows maintainers to infer the applicable subpackage(s). -->\r\n\r\n<!-- READ THIS FOR MANUAL BACKPORT FROM A MAINTAINER:\r\nApply \"skip-basebranch-check\" label **before** you open the PR! -->\r\n\r\nThis pull request ensures that when a mixin column of one table is added to another with `copy=False`, the columns will, as requested, share the data, but ensure that they are not the same instance.\r\n\r\nThis turned out to be much easier than worried about in #13840 - just take a full slice `[:]` of the column (which every mixin has to support).\r\n\r\n<!-- If the pull request closes any open issues you can add this.\r\nIf you replace <Issue Number> with a number, GitHub will automatically link it.\r\nIf this pull request is unrelated to any issues, please remove\r\nthe following line. -->\r\n\r\nFixes #13840\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. Codestyle issues can be fixed by the [bot](https://docs.astropy.org/en/latest/development/workflow/development_workflow.html#pre-commit).\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. If this is a manual backport, use the `skip-changelog-checks` label unless special changelog handling is necessary.\r\n- [x] Is this a big PR that makes a \"What's new?\" entry worthwhile and if so, is (1) a \"what's new\" entry included in this PR and (2) the \"whatsnew-needed\" label applied?\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.\r\n",
      "issue_id": 13840,
      "issue_title": "Creating a mixin column in a new table from columns of another table renames columns in original table.",
      "issue_body": "### Description\r\n\r\nConsider the following code, where a subset of columns from another table should be included in a new table with new names, prerably without copying the actual payload data:\r\n\r\n```python\r\nfrom astropy.table import QTable, Table\r\nimport astropy.units as u\r\n\r\n\r\ntable1 = QTable({\r\n    'foo': [1, 2, 3] * u.deg,\r\n    'bar': [4, 5, 6] * u.m,\r\n    'baz': [7, 8, 9] * u.TeV,\r\n})\r\n\r\nprint(table1.colnames)\r\ntable2 = QTable({\r\n    \"new\": table1[\"foo\"],\r\n    \"name\": table1[\"bar\"]\r\n}, copy=False)\r\nprint(table1.colnames)\r\n```\r\n\r\nIf any of the two classes or both are a `Table`, not a `QTable`, the code works as expected.\r\n\r\n### Expected behavior\r\n\r\nData in the columns is not copied, but column names in original table stay the same.\r\n\r\n```\r\n['foo', 'bar', 'baz']\r\n['foo', 'bar', 'baz']\r\n```\r\n\r\n### Actual behavior\r\n\r\nColumn names do change in both tables:\r\n\r\n```\r\n['foo', 'bar', 'baz']\r\n['new', 'name', 'baz']\r\n```\r\n\r\n### Steps to Reproduce\r\n\r\nSee above.\r\n\r\n### System Details\r\n<!-- Even if you do not think this is necessary, it is useful information for the maintainers.\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 numpy; print(\"Numpy\", numpy.__version__)\r\nimport erfa; print(\"pyerfa\", erfa.__version__)\r\nimport astropy; print(\"astropy\", astropy.__version__)\r\nimport scipy; print(\"Scipy\", scipy.__version__)\r\nimport matplotlib; print(\"Matplotlib\", matplotlib.__version__)\r\n-->\r\n\r\n```\r\nLinux-5.15.71-1-MANJARO-x86_64-with-glibc2.36\r\nPython 3.10.6 | packaged by conda-forge | (main, Aug 22 2022, 20:35:26) [GCC 10.4.0]\r\nNumpy 1.23.3\r\npyerfa 2.0.0.1\r\nastropy 5.1\r\nScipy 1.9.1\r\nMatplotlib 3.6.1\r\n```\r\n\r\n(also tested with current `main` branch)",
      "issue_closed_at": "2022-10-20T10:44:42Z",
      "base_commit": "3b448815e21b117d34fe63007b8ef63ee084fefb",
      "changes": [
        {
          "file": "astropy/table/table.py",
          "type": "function",
          "name": "_convert_data_to_col",
          "class_name": "Table",
          "code": "def _convert_data_to_col(self, data, copy=True, default_name=None, dtype=None, name=None):\n        \"\"\"\n        Convert any allowed sequence data ``col`` to a column object that can be used\n        directly in the self.columns dict.  This could be a Column, MaskedColumn,\n        or mixin column.\n\n        The final column name is determined by::\n\n            name or data.info.name or def_name\n\n        If ``data`` has no ``info`` then ``name = name or def_name``.\n\n        The behavior of ``copy`` for Column objects is:\n        - copy=True: new class instance with a copy of data and deep copy of meta\n        - copy=False: new class instance with same data and a key-only copy of meta\n\n        For mixin columns:\n        - copy=True: new class instance with copy of data and deep copy of meta\n        - copy=False: original instance (no copy at all)\n\n        Parameters\n        ----------\n        data : object (column-like sequence)\n            Input column data\n        copy : bool\n            Make a copy\n        default_name : str\n            Default name\n        dtype : np.dtype or None\n            Data dtype\n        name : str or None\n            Column name\n\n        Returns\n        -------\n        col : Column, MaskedColumn, mixin-column type\n            Object that can be used as a column in self\n        \"\"\"\n\n        data_is_mixin = self._is_mixin_for_table(data)\n        masked_col_cls = (self.ColumnClass\n                          if issubclass(self.ColumnClass, self.MaskedColumn)\n                          else self.MaskedColumn)\n\n        try:\n            data0_is_mixin = self._is_mixin_for_table(data[0])\n        except Exception:\n            # Need broad exception, cannot predict what data[0] raises for arbitrary data\n            data0_is_mixin = False\n\n        # If the data is not an instance of Column or a mixin class, we can\n        # check the registry of mixin 'handlers' to see if the column can be\n        # converted to a mixin class\n        if (handler := get_mixin_handler(data)) is not None:\n            original_data = data\n            data = handler(data)\n            if not (data_is_mixin := self._is_mixin_for_table(data)):\n                fully_qualified_name = (original_data.__class__.__module__ + '.'\n                                        + original_data.__class__.__name__)\n                raise TypeError('Mixin handler for object of type '\n                                f'{fully_qualified_name} '\n                                'did not return a valid mixin column')\n\n        # Get the final column name using precedence.  Some objects may not\n        # have an info attribute. Also avoid creating info as a side effect.\n        if not name:\n            if isinstance(data, Column):\n                name = data.name or default_name\n            elif 'info' in getattr(data, '__dict__', ()):\n                name = data.info.name or default_name\n            else:\n                name = default_name\n\n        if isinstance(data, Column):\n            # If self.ColumnClass is a subclass of col, then \"upgrade\" to ColumnClass,\n            # otherwise just use the original class.  The most common case is a\n            # table with masked=True and ColumnClass=MaskedColumn.  Then a Column\n            # gets upgraded to MaskedColumn, but the converse (pre-4.0) behavior\n            # of downgrading from MaskedColumn to Column (for non-masked table)\n            # does not happen.\n            col_cls = self._get_col_cls_for_table(data)\n\n        elif data_is_mixin:\n            # Copy the mixin column attributes if they exist since the copy below\n            # may not get this attribute.\n            col = col_copy(data, copy_indices=self._init_indices) if copy else data\n            col.info.name = name\n            return col\n\n        elif data0_is_mixin:\n            # Handle case of a sequence of a mixin, e.g. [1*u.m, 2*u.m].\n            try:\n                col = data[0].__class__(data)\n                col.info.name = name\n                return col\n            except Exception:\n                # If that didn't work for some reason, just turn it into np.array of object\n                data = np.array(data, dtype=object)\n                col_cls = self.ColumnClass\n\n        elif isinstance(data, (np.ma.MaskedArray, Masked)):\n            # Require that col_cls be a subclass of MaskedColumn, remembering\n            # that ColumnClass could be a user-defined subclass (though more-likely\n            # could be MaskedColumn).\n            col_cls = masked_col_cls\n\n        elif data is None:\n            # Special case for data passed as the None object (for broadcasting\n            # to an object column). Need to turn data into numpy `None` scalar\n            # object, otherwise `Column` interprets data=None as no data instead\n            # of a object column of `None`.\n            data = np.array(None)\n            col_cls = self.ColumnClass\n\n        elif not hasattr(data, 'dtype'):\n            # `data` is none of the above, convert to numpy array or MaskedArray\n            # assuming only that it is a scalar or sequence or N-d nested\n            # sequence. This function is relatively intricate and tries to\n            # maintain performance for common cases while handling things like\n            # list input with embedded np.ma.masked entries. If `data` is a\n            # scalar then it gets returned unchanged so the original object gets\n            # passed to `Column` later.\n            data = _convert_sequence_data_to_array(data, dtype)\n            copy = False  # Already made a copy above\n            col_cls = masked_col_cls if isinstance(data, np.ma.MaskedArray) else self.ColumnClass\n\n        else:\n            col_cls = self.ColumnClass\n\n        try:\n            col = col_cls(name=name, data=data, dtype=dtype,\n                          copy=copy, copy_indices=self._init_indices)\n        except Exception:\n            # Broad exception class since we don't know what might go wrong\n            raise ValueError('unable to convert data to Column for Table')\n\n        col = self._convert_col_for_table(col)\n\n        return col"
        },
        {
          "file": "astropy/table/table_helpers.py",
          "type": "function",
          "name": "__getitem__",
          "class_name": "ArrayWrapper",
          "code": "def __getitem__(self, item):\n        if isinstance(item, (int, np.integer)):\n            out = self.data[item]\n        else:\n            out = self.__class__(self.data[item])\n            if 'info' in self.__dict__:\n                out.info = self.info\n        return out"
        }
      ]
    },
    {
      "pr_number": 7157,
      "pr_title": "Alternate fix for FITS regression with table keywords",
      "pr_body": "This is an alternative to https://github.com/astropy/astropy/pull/7153 to fix #7145 - basically it just prevents time-related keywords from being automatically removed from the Header. Time-related keywords can still be overwritten if the corresponding attributes are set on Column objects, but this doesn't affect backward-compatibility since those Column attributes didn't exist before. This approach thus guarantees full backward-compatibility (unless I've overlooked something), and emits a deprecation warning to mention that these keywords will be dropped in future. Example usage:\r\n\r\n```python\r\nIn [1]: import astropy\r\n   ...: from astropy.io import fits\r\n   ...: import numpy as np\r\n   ...: \r\n   ...: def create_column_descriptions():\r\n   ...:     col = []\r\n   ...:     col.append(fits.Column(name=\"TIME\", format=\"1E\", unit=\"s\"))\r\n   ...:     col.append(fits.Column(name=\"RAWX\", format=\"1I\", unit=\"pixel\"))\r\n   ...:     cd = fits.ColDefs(col)\r\n   ...: \r\n   ...:     return cd\r\n   ...: \r\n   ...: def create_header():\r\n   ...:     hdr = fits.Header()\r\n   ...:     hdr['RA'] = (1.581250000000E+00, 'RA of reference aperture center ')\r\n   ...:     hdr['DEC'] = (2.020277777778E+01, 'Declination of reference aperture center')\r\n   ...:     hdr['TCTYP2'] = ('RA---TAN', 'axis type for dimension 1')\r\n   ...:     hdr['TCRVL2'] = (-999.0, 'sky coordinates of 1st axis')\r\n   ...: \r\n   ...:     return hdr\r\n   ...: \r\n   ...: columns = create_column_descriptions()\r\n   ...: header = create_header()\r\n   ...: \r\n\r\nIn [2]: print(header.tostring(sep='\\n'))\r\n   ...: print(columns)\r\n   ...: \r\nRA      =              1.58125 / RA of reference aperture center                \r\nDEC     =       20.20277777778 / Declination of reference aperture center       \r\nTCTYP2  = 'RA---TAN'           / axis type for dimension 1                      \r\nTCRVL2  =               -999.0 / sky coordinates of 1st axis                    \r\nEND                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         \r\nColDefs(\r\n    name = 'TIME'; format = '1E'; unit = 's'\r\n    name = 'RAWX'; format = '1I'; unit = 'pixel'\r\n)\r\n```\r\n\r\nSo far so good. Let's make a new HDU from the columns and header:\r\n\r\n```python\r\nIn [3]: hdu = fits.BinTableHDU.from_columns(columns, header)\r\n   ...: \r\n/Users/tom/Dropbox/Code/Astropy/astropy/astropy/io/fits/hdu/table.py:341: UserWarning: The following keywords are now recognized as special column-related attributes and should be set via the Column objects: TCRVLn, TCTYPn. In future, these values will be dropped from manually specified headers automatically and replaced with values generated based on the Column objects.\r\n  \"Column objects.\".format(keys))\r\n```\r\n\r\nWe see that this causes the above warning. The header still contains those keywords, and the columns don't (consistent with previous behavior):\r\n\r\n```\r\nIn [4]: print(hdu.header.tostring(sep='\\n'))\r\n   ...: print(hdu.columns)\r\n   ...: \r\nXTENSION= 'BINTABLE'           / binary table extension                         \r\nBITPIX  =                    8 / array data type                                \r\nNAXIS   =                    2 / number of array dimensions                     \r\nNAXIS1  =                    6 / length of dimension 1                          \r\nNAXIS2  =                    0 / length of dimension 2                          \r\nPCOUNT  =                    0 / number of group parameters                     \r\nGCOUNT  =                    1 / number of groups                               \r\nTFIELDS =                    2 / number of table fields                         \r\nRA      =              1.58125 / RA of reference aperture center                \r\nDEC     =       20.20277777778 / Declination of reference aperture center       \r\nTCTYP2  = 'RA---TAN'           / axis type for dimension 1                      \r\nTCRVL2  =               -999.0 / sky coordinates of 1st axis                    \r\nTTYPE1  = 'TIME    '                                                            \r\nTFORM1  = '1E      '                                                            \r\nTUNIT1  = 's       '                                                            \r\nTTYPE2  = 'RAWX    '                                                            \r\nTFORM2  = '1I      '                                                            \r\nTUNIT2  = 'pixel   '                                                            \r\nEND                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           \r\nColDefs(\r\n    name = 'TIME'; format = '1E'; unit = 's'\r\n    name = 'RAWX'; format = '1I'; unit = 'pixel'\r\n)\r\n```\r\n\r\nNow let's write out the file and read in back in:\r\n\r\n```python\r\nIn [5]: hdu.writeto('time.fits', overwrite=True)\r\n   ...: \r\n   ...: hdu2 = fits.open('time.fits')[1]\r\n   ...: \r\n\r\nIn [6]: print(hdu2.header.tostring(sep='\\n'))\r\n   ...: print(hdu2.columns)\r\n   ...: \r\nXTENSION= 'BINTABLE'           / binary table extension                         \r\nBITPIX  =                    8 / array data type                                \r\nNAXIS   =                    2 / number of array dimensions                     \r\nNAXIS1  =                    6 / length of dimension 1                          \r\nNAXIS2  =                    0 / length of dimension 2                          \r\nPCOUNT  =                    0 / number of group parameters                     \r\nGCOUNT  =                    1 / number of groups                               \r\nTFIELDS =                    2 / number of table fields                         \r\nRA      =              1.58125 / RA of reference aperture center                \r\nDEC     =       20.20277777778 / Declination of reference aperture center       \r\nTCTYP2  = 'RA---TAN'           / axis type for dimension 1                      \r\nTCRVL2  =               -999.0 / sky coordinates of 1st axis                    \r\nTTYPE1  = 'TIME    '                                                            \r\nTFORM1  = '1E      '                                                            \r\nTUNIT1  = 's       '                                                            \r\nTTYPE2  = 'RAWX    '                                                            \r\nTFORM2  = '1I      '                                                            \r\nTUNIT2  = 'pixel   '                                                            \r\nEND                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           \r\nColDefs(\r\n    name = 'TIME'; format = '1E'; unit = 's'\r\n    name = 'RAWX'; format = '1I'; unit = 'pixel'; coord_type = 'RA---TAN'; coord_ref_value = -999.0\r\n)\r\n```\r\n\r\nAt this point, the new column attributes are picked up (as expected) and they are still in the header as everything is now in sync. No warning was emitted when reading a file.\r\n\r\nLet's just set a column attribute:\r\n\r\n```python\r\nIn [9]: hdu2.columns[1].coord_type = 'DEC--TAN'\r\n\r\nIn [10]: hdu2.header['TCTYP2']\r\nOut[10]: 'DEC--TAN'\r\n```\r\n\r\nThe attributes in the header are now kept in sync and are not cleared by default when initializing from a header.",
      "issue_id": 7145,
      "issue_title": "New FITS table column attributes break existing code",
      "issue_body": "Change in behaviour of binTableHDU.fromcolumns:\r\n\r\nVersion 2.0.3: passing a header containing keywords with names like \"TCRPX2\" to from_columns creates an hdu object with these keywords in the header\r\n\r\nVersion 3.dev: passing a header with such keywords results in them not making it to the header of the output\r\n\r\nThis code demonstrates:\r\n\r\n```python\r\n#! /usr/bin/env python\r\nimport astropy\r\nfrom astropy.io import fits\r\nimport numpy as np\r\n\r\ndef test_fromcolumns():\r\n    print(astropy.__version__)\r\n    cd = create_column_descriptions()\r\n    header = create_header()\r\n\r\n    print(\"Header that is input to fits.BinTableHDU.from_columns:\")\r\n    print(header)\r\n\r\n    hdu = fits.BinTableHDU.from_columns(cd, header)\r\n\r\n    print(\"Header of hdu object returned from fits.BinTableHDU.from_columns:\")\r\n    print(hdu.header)\r\n\r\n    return\r\n\r\ndef create_column_descriptions():\r\n    col = []\r\n    col.append(fits.Column(name=\"TIME\", format=\"1E\", unit=\"s\"))\r\n    col.append(fits.Column(name=\"RAWX\", format=\"1I\", unit=\"pixel\"))\r\n    col.append(fits.Column(name=\"RAWY\", format=\"1I\", unit=\"pixel\"))\r\n    cd = fits.ColDefs(col)\r\n\r\n    return cd\r\n\r\ndef create_header():\r\n    hdr = fits.Header()\r\n    hdr['RA'] = (1.581250000000E+00, 'RA of reference aperture center ')\r\n    hdr['DEC'] = (2.020277777778E+01, 'Declination of reference aperture center')\r\n    hdr['TCTYP2'] = ('RA---TAN', 'axis type for dimension 1')\r\n    hdr['TCTYP3'] = ('ANGLE   ', 'axis type for dimension 2')\r\n    hdr['TCRVL2'] = (-999.0, 'sky coordinates of 1st axis')\r\n    hdr['TCRVL3'] = (-999.0, 'sky coordinates of 2nd axis')\r\n    hdr['TCRPX2'] = (1.0, 'axis1 pixel of tangent plane direction')\r\n    hdr['TCRPX3'] = (1.0, 'axis2 pixel of tangent plane direction')\r\n    hdr['TALEN2'] = (16384, 'length of axis 1')\r\n    hdr['TALEN3'] = (1024, 'length of axis 2')\r\n    hdr['TC2_2'] = (0.0, 'partial of first axis coordinate w.r.t. x')\r\n    hdr['TC2_3'] = (0.0, 'partial of first axis coordinate w.r.t. y')\r\n    hdr['TC3_2'] = (0.0, 'partial of second axis coordinate w.r.t. x')\r\n    hdr['TC3_3'] = (0.0, 'partial of second axis coordinate w.r.t. y')\r\n    hdr['TCUNI2'] = ('angstrom', 'units of first coordinate value')\r\n    hdr['TCUNI3'] = ('deg     ', 'units of second coordinate value')\r\n\r\n    return hdr\r\n\r\n```\r\n\r\nEdit: Added syntax highlighting.",
      "issue_closed_at": "2018-02-05T17:19:56Z",
      "base_commit": "c7651ff2c95e7bfa62c326a4453e12ef632444ee",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/table.py",
          "type": "line",
          "name": "line 29",
          "code": "from ..util import _is_int, _str_to_num\n\nfrom ....utils import lazyproperty\nfrom ....utils.exceptions import AstropyUserWarning\nfrom ....utils.decorators import deprecated_renamed_argument\n\n"
        },
        {
          "file": "astropy/io/fits/hdu/table.py",
          "type": "class",
          "name": "_TableBaseHDU",
          "code": "class _TableBaseHDU(ExtensionHDU, _TableLikeHDU):\n    \"\"\"\n    FITS table extension base HDU class.\n\n    Parameters\n    ----------\n    data : array\n        Data to be used.\n    header : `Header` instance\n        Header to be used.\n    name : str\n        Name to be populated in ``EXTNAME`` keyword.\n    uint : bool, optional\n        Set to `True` if the table contains unsigned integer columns.\n    ver : int > 0 or None, optional\n        The ver of the HDU, will be the value of the keyword ``EXTVER``.\n        If not given or None, it defaults to the value of the ``EXTVER``\n        card of the ``header`` or 1.\n        (default: None)\n    character_as_bytes : bool\n        Whether to return bytes for string columns. By default this is `False`\n        and (unicode) strings are returned, but this does not respect memory\n        mapping and loads the whole column in memory when accessed.\n    \"\"\"\n\n    _manages_own_heap = False\n    \"\"\"\n    This flag implies that when writing VLA tables (P/Q format) the heap\n    pointers that go into P/Q table columns should not be reordered or\n    rearranged in any way by the default heap management code.\n\n    This is included primarily as an optimization for compressed image HDUs\n    which perform their own heap maintenance.\n    \"\"\"\n\n    def __init__(self, data=None, header=None, name=None, uint=False, ver=None,\n                 character_as_bytes=False):\n\n        super().__init__(data=data, header=header, name=name, ver=ver)\n\n        if header is not None and not isinstance(header, Header):\n            raise ValueError('header must be a Header object.')\n\n        self._uint = uint\n        self._character_as_bytes = character_as_bytes\n\n        if data is DELAYED:\n            # this should never happen\n            if header is None:\n                raise ValueError('No header to setup HDU.')\n\n            # if the file is read the first time, no need to copy, and keep it\n            # unchanged\n            else:\n                self._header = header\n        else:\n            # construct a list of cards of minimal header\n            cards = [\n                ('XTENSION', '', ''),\n                ('BITPIX', 8, 'array data type'),\n                ('NAXIS', 2, 'number of array dimensions'),\n                ('NAXIS1', 0, 'length of dimension 1'),\n                ('NAXIS2', 0, 'length of dimension 2'),\n                ('PCOUNT', 0, 'number of group parameters'),\n                ('GCOUNT', 1, 'number of groups'),\n                ('TFIELDS', 0, 'number of table fields')]\n\n            if header is not None:\n                # Make a \"copy\" (not just a view) of the input header, since it\n                # may get modified.  the data is still a \"view\" (for now)\n                hcopy = header.copy(strip=True)\n                cards.extend(hcopy.cards)\n\n            self._header = Header(cards)\n\n            if isinstance(data, np.ndarray) and data.dtype.fields is not None:\n                # self._data_type is FITS_rec.\n                if isinstance(data, self._data_type):\n                    self.data = data\n                else:\n                    self.data = self._data_type.from_columns(data)\n\n                # TODO: Too much of the code in this class uses header keywords\n                # in making calculations related to the data size.  This is\n                # unreliable, however, in cases when users mess with the header\n                # unintentionally--code that does this should be cleaned up.\n                self._header['NAXIS1'] = self.data._raw_itemsize\n                self._header['NAXIS2'] = self.data.shape[0]\n                self._header['TFIELDS'] = len(self.data._coldefs)\n\n                self.columns = self.data._coldefs\n                self.update()\n\n                with suppress(TypeError, AttributeError):\n                    # Make the ndarrays in the Column objects of the ColDefs\n                    # object of the HDU reference the same ndarray as the HDU's\n                    # FITS_rec object.\n                    for idx, col in enumerate(self.columns):\n                        col.array = self.data.field(idx)\n\n                    # Delete the _arrays attribute so that it is recreated to\n                    # point to the new data placed in the column objects above\n                    del self.columns._arrays\n            elif data is None:\n                pass\n            else:\n                raise TypeError('Table data has incorrect type.')\n\n        if not (isinstance(self._header[0], str) and\n                self._header[0].rstrip() == self._extension):\n            self._header[0] = (self._extension, self._ext_comment)\n\n        # Ensure that the correct EXTNAME is set on the new header if one was\n        # created, or that it overrides the existing EXTNAME if different\n        if name:\n            self.name = name\n        if ver is not None:\n            self.ver = ver\n\n    @classmethod\n    def match_header(cls, header):\n        \"\"\"\n        This is an abstract type that implements the shared functionality of\n        the ASCII and Binary Table HDU types, which should be used instead of\n        this.\n        \"\"\"\n\n        raise NotImplementedError\n\n    @lazyproperty\n    def columns(self):\n        \"\"\"\n        The :class:`ColDefs` objects describing the columns in this table.\n        \"\"\"\n\n        if self._has_data and hasattr(self.data, '_coldefs'):\n            return self.data._coldefs\n        return self._columns_type(self)\n\n    @lazyproperty\n    def data(self):\n        data = self._get_tbdata()\n        data._coldefs = self.columns\n        data._character_as_bytes = self._character_as_bytes\n        # Columns should now just return a reference to the data._coldefs\n        del self.columns\n        return data\n\n    @data.setter\n    def data(self, data):\n        if 'data' in self.__dict__:\n            if self.__dict__['data'] is data:\n                return\n            else:\n                self._data_replaced = True\n        else:\n            self._data_replaced = True\n\n        self._modified = True\n\n        if data is None and self.columns:\n            # Create a new table with the same columns, but empty rows\n            formats = ','.join(self.columns._recformats)\n            data = np.rec.array(None, formats=formats,\n                                names=self.columns.names,\n                                shape=0)\n\n        if isinstance(data, np.ndarray) and data.dtype.fields is not None:\n            # Go ahead and always make a view, even if the data is already the\n            # correct class (self._data_type) so we can update things like the\n            # column defs, if necessary\n            data = data.view(self._data_type)\n\n            if not isinstance(data.columns, self._columns_type):\n                # This would be the place, if the input data was for an ASCII\n                # table and this is binary table, or vice versa, to convert the\n                # data to the appropriate format for the table type\n                new_columns = self._columns_type(data.columns)\n                data = FITS_rec.from_columns(new_columns)\n\n            self.__dict__['data'] = data\n\n            self.columns = self.data.columns\n            self.update()\n\n            with suppress(TypeError, AttributeError):\n                # Make the ndarrays in the Column objects of the ColDefs\n                # object of the HDU reference the same ndarray as the HDU's\n                # FITS_rec object.\n                for idx, col in enumerate(self.columns):\n                    col.array = self.data.field(idx)\n\n                # Delete the _arrays attribute so that it is recreated to\n                # point to the new data placed in the column objects above\n                del self.columns._arrays\n        elif data is None:\n            pass\n        else:\n            raise TypeError('Table data has incorrect type.')\n\n        # returning the data signals to lazyproperty that we've already handled\n        # setting self.__dict__['data']\n        return data\n\n    @property\n    def _nrows(self):\n        if not self._data_loaded:\n            return self._header.get('NAXIS2', 0)\n        else:\n            return len(self.data)\n\n    @lazyproperty\n    def _theap(self):\n        size = self._header['NAXIS1'] * self._header['NAXIS2']\n        return self._header.get('THEAP', size)\n\n    # TODO: Need to either rename this to update_header, for symmetry with the\n    # Image HDUs, or just at some point deprecate it and remove it altogether,\n    # since header updates should occur automatically when necessary...\n    def update(self):\n        \"\"\"\n        Update header keywords to reflect recent changes of columns.\n        \"\"\"\n\n        self._header.set('NAXIS1', self.data._raw_itemsize, after='NAXIS')\n        self._header.set('NAXIS2', self.data.shape[0], after='NAXIS1')\n        self._header.set('TFIELDS', len(self.columns), after='GCOUNT')\n\n        self._clear_table_keywords()\n        self._populate_table_keywords()\n\n    def copy(self):\n        \"\"\"\n        Make a copy of the table HDU, both header and data are copied.\n        \"\"\"\n\n        # touch the data, so it's defined (in the case of reading from a\n        # FITS file)\n        return self.__class__(data=self.data.copy(),\n                              header=self._header.copy())\n\n    def _prewriteto(self, checksum=False, inplace=False):\n        if self._has_data:\n            self.data._scale_back(\n                update_heap_pointers=not self._manages_own_heap)\n            # check TFIELDS and NAXIS2\n            self._header['TFIELDS'] = len(self.data._coldefs)\n            self._header['NAXIS2'] = self.data.shape[0]\n\n            # calculate PCOUNT, for variable length tables\n            tbsize = self._header['NAXIS1'] * self._header['NAXIS2']\n            heapstart = self._header.get('THEAP', tbsize)\n            self.data._gap = heapstart - tbsize\n            pcount = self.data._heapsize + self.data._gap\n            if pcount > 0:\n                self._header['PCOUNT'] = pcount\n\n            # update the other T****n keywords\n            self._populate_table_keywords()\n\n            # update TFORM for variable length columns\n            for idx in range(self.data._nfields):\n                format = self.data._coldefs._recformats[idx]\n                if isinstance(format, _FormatP):\n                    _max = self.data.field(idx).max\n                    # May be either _FormatP or _FormatQ\n                    format_cls = format.__class__\n                    format = format_cls(format.dtype, repeat=format.repeat,\n                                        max=_max)\n                    self._header['TFORM' + str(idx + 1)] = format.tform\n        return super()._prewriteto(checksum, inplace)\n\n    def _verify(self, option='warn'):\n        \"\"\"\n        _TableBaseHDU verify method.\n        \"\"\"\n\n        errs = super()._verify(option=option)\n        self.req_cards('NAXIS', None, lambda v: (v == 2), 2, option, errs)\n        self.req_cards('BITPIX', None, lambda v: (v == 8), 8, option, errs)\n        self.req_cards('TFIELDS', 7,\n                       lambda v: (_is_int(v) and v >= 0 and v <= 999), 0,\n                       option, errs)\n        tfields = self._header['TFIELDS']\n        for idx in range(tfields):\n            self.req_cards('TFORM' + str(idx + 1), None, None, None, option,\n                           errs)\n        return errs\n\n    def _summary(self):\n        \"\"\"\n        Summarize the HDU: name, dimensions, and formats.\n        \"\"\"\n\n        class_name = self.__class__.__name__\n\n        # if data is touched, use data info.\n        if self._data_loaded:\n            if self.data is None:\n                shape, format = (), ''\n                nrows = 0\n            else:\n                nrows = len(self.data)\n\n            ncols = len(self.columns)\n            format = self.columns.formats\n\n        # if data is not touched yet, use header info.\n        else:\n            shape = ()\n            nrows = self._header['NAXIS2']\n            ncols = self._header['TFIELDS']\n            format = ', '.join([self._header['TFORM' + str(j + 1)]\n                                for j in range(ncols)])\n            format = '[{}]'.format(format)\n        dims = \"{}R x {}C\".format(nrows, ncols)\n        ncards = len(self._header)\n\n        return (self.name, self.ver, class_name, ncards, dims, format)\n\n    def _update_column_removed(self, columns, idx):\n        super()._update_column_removed(columns, idx)\n\n        # Fix the header to reflect the column removal\n        self._clear_table_keywords(index=idx)\n\n    def _update_column_attribute_changed(self, column, col_idx, attr,\n                                         old_value, new_value):\n        \"\"\"\n        Update the header when one of the column objects is updated.\n        \"\"\"\n\n        # base_keyword is the keyword without the index such as TDIM\n        # while keyword is like TDIM1\n        base_keyword = ATTRIBUTE_TO_KEYWORD[attr]\n        keyword = base_keyword + str(col_idx + 1)\n\n        if keyword in self._header:\n            if new_value is None:\n                # If the new value is None, i.e. None was assigned to the\n                # column attribute, then treat this as equivalent to deleting\n                # that attribute\n                del self._header[keyword]\n            else:\n                self._header[keyword] = new_value\n        else:\n            keyword_idx = KEYWORD_NAMES.index(base_keyword)\n            # Determine the appropriate keyword to insert this one before/after\n            # if it did not already exist in the header\n            for before_keyword in reversed(KEYWORD_NAMES[:keyword_idx]):\n                before_keyword += str(col_idx + 1)\n                if before_keyword in self._header:\n                    self._header.insert(before_keyword, (keyword, new_value),\n                                        after=True)\n                    break\n            else:\n                for after_keyword in KEYWORD_NAMES[keyword_idx + 1:]:\n                    after_keyword += str(col_idx + 1)\n                    if after_keyword in self._header:\n                        self._header.insert(after_keyword,\n                                            (keyword, new_value))\n                        break\n                else:\n                    # Just append\n                    self._header[keyword] = new_value\n\n    def _clear_table_keywords(self, index=None):\n        \"\"\"\n        Wipe out any existing table definition keywords from the header.\n\n        If specified, only clear keywords for the given table index (shifting\n        up keywords for any other columns).  The index is zero-based.\n        Otherwise keywords for all columns.\n        \"\"\"\n\n        # First collect all the table structure related keyword in the header\n        # into a single list so we can then sort them by index, which will be\n        # useful later for updating the header in a sensible order (since the\n        # header *might* not already be written in a reasonable order)\n        table_keywords = []\n\n        for idx, keyword in enumerate(self._header.keys()):\n            match = TDEF_RE.match(keyword)\n            try:\n                base_keyword = match.group('label')\n            except Exception:\n                continue                # skip if there is no match\n\n            if base_keyword in KEYWORD_TO_ATTRIBUTE:\n                num = int(match.group('num')) - 1  # convert to zero-base\n                table_keywords.append((idx, match.group(0), base_keyword,\n                                       num))\n\n        # First delete\n        rev_sorted_idx_0 = sorted(table_keywords, key=operator.itemgetter(0),\n                                  reverse=True)\n        for idx, keyword, _, num in rev_sorted_idx_0:\n            if index is None or index == num:\n                del self._header[idx]\n\n        # Now shift up remaining column keywords if only one column was cleared\n        if index is not None:\n            sorted_idx_3 = sorted(table_keywords, key=operator.itemgetter(3))\n            for _, keyword, base_keyword, num in sorted_idx_3:\n                if num <= index:\n                    continue\n\n                old_card = self._header.cards[keyword]\n                new_card = (base_keyword + str(num), old_card.value,\n                            old_card.comment)\n                self._header.insert(keyword, new_card)\n                del self._header[keyword]\n\n            # Also decrement TFIELDS\n            if 'TFIELDS' in self._header:\n                self._header['TFIELDS'] -= 1\n\n    def _populate_table_keywords(self):\n        \"\"\"Populate the new table definition keywords from the header.\"\"\"\n\n        for idx, column in enumerate(self.columns):\n            for keyword, attr in KEYWORD_TO_ATTRIBUTE.items():\n                val = getattr(column, attr)\n                if val is not None:\n                    keyword = keyword + str(idx + 1)\n                    self._header[keyword] = val"
        },
        {
          "file": "astropy/io/fits/hdu/table.py",
          "type": "function",
          "name": "__init__",
          "class_name": "BinTableHDU",
          "code": "def __init__(self, data=None, header=None, name=None, uint=False, ver=None,\n                 character_as_bytes=False):\n        from ....table import Table\n        if isinstance(data, Table):\n            from ..convenience import table_to_hdu\n            hdu = table_to_hdu(data)\n            if header is not None:\n                hdu.header.update(header)\n            data = hdu.data\n            header = hdu.header\n\n        super().__init__(data, header, name=name, uint=uint, ver=ver,\n                         character_as_bytes=character_as_bytes)"
        },
        {
          "file": "astropy/io/fits/hdu/table.py",
          "type": "function",
          "name": "__init__",
          "class_name": "BinTableHDU",
          "code": "def __init__(self, data=None, header=None, name=None, uint=False, ver=None,\n                 character_as_bytes=False):\n        from ....table import Table\n        if isinstance(data, Table):\n            from ..convenience import table_to_hdu\n            hdu = table_to_hdu(data)\n            if header is not None:\n                hdu.header.update(header)\n            data = hdu.data\n            header = hdu.header\n\n        super().__init__(data, header, name=name, uint=uint, ver=ver,\n                         character_as_bytes=character_as_bytes)"
        },
        {
          "file": "astropy/io/fits/hdu/table.py",
          "type": "function",
          "name": "_clear_table_keywords",
          "class_name": "_TableBaseHDU",
          "code": "def _clear_table_keywords(self, index=None):\n        \"\"\"\n        Wipe out any existing table definition keywords from the header.\n\n        If specified, only clear keywords for the given table index (shifting\n        up keywords for any other columns).  The index is zero-based.\n        Otherwise keywords for all columns.\n        \"\"\"\n\n        # First collect all the table structure related keyword in the header\n        # into a single list so we can then sort them by index, which will be\n        # useful later for updating the header in a sensible order (since the\n        # header *might* not already be written in a reasonable order)\n        table_keywords = []\n\n        for idx, keyword in enumerate(self._header.keys()):\n            match = TDEF_RE.match(keyword)\n            try:\n                base_keyword = match.group('label')\n            except Exception:\n                continue                # skip if there is no match\n\n            if base_keyword in KEYWORD_TO_ATTRIBUTE:\n                num = int(match.group('num')) - 1  # convert to zero-base\n                table_keywords.append((idx, match.group(0), base_keyword,\n                                       num))\n\n        # First delete\n        rev_sorted_idx_0 = sorted(table_keywords, key=operator.itemgetter(0),\n                                  reverse=True)\n        for idx, keyword, _, num in rev_sorted_idx_0:\n            if index is None or index == num:\n                del self._header[idx]\n\n        # Now shift up remaining column keywords if only one column was cleared\n        if index is not None:\n            sorted_idx_3 = sorted(table_keywords, key=operator.itemgetter(3))\n            for _, keyword, base_keyword, num in sorted_idx_3:\n                if num <= index:\n                    continue\n\n                old_card = self._header.cards[keyword]\n                new_card = (base_keyword + str(num), old_card.value,\n                            old_card.comment)\n                self._header.insert(keyword, new_card)\n                del self._header[keyword]\n\n            # Also decrement TFIELDS\n            if 'TFIELDS' in self._header:\n                self._header['TFIELDS'] -= 1"
        }
      ]
    },
    {
      "pr_number": 4930,
      "pr_title": "Make Table inherit primary_key attribute from its parent Table.",
      "pr_body": "This is just a rebased version of #4711.\n\nCloses #4711.\nFixes #4672.\n",
      "issue_id": 4711,
      "issue_title": "Make Table inherit primary_key attribute from its parent Table. Fix for [#4672]",
      "issue_body": "This is a fix for issue #4672.\n\nIn the issue, running an example from the astropy docs for [Table indexing](http://astropy.readthedocs.org/en/latest/table/indexing.html) would run into an error. The error was a missing `primary_key` for the table in question. The error arises when a new Table is made from an existing one, either by slicing or reference, but the `primary_key` attribute is not transferred to the new Table. This PR fixes that.\n\nI added one test for this case, but I couldn't figure out a way to test for `.loc` as @taldcroft suggested in the issue. I'm actually not that familiar with Tables or with writing test cases (this is my first attempt to do so), so any feedback would be appreciated.   \n",
      "issue_closed_at": "2016-05-16T13:01:26Z",
      "base_commit": "3c47c1067036a6cf3dc13ebc0fc268948d4ccc29",
      "changes": [
        {
          "file": "astropy/table/table.py",
          "type": "function",
          "name": "_init_from_table",
          "class_name": "Table",
          "code": "def _init_from_table(self, data, names, dtype, n_cols, copy):\n        \"\"\"Initialize table from an existing Table object \"\"\"\n\n        table = data  # data is really a Table, rename for clarity\n        self.meta.clear()\n        self.meta.update(deepcopy(table.meta))\n        cols = list(table.columns.values())\n\n        self._init_from_list(cols, names, dtype, n_cols, copy)"
        },
        {
          "file": "astropy/table/table.py",
          "type": "function",
          "name": "_new_from_slice",
          "class_name": "Table",
          "code": "def _new_from_slice(self, slice_):\n        \"\"\"Create a new table as a referenced slice from self.\"\"\"\n\n        table = self.__class__(masked=self.masked)\n        table.meta.clear()\n        table.meta.update(deepcopy(self.meta))\n        cols = self.columns.values()\n\n        newcols = []\n        for col in cols:\n            col.info._copy_indices = self._copy_indices\n            newcol = col[slice_]\n            if col.info.indices:\n                newcol = col.info.slice_indices(newcol, slice_, len(col))\n            newcols.append(newcol)\n            col.info._copy_indices = True\n\n        self._make_table_from_cols(table, newcols)\n        return table"
        }
      ]
    }
  ]
}