{
  "instance_id": "astropy__astropy-6938",
  "repo": "astropy/astropy",
  "created_at": "2017-12-07T00:01:14Z",
  "problem_statement": "Possible bug in io.fits related to D exponents\nI came across the following code in ``fitsrec.py``:\r\n\r\n```python\r\n        # Replace exponent separator in floating point numbers\r\n        if 'D' in format:\r\n            output_field.replace(encode_ascii('E'), encode_ascii('D'))\r\n```\r\n\r\nI think this may be incorrect because as far as I can tell ``replace`` is not an in-place operation for ``chararray`` (it returns a copy). Commenting out this code doesn't cause any tests to fail so I think this code isn't being tested anyway.\n",
  "patch": "diff --git a/astropy/io/fits/fitsrec.py b/astropy/io/fits/fitsrec.py\n--- a/astropy/io/fits/fitsrec.py\n+++ b/astropy/io/fits/fitsrec.py\n@@ -1261,7 +1261,7 @@ def _scale_back_ascii(self, col_idx, input_field, output_field):\n \n         # Replace exponent separator in floating point numbers\n         if 'D' in format:\n-            output_field.replace(encode_ascii('E'), encode_ascii('D'))\n+            output_field[:] = output_field.replace(b'E', b'D')\n \n \n def _get_recarray_field(array, key):\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": 5345,
      "pr_title": "Ensure column is converted correctly if unit is set to logarithmic unit",
      "pr_body": "Fixes #5342.\n\nThe main fix is to adapt `QTable._convert_col_for_table`. But it also turned out `!=` never worked for function units, so that is fixed as well.\n\nAssigned to @taldcrof as the main change is in `table`.\n",
      "issue_id": 5342,
      "issue_title": "Handling of logarithmic units in tables inconsistent and breaks matching",
      "issue_body": "I have tried to use logarithmic units in my table columns but this seems to work less than optimal - at least the way I have created them. I have an example piece of code below, but the two problems are:\n1. A column created to have  a logarithmic unit [e.g.: t['log_R'] = u.dex(u.pc)], is created as a Quantity column and has no 'physical' attribute:\n   e.g. t['log_R'].quantity.value works, but t['log_R'].quantity.physical gives\n\n```\n/Users/jarle/anaconda/lib/python2.7/site-packages/astropy/units/quantity.pyc in __getattr__(self, attr)\n    732                 \"'{0}' object has no '{1}' member\".format(\n    733                     self.__class__.__name__,\n--> 734                     attr))\n    735 \n    736         def get_virtual_unit_attribute():\n\nAttributeError: 'Quantity' object has no 'physical' member\n```\n1. When trying to join tables with logarithmic quantities, the join fails with:\n\n```\nTraceback (most recent call last):\n  File \"/Users/jarle/Desktop/example_problem.py\", line 27, in <module>\n    print astropy.table.join(tbl1, tbl2, keys=['col1'])\n  File \"/Users/jarle/anaconda/lib/python2.7/site-packages/astropy/table/operations.py\", line 188, in join\n    _merge_col_meta(out, [left, right], col_name_map, metadata_conflicts=metadata_conflicts)\n  File \"/Users/jarle/anaconda/lib/python2.7/site-packages/astropy/table/operations.py\", line 70, in _merge_col_meta\n    elif left_attr != right_attr:\n  File \"/Users/jarle/anaconda/lib/python2.7/site-packages/astropy/units/function/core.py\", line 291, in __ne__\n    self.funtional_unit != getattr(other, 'function_unit', other))\nAttributeError: 'DexUnit' object has no attribute 'funtional_unit'\n```\n\nAn example code that reproduces this is given below. \n\nThis is certainly not what I expected, but it might be that I have misunderstood the way I should indicate units on the columns! \n\n```\n# Show how joins break when logarithmic units are included  - the first join succeeds \n# and prints out the combined table, while the second fails\nimport astropy.units as u\nfrom astropy.io import ascii\nimport astropy.table\n\n# Create two simple example tables\nt1 = \"\"\"\na 5.2\nb 7.2\n\"\"\"\n\nt2 = \"\"\"\na 6.3\nc 7.4\nb 1.2\n\"\"\"\n\ntbl1 = ascii.read(t1, format='fixed_width_no_header', delimiter=' ')\ntbl2 = ascii.read(t2, format='fixed_width_no_header', delimiter=' ')\n\n# Simple units:\ntbl1['col2'].unit = u.kilometer\n\nprint astropy.table.join(tbl1, tbl2, keys=['col1'])\n\n# Logarithmic units\ntbl1['col2'].unit = u.dex()\nprint astropy.table.join(tbl1, tbl2, keys=['col1'])\n\n```\n",
      "issue_closed_at": "2016-09-24T15:09:40Z",
      "base_commit": "f3a5da293bb4e806855986b449dcdae5f2ad5891",
      "changes": [
        {
          "file": "astropy/table/table.py",
          "type": "function",
          "name": "_add_as_mixin_column",
          "class_name": "QTable",
          "code": "def _add_as_mixin_column(self, col):\n        \"\"\"\n        Determine if ``col`` should be added to the table directly as\n        a mixin column.\n        \"\"\"\n        return has_info_class(col, MixinInfo)"
        },
        {
          "file": "astropy/units/function/core.py",
          "type": "function",
          "name": "__eq__",
          "class_name": "FunctionQuantity",
          "code": "def __eq__(self, other):\n        try:\n            return self._comparison(other, self.value.__eq__)\n        except UnitsError:\n            return False"
        }
      ]
    },
    {
      "pr_number": 5589,
      "pr_title": "Fix up lazy loading issues",
      "pr_body": "Fix or address these issues which resulted from lazy loading. In each case the goal is to maintain the current (1.x) behavior of io.fits.\r\n\r\n- [x] Handling of invalid key in the `in` operator (closes #5583)\r\n- [x] Slices not working (closes #5585)\r\n- [x] Variable name incorrect (closes #5594)\r\n- ~~[ ] Address behavior of `close` #5582~~ Won't change unless someone requests it.\r\n\r\n@eteq -- I believe I've milestoned this correctly, but please modify if not.\r\n\r\nedit: changed text of second todo.",
      "issue_id": 5594,
      "issue_title": "Regression in io.fits due to undefined variable",
      "issue_body": "@mwcraig @embray  - I'm running into the following failure in glue due to the recent changes in io.fits:\r\n\r\n```python\r\n    def _read_next_hdu(self):\r\n        \"\"\"\r\n            Lazily load a single HDU from the fileobj or data string the `HDUList`\r\n            was opened from, unless no further HDUs are found.\r\n    \r\n            Returns True if a new HDU was loaded, or False otherwise.\r\n            \"\"\"\r\n    \r\n        if self._read_all:\r\n            return False\r\n    \r\n        saved_compression_enabled = compressed.COMPRESSION_ENABLED\r\n        fileobj, data, kwargs = self._file, self._data, self._open_kwargs\r\n    \r\n        try:\r\n            self._in_read_next_hdu = True\r\n    \r\n            if ('disable_image_compression' in kwargs and\r\n                kwargs['disable_image_compression']):\r\n                compressed.COMPRESSION_ENABLED = False\r\n    \r\n            # read all HDUs\r\n            try:\r\n                if fileobj is not None:\r\n                    try:\r\n                        # Make sure we're back to the end of the last read\r\n                        # HDU\r\n                        if len(self) > 0:\r\n                            last = self[len(self) - 1]\r\n                            if last._data_offset is not None:\r\n                                offset = last._data_offset + last._data_size\r\n                                fileobj.seek(offset, os.SEEK_SET)\r\n    \r\n                        hdu = _BaseHDU.readfrom(fileobj, **kwargs)\r\n                    except EOFError:\r\n                        self._read_all = True\r\n                        return False\r\n                    except ValueError:\r\n                        # A ValueError can occur when trying to perform I/O\r\n                        # on a closed file\r\n                        if fileobj.closed:\r\n                            self._read_all = True\r\n                            return False\r\n                        else:\r\n                            raise\r\n                    except IOError:\r\n                        if fileobj.writeonly:\r\n                            self._read_all = True\r\n                            return False\r\n                        else:\r\n                            raise\r\n                else:\r\n                    if not data:\r\n                        self._read_all = True\r\n                        return False\r\n                    hdu = _BaseHDU.fromstring(data, **kwargs)\r\n                    self._data = data[hdu._data_offset + hdu._data_size:]\r\n    \r\n                super(HDUList, self).append(hdu)\r\n                if len(self) == 1:\r\n                    # Check for an extension HDU and update the EXTEND\r\n                    # keyword of the primary HDU accordingly\r\n                    self.update_extend()\r\n    \r\n                hdu._new = False\r\n                if 'checksum' in kwargs:\r\n                    hdu._output_checksum = kwargs['checksum']\r\n            # check in the case there is extra space after the last HDU or\r\n            # corrupted HDU\r\n            except (VerifyError, ValueError) as exc:\r\n                warnings.warn(\r\n                    'Error validating header for HDU #{} (note: Astropy '\r\n                    'uses zero-based indexing).\\n{}\\n'\r\n                    'There may be extra bytes after the last HDU or the '\r\n                    'file is corrupted.'.format(\r\n>                       len(hdulist), indent(str(exc))), VerifyWarning)\r\nE                       NameError: global name 'hdulist' is not defined\r\n\r\n../../../miniconda/envs/test/lib/python2.7/site-packages/astropy/io/fits/hdu/hdulist.py:1150: NameError\r\n============================ pytest-warning summary ============================\r\n```",
      "issue_closed_at": "2016-12-13T01:48:53Z",
      "base_commit": "a72fdfb6b07356805182cb3a2fc20487404b073f",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/hdulist.py",
          "type": "function",
          "name": "__getitem__",
          "class_name": "HDUList",
          "code": "def __getitem__(self, key):\n        \"\"\"\n        Get an HDU from the `HDUList`, indexed by number or name.\n        \"\"\"\n\n        if isinstance(key, slice):\n            hdus = super(HDUList, self).__getitem__(key)\n            return HDUList(hdus)\n\n        # Originally this used recursion, but hypothetically an HDU with\n        # a very large number of HDUs could blow the stack, so use a loop\n        # instead\n        return self._try_while_unread_hdus(super(HDUList, self).__getitem__,\n                                           self._positive_index_of(key))"
        },
        {
          "file": "astropy/io/fits/hdu/hdulist.py",
          "type": "function",
          "name": "fromstring",
          "class_name": "HDUList",
          "code": "def fromstring(cls, data, **kwargs):\n        \"\"\"\n        Creates an `HDUList` instance from a string or other in-memory data\n        buffer containing an entire FITS file.  Similar to\n        :meth:`HDUList.fromfile`, but does not accept the mode or memmap\n        arguments, as they are only relevant to reading from a file on disk.\n\n        This is useful for interfacing with other libraries such as CFITSIO,\n        and may also be useful for streaming applications.\n\n        Parameters\n        ----------\n        data : str, buffer, memoryview, etc.\n            A string or other memory buffer containing an entire FITS file.  It\n            should be noted that if that memory is read-only (such as a Python\n            string) the returned :class:`HDUList`'s data portions will also be\n            read-only.\n\n        kwargs : dict\n            Optional keyword arguments.  See\n            :func:`astropy.io.fits.open` for details.\n\n        Returns\n        -------\n        hdul : HDUList\n            An :class:`HDUList` object representing the in-memory FITS file.\n        \"\"\"\n\n        try:\n            # Test that the given object supports the buffer interface by\n            # ensuring an ndarray can be created from it\n            np.ndarray((), dtype='ubyte', buffer=data)\n        except TypeError:\n            raise TypeError(\n                'The provided object %r does not contain an underlying '\n                'memory buffer.  fromstring() requires an object that '\n                'supports the buffer interface such as bytes, str '\n                '(in Python 2.x but not in 3.x), buffer, memoryview, '\n                'ndarray, etc.  This restriction is to ensure that '\n                'efficient access to the array/table data is possible.'\n                % data)\n\n        return cls._readfrom(data=data, **kwargs)"
        },
        {
          "file": "astropy/io/fits/hdu/hdulist.py",
          "type": "function",
          "name": "index_of",
          "class_name": "HDUList",
          "code": "def index_of(self, key):\n        \"\"\"\n        Get the index of an HDU from the `HDUList`.\n\n        Parameters\n        ----------\n        key : int, str or tuple of (string, int)\n           The key identifying the HDU.  If ``key`` is a tuple, it is of the\n           form ``(key, ver)`` where ``ver`` is an ``EXTVER`` value that must\n           match the HDU being searched for.\n\n           If the key is ambiguous (e.g. there are multiple 'SCI' extensions)\n           the first match is returned.  For a more precise match use the\n           ``(name, ver)`` pair.\n\n           If even the ``(name, ver)`` pair is ambiguous (it shouldn't be\n           but it's not impossible) the numeric index must be used to index\n           the duplicate HDU.\n\n        Returns\n        -------\n        index : int\n           The index of the HDU in the `HDUList`.\n        \"\"\"\n\n        if _is_int(key):\n            return key\n        elif isinstance(key, tuple):\n            _key, _ver = key\n        else:\n            _key = key\n            _ver = None\n\n        if not isinstance(_key, string_types):\n            raise TypeError(\n                '%s indices must be integers, extension names as strings, '\n                'or (extname, version) tuples; got %r' %\n                (self.__class__.__name__, _key))\n\n        _key = (_key.strip()).upper()\n\n        found = None\n        for idx, hdu in enumerate(self):\n            name = hdu.name\n            if isinstance(name, string_types):\n                name = name.strip().upper()\n            # 'PRIMARY' should always work as a reference to the first HDU\n            if ((name == _key or (_key == 'PRIMARY' and idx == 0)) and\n                (_ver is None or _ver == hdu.ver)):\n                found = idx\n                break\n\n        if (found is None):\n            raise KeyError('Extension {} not found.'.format(repr(key)))\n        else:\n            return found"
        },
        {
          "file": "astropy/io/fits/hdu/hdulist.py",
          "type": "function",
          "name": "_positive_index_of",
          "class_name": "HDUList",
          "code": "def _positive_index_of(self, key):\n        \"\"\"\n        Same as index_of, but ensures always returning a positive index\n        or zero.\n\n        (Really this should be called non_negative_index_of but it felt\n        too long.)\n\n        This means that if the key is a negative integer, we have to\n        convert it to the corresponding positive index.  This means\n        knowing the length of the HDUList, which in turn means loading\n        all HDUs.  Therefore using negative indices on HDULists is inherently\n        inefficient.\n        \"\"\"\n\n        index = self.index_of(key)\n\n        if index >= 0:\n            return index\n\n        if abs(index) > len(self):\n            raise IndexError(\n                'Extension %s is out of bound or not found.' % index)\n\n        return len(self) + index"
        },
        {
          "file": "astropy/io/fits/hdu/hdulist.py",
          "type": "function",
          "name": "_read_next_hdu",
          "class_name": "HDUList",
          "code": "def _read_next_hdu(self):\n        \"\"\"\n        Lazily load a single HDU from the fileobj or data string the `HDUList`\n        was opened from, unless no further HDUs are found.\n\n        Returns True if a new HDU was loaded, or False otherwise.\n        \"\"\"\n\n        if self._read_all:\n            return False\n\n        saved_compression_enabled = compressed.COMPRESSION_ENABLED\n        fileobj, data, kwargs = self._file, self._data, self._open_kwargs\n\n        try:\n            self._in_read_next_hdu = True\n\n            if ('disable_image_compression' in kwargs and\n                kwargs['disable_image_compression']):\n                compressed.COMPRESSION_ENABLED = False\n\n            # read all HDUs\n            try:\n                if fileobj is not None:\n                    try:\n                        # Make sure we're back to the end of the last read\n                        # HDU\n                        if len(self) > 0:\n                            last = self[len(self) - 1]\n                            if last._data_offset is not None:\n                                offset = last._data_offset + last._data_size\n                                fileobj.seek(offset, os.SEEK_SET)\n\n                        hdu = _BaseHDU.readfrom(fileobj, **kwargs)\n                    except EOFError:\n                        self._read_all = True\n                        return False\n                    except ValueError:\n                        # A ValueError can occur when trying to perform I/O\n                        # on a closed file\n                        if fileobj.closed:\n                            self._read_all = True\n                            return False\n                        else:\n                            raise\n                    except IOError:\n                        if fileobj.writeonly:\n                            self._read_all = True\n                            return False\n                        else:\n                            raise\n                else:\n                    if not data:\n                        self._read_all = True\n                        return False\n                    hdu = _BaseHDU.fromstring(data, **kwargs)\n                    self._data = data[hdu._data_offset + hdu._data_size:]\n\n                super(HDUList, self).append(hdu)\n                if len(self) == 1:\n                    # Check for an extension HDU and update the EXTEND\n                    # keyword of the primary HDU accordingly\n                    self.update_extend()\n\n                hdu._new = False\n                if 'checksum' in kwargs:\n                    hdu._output_checksum = kwargs['checksum']\n            # check in the case there is extra space after the last HDU or\n            # corrupted HDU\n            except (VerifyError, ValueError) as exc:\n                warnings.warn(\n                    'Error validating header for HDU #{} (note: Astropy '\n                    'uses zero-based indexing).\\n{}\\n'\n                    'There may be extra bytes after the last HDU or the '\n                    'file is corrupted.'.format(\n                        len(hdulist), indent(str(exc))), VerifyWarning)\n                del exc\n                self._read_all = True\n                return False\n        finally:\n            compressed.COMPRESSION_ENABLED = saved_compression_enabled\n            self._in_read_next_hdu = False\n\n        return True"
        }
      ]
    },
    {
      "pr_number": 6286,
      "pr_title": "Coordinates/velocity bugfix",
      "pr_body": "I think this is a change that I had meant to include in #6219 - this relaxes the `_check_base()` method on any differential to only check that the class is compatible with the representation base (i.e. that the differential class is in `rep._compatible_differentials`). This is needed so that, e.g., transformations like this can succeed (`UnitSphericalRepresentation` + `SphericalCosLatDifferential`):\r\n```python\r\nc1 = coord.FK5(ra=150*u.deg, dec=-17*u.deg, radial_velocity=83*u.km/u.s,\r\n               pm_ra_cosdec=-41*u.mas/u.yr, pm_dec=16*u.mas/u.yr)\r\nc1.transform_to(coord.Galactic)\r\n```\r\n\r\nFixes #6283 \r\n\r\ncc @eteq @mhvk ",
      "issue_id": 6283,
      "issue_title": "Full-space velocity but unit-spherical position doesn't transform properly",
      "issue_body": "After #6219 was merged, I found what looks like a remaining bug - this fails:\r\n\r\n```python\r\nc1 = coord.FK5(ra=150*u.deg, dec=-17*u.deg, radial_velocity=83*u.km/u.s,\r\n               pm_ra_cosdec=-41*u.mas/u.yr, pm_dec=16*u.mas/u.yr)\r\nc1.transform_to(coord.Galactic)\r\n```\r\nwith\r\n```TypeError: need a base of the correct representation type, <class 'astropy.coordinates.representation.SphericalRepresentation'>, not <class 'astropy.coordinates.representation.UnitSphericalRepresentation'>```\r\n\r\ncc @eteq ",
      "issue_closed_at": "2017-06-27T14:54:42Z",
      "base_commit": "abefd8ffffafcd1f99c72751a66b249f18f1a928",
      "changes": [
        {
          "file": "astropy/coordinates/representation.py",
          "type": "function",
          "name": "_validate_differentials",
          "class_name": "BaseRepresentation",
          "code": "def _validate_differentials(self, differentials):\n        \"\"\"\n        Validate that the provided differentials are appropriate for this\n        representation and recast/reshape as necessary and then return.\n\n        Note that this does *not* set the differentials on\n        ``self._differentials``, but rather leaves that for the caller.\n        \"\"\"\n\n        # Now handle the actual validation of any specified differential classes\n        if differentials is None:\n            differentials = dict()\n\n        elif isinstance(differentials, BaseDifferential):\n            # We can't handle auto-determining the key for this combo\n            if (isinstance(differentials, RadialDifferential) and\n                    isinstance(self, UnitSphericalRepresentation)):\n                raise ValueError(\"To attach a RadialDifferential to a \"\n                                 \"UnitSphericalRepresentation, you must supply \"\n                                 \"a dictionary with an appropriate key.\")\n\n            key = differentials._get_deriv_key(self)\n            differentials = {key: differentials}\n\n        for key in differentials:\n            try:\n                diff = differentials[key]\n            except TypeError:\n                raise TypeError(\"'differentials' argument must be a \"\n                                \"dictionary-like object\")\n\n            if diff.__class__ not in self._compatible_differentials:\n                raise TypeError(\"Differential '{0}' is not compatible with \"\n                                \"this representation '{1}'\".format(repr(diff),\n                                                                   repr(self)))\n\n            if (isinstance(diff, RadialDifferential) and\n                    isinstance(self, UnitSphericalRepresentation)):\n                # We trust the passing of a key for a RadialDifferential\n                # attached to a UnitSphericalRepresentation because it will not\n                # have a paired component name (UnitSphericalRepresentation has\n                # no .distance) to automatically determine the expected key\n                pass\n\n            else:\n                expected_key = diff._get_deriv_key(self)\n                if key != expected_key:\n                    raise ValueError(\"For differential object '{0}', expected \"\n                                     \"unit key = '{1}' but received key = '{2}'\"\n                                     .format(repr(diff), expected_key, key))\n\n            # For now, we are very rigid: differentials must have the same shape\n            # as the representation. This makes it easier to handle __getitem__\n            # and any other shape-changing operations on representations that\n            # have associated differentials\n            if diff.shape != self.shape:\n                # TODO: message of IncompatibleShapeError is not customizable,\n                #       so use a valueerror instead?\n                raise ValueError(\"Shape of differentials must be the same \"\n                                 \"as the shape of the representation ({0} vs \"\n                                 \"{1})\".format(diff.shape, self.shape))\n\n        return differentials"
        },
        {
          "file": "astropy/coordinates/representation.py",
          "type": "class",
          "name": "BaseDifferential",
          "code": "class BaseDifferential(BaseRepresentationOrDifferential):\n    r\"\"\"A base class representing differentials of representations.\n\n    These represent differences or derivatives along each component.\n    E.g., for physics spherical coordinates, these would be\n    :math:`\\delta r, \\delta \\theta, \\delta \\phi`.\n\n    Parameters\n    ----------\n    d_comp1, d_comp2, d_comp3 : `~astropy.units.Quantity` or subclass\n        The components of the 3D differentials.  The names are the keys and the\n        subclasses the values of the ``attr_classes`` attribute.\n    copy : bool, optional\n        If `True` (default), arrays will be copied rather than referenced.\n\n    Notes\n    -----\n    All differential representation classes should subclass this base class,\n    and define an ``base_representation`` attribute with the class of the\n    regular `~astropy.coordinates.BaseRepresentation` for which differential\n    coordinates are provided. This will set up a default ``attr_classes``\n    instance with names equal to the base component names prefixed by ``d_``,\n    and all classes set to `~astropy.units.Quantity`, plus properties to access\n    those, and a default ``__init__`` for initialization.\n    \"\"\"\n\n    recommended_units = {}  # subclasses can override\n\n    @classmethod\n    def _check_base(cls, base):\n        if not isinstance(base, cls.base_representation):\n            raise TypeError('need a base of the correct representation type, '\n                            '{0}, not {1}.'.format(cls.base_representation,\n                                                   type(base)))\n\n    def _get_deriv_key(self, base):\n        \"\"\"Given a base (representation instance), determine the unit of the\n        derivative by removing the representation unit from the component units\n        of this differential.\n        \"\"\"\n\n        # This check is just a last resort so we don't return a strange unit key\n        # from accidentally passing in the wrong base.\n        if self.__class__ not in base._compatible_differentials:\n            raise TypeError(\"Differential class {0} is not compatible with the \"\n                            \"base (representation) class {1}\"\n                            .format(self.__class__, base.__class__))\n\n        for name in base.components:\n            comp = getattr(base, name)\n            d_comp = getattr(self, 'd_{0}'.format(name), None)\n            if d_comp:\n                d_unit = comp.unit / d_comp.unit\n                # Get the si unit without a scale by going via Quantity;\n                # `.si` causes the scale to be included in the value.\n                return str(u.Quantity(1., d_unit).si.unit)\n\n        else:\n            raise RuntimeError(\"Invalid representation-differential match! Not \"\n                               \"sure how we got into this state.\")\n\n    @classmethod\n    def _get_base_vectors(cls, base):\n        \"\"\"Get unit vectors and scale factors from base.\n\n        Parameters\n        ----------\n        base : instance of ``self.base_representation``\n            The points for which the unit vectors and scale factors should be\n            retrieved.\n\n        Returns\n        -------\n        unit_vectors : dict of `CartesianRepresentation`\n            In the directions of the coordinates of base.\n        scale_factors : dict of `~astropy.units.Quantity`\n            Scale factors for each of the coordinates\n\n        Raises\n        ------\n        TypeError : if the base is not of the correct type\n        \"\"\"\n        cls._check_base(base)\n        return base.unit_vectors(), base.scale_factors()\n\n    def to_cartesian(self, base):\n        \"\"\"Convert the differential to 3D rectangular cartesian coordinates.\n\n        Parameters\n        ----------\n        base : instance of ``self.base_representation``\n             The points for which the differentials are to be converted: each of\n             the components is multiplied by its unit vectors and scale factors.\n\n        Returns\n        -------\n        This object as a `CartesianDifferential`\n        \"\"\"\n        base_e, base_sf = self._get_base_vectors(base)\n        return functools.reduce(\n            operator.add, (getattr(self, d_c) * base_sf[c] * base_e[c]\n                           for d_c, c in zip(self.components, base.components)))\n\n    @classmethod\n    def from_cartesian(cls, other, base):\n        \"\"\"Convert the differential from 3D rectangular cartesian coordinates to\n        the desired class.\n\n        Parameters\n        ----------\n        other :\n            The object to convert into this differential.\n        base : instance of ``self.base_representation``\n             The points for which the differentials are to be converted: each of\n             the components is multiplied by its unit vectors and scale factors.\n\n        Returns\n        -------\n        A new differential object that is this class' type.\n        \"\"\"\n        base_e, base_sf = cls._get_base_vectors(base)\n        return cls(*(other.dot(e / base_sf[component])\n                     for component, e in six.iteritems(base_e)), copy=False)\n\n    def represent_as(self, other_class, base):\n        \"\"\"Convert coordinates to another representation.\n\n        If the instance is of the requested class, it is returned unmodified.\n        By default, conversion is done via cartesian coordinates.\n\n        Parameters\n        ----------\n        other_class : `~astropy.coordinates.BaseRepresentation` subclass\n            The type of representation to turn the coordinates into.\n        base : instance of ``self.base_representation``, optional\n            Base relative to which the differentials are defined.  If the other\n            class is a differential representation, the base will be converted\n            to its ``base_representation``.\n        \"\"\"\n        if other_class is self.__class__:\n            return self\n\n        # The default is to convert via cartesian coordinates.\n        self_cartesian = self.to_cartesian(base)\n        if issubclass(other_class, BaseDifferential):\n            base = base.represent_as(other_class.base_representation)\n            return other_class.from_cartesian(self_cartesian, base)\n        else:\n            return other_class.from_cartesian(self_cartesian)\n\n    @classmethod\n    def from_representation(cls, representation, base):\n        \"\"\"Create a new instance of this representation from another one.\n\n        Parameters\n        ----------\n        representation : `~astropy.coordinates.BaseRepresentation` instance\n            The presentation that should be converted to this class.\n        base : instance of ``cls.base_representation``\n            The base relative to which the differentials will be defined. If\n            the representation is a differential itself, the base will be\n            converted to its ``base_representation`` to help convert it.\n        \"\"\"\n        if isinstance(representation, BaseDifferential):\n            cartesian = representation.to_cartesian(\n                base.represent_as(representation.base_representation))\n        else:\n            cartesian = representation.to_cartesian()\n\n        return cls.from_cartesian(cartesian, base)\n\n    def _scale_operation(self, op, *args):\n        \"\"\"Scale all components.\n\n        Parameters\n        ----------\n        op : `~operator` callable\n            Operator to apply (e.g., `~operator.mul`, `~operator.neg`, etc.\n        *args\n            Any arguments required for the operator (typically, what is to\n            be multiplied with, divided by).\n        \"\"\"\n        scaled_attrs = [op(getattr(self, c), *args) for c in self.components]\n        return self.__class__(*scaled_attrs, copy=False)\n\n    def _combine_operation(self, op, other, reverse=False):\n        \"\"\"Combine two differentials, or a differential with a representation.\n\n        If ``other`` is of the same differential type as ``self``, the\n        components will simply be combined.  If ``other`` is a representation,\n        it will be used as a base for which to evaluate the differential,\n        and the result is a new representation.\n\n        Parameters\n        ----------\n        op : `~operator` callable\n            Operator to apply (e.g., `~operator.add`, `~operator.sub`, etc.\n        other : `~astropy.coordinates.BaseRepresentation` instance\n            The other differential or representation.\n        reverse : bool\n            Whether the operands should be reversed (e.g., as we got here via\n            ``self.__rsub__`` because ``self`` is a subclass of ``other``).\n        \"\"\"\n        if isinstance(self, type(other)):\n            first, second = (self, other) if not reverse else (other, self)\n            return self.__class__(*[op(getattr(first, c), getattr(second, c))\n                                    for c in self.components])\n        else:\n            try:\n                self_cartesian = self.to_cartesian(other)\n            except TypeError:\n                return NotImplemented\n\n            return other._combine_operation(op, self_cartesian, not reverse)\n\n    def __sub__(self, other):\n        # avoid \"differential - representation\".\n        if isinstance(other, BaseRepresentation):\n            return NotImplemented\n        return super(BaseDifferential, self).__sub__(other)\n\n    def norm(self, base=None):\n        \"\"\"Vector norm.\n\n        The norm is the standard Frobenius norm, i.e., the square root of the\n        sum of the squares of all components with non-angular units.\n\n        Parameters\n        ----------\n        base : instance of ``self.base_representation``\n            Base relative to which the differentials are defined. This is\n            required to calculate the physical size of the differential for\n            all but cartesian differentials.\n\n        Returns\n        -------\n        norm : `astropy.units.Quantity`\n            Vector norm, with the same shape as the representation.\n        \"\"\"\n        return self.to_cartesian(base).norm()"
        },
        {
          "file": "astropy/coordinates/representation.py",
          "type": "function",
          "name": "_get_deriv_key",
          "class_name": "BaseDifferential",
          "code": "def _get_deriv_key(self, base):\n        \"\"\"Given a base (representation instance), determine the unit of the\n        derivative by removing the representation unit from the component units\n        of this differential.\n        \"\"\"\n\n        # This check is just a last resort so we don't return a strange unit key\n        # from accidentally passing in the wrong base.\n        if self.__class__ not in base._compatible_differentials:\n            raise TypeError(\"Differential class {0} is not compatible with the \"\n                            \"base (representation) class {1}\"\n                            .format(self.__class__, base.__class__))\n\n        for name in base.components:\n            comp = getattr(base, name)\n            d_comp = getattr(self, 'd_{0}'.format(name), None)\n            if d_comp:\n                d_unit = comp.unit / d_comp.unit\n                # Get the si unit without a scale by going via Quantity;\n                # `.si` causes the scale to be included in the value.\n                return str(u.Quantity(1., d_unit).si.unit)\n\n        else:\n            raise RuntimeError(\"Invalid representation-differential match! Not \"\n                               \"sure how we got into this state.\")"
        }
      ]
    }
  ]
}