{
  "instance_id": "astropy__astropy-7746",
  "repo": "astropy/astropy",
  "created_at": "2018-08-20T14:07:20Z",
  "problem_statement": "Issue when passing empty lists/arrays to WCS transformations\nThe following should not fail but instead should return empty lists/arrays:\r\n\r\n```\r\nIn [1]: from astropy.wcs import WCS\r\n\r\nIn [2]: wcs = WCS('2MASS_h.fits')\r\n\r\nIn [3]: wcs.wcs_pix2world([], [], 0)\r\n---------------------------------------------------------------------------\r\nInconsistentAxisTypesError                Traceback (most recent call last)\r\n<ipython-input-3-e2cc0e97941a> in <module>()\r\n----> 1 wcs.wcs_pix2world([], [], 0)\r\n\r\n~/Dropbox/Code/Astropy/astropy/astropy/wcs/wcs.py in wcs_pix2world(self, *args, **kwargs)\r\n   1352         return self._array_converter(\r\n   1353             lambda xy, o: self.wcs.p2s(xy, o)['world'],\r\n-> 1354             'output', *args, **kwargs)\r\n   1355     wcs_pix2world.__doc__ = \"\"\"\r\n   1356         Transforms pixel coordinates to world coordinates by doing\r\n\r\n~/Dropbox/Code/Astropy/astropy/astropy/wcs/wcs.py in _array_converter(self, func, sky, ra_dec_order, *args)\r\n   1267                     \"a 1-D array for each axis, followed by an origin.\")\r\n   1268 \r\n-> 1269             return _return_list_of_arrays(axes, origin)\r\n   1270 \r\n   1271         raise TypeError(\r\n\r\n~/Dropbox/Code/Astropy/astropy/astropy/wcs/wcs.py in _return_list_of_arrays(axes, origin)\r\n   1223             if ra_dec_order and sky == 'input':\r\n   1224                 xy = self._denormalize_sky(xy)\r\n-> 1225             output = func(xy, origin)\r\n   1226             if ra_dec_order and sky == 'output':\r\n   1227                 output = self._normalize_sky(output)\r\n\r\n~/Dropbox/Code/Astropy/astropy/astropy/wcs/wcs.py in <lambda>(xy, o)\r\n   1351             raise ValueError(\"No basic WCS settings were created.\")\r\n   1352         return self._array_converter(\r\n-> 1353             lambda xy, o: self.wcs.p2s(xy, o)['world'],\r\n   1354             'output', *args, **kwargs)\r\n   1355     wcs_pix2world.__doc__ = \"\"\"\r\n\r\nInconsistentAxisTypesError: ERROR 4 in wcsp2s() at line 2646 of file cextern/wcslib/C/wcs.c:\r\nncoord and/or nelem inconsistent with the wcsprm.\r\n```\n",
  "patch": "diff --git a/astropy/wcs/wcs.py b/astropy/wcs/wcs.py\n--- a/astropy/wcs/wcs.py\n+++ b/astropy/wcs/wcs.py\n@@ -1212,6 +1212,9 @@ def _array_converter(self, func, sky, *args, ra_dec_order=False):\n         \"\"\"\n \n         def _return_list_of_arrays(axes, origin):\n+            if any([x.size == 0 for x in axes]):\n+                return axes\n+\n             try:\n                 axes = np.broadcast_arrays(*axes)\n             except ValueError:\n@@ -1235,6 +1238,8 @@ def _return_single_array(xy, origin):\n                 raise ValueError(\n                     \"When providing two arguments, the array must be \"\n                     \"of shape (N, {0})\".format(self.naxis))\n+            if 0 in xy.shape:\n+                return xy\n             if ra_dec_order and sky == 'input':\n                 xy = self._denormalize_sky(xy)\n             result = func(xy, origin)\n",
  "similar_bug_items": [
    {
      "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": 6938,
      "pr_title": "Make sure that the 'D' format is used in fits ascii tables",
      "pr_body": "Fix #6921",
      "issue_id": 6921,
      "issue_title": "Possible bug in io.fits related to D exponents",
      "issue_body": "I 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.",
      "issue_closed_at": "2018-01-15T21:21:14Z",
      "base_commit": "c76af9ed6bb89bfba45b9f5bc1e635188278e2fa",
      "changes": [
        {
          "file": "astropy/io/fits/fitsrec.py",
          "type": "function",
          "name": "_scale_back_ascii",
          "class_name": "FITS_rec",
          "code": "def _scale_back_ascii(self, col_idx, input_field, output_field):\n        \"\"\"\n        Convert internal array values back to ASCII table representation.\n\n        The ``input_field`` is the internal representation of the values, and\n        the ``output_field`` is the character array representing the ASCII\n        output that will be written.\n        \"\"\"\n\n        starts = self._coldefs.starts[:]\n        spans = self._coldefs.spans\n        format = self._coldefs[col_idx].format\n\n        # The the index of the \"end\" column of the record, beyond\n        # which we can't write\n        end = super().field(-1).itemsize\n        starts.append(end + starts[-1])\n\n        if col_idx > 0:\n            lead = starts[col_idx] - starts[col_idx - 1] - spans[col_idx - 1]\n        else:\n            lead = 0\n\n        if lead < 0:\n            warnings.warn('Column {!r} starting point overlaps the previous '\n                          'column.'.format(col_idx + 1))\n\n        trail = starts[col_idx + 1] - starts[col_idx] - spans[col_idx]\n\n        if trail < 0:\n            warnings.warn('Column {!r} ending point overlaps the next '\n                          'column.'.format(col_idx + 1))\n\n        # TODO: It would be nice if these string column formatting\n        # details were left to a specialized class, as is the case\n        # with FormatX and FormatP\n        if 'A' in format:\n            _pc = '{:'\n        else:\n            _pc = '{:>'\n\n        fmt = ''.join([_pc, format[1:], ASCII2STR[format[0]], '}',\n                       (' ' * trail)])\n\n        # Even if the format precision is 0, we should output a decimal point\n        # as long as there is space to do so--not including a decimal point in\n        # a float value is discouraged by the FITS Standard\n        trailing_decimal = (format.precision == 0 and\n                            format.format in ('F', 'E', 'D'))\n\n        # not using numarray.strings's num2char because the\n        # result is not allowed to expand (as C/Python does).\n        for jdx, value in enumerate(input_field):\n            value = fmt.format(value)\n            if len(value) > starts[col_idx + 1] - starts[col_idx]:\n                raise ValueError(\n                    \"Value {!r} does not fit into the output's itemsize of \"\n                    \"{}.\".format(value, spans[col_idx]))\n\n            if trailing_decimal and value[0] == ' ':\n                # We have some extra space in the field for the trailing\n                # decimal point\n                value = value[1:] + '.'\n\n            output_field[jdx] = value\n\n        # Replace exponent separator in floating point numbers\n        if 'D' in format:\n            output_field.replace(encode_ascii('E'), encode_ascii('D'))"
        }
      ]
    },
    {
      "pr_number": 3657,
      "pr_title": "#3564 assume sensible default for col_ends or col_starts if omitted for fixed-width table",
      "pr_body": "This is to resolve #3564. I've changed the structure of an if-statement in io.ascii.fixedwidth.py inside the FixedWidthHeader.get_fixedwidth_params method to allow me to handle the case where only one of col_starts or col_ends is specified. The behavious for other cases is not changed.\n\nI've also added tests, and two examples in the documentation.\n\nI have run\n\n```\npython setup.py test --package=io.ascii\n```\n\nusing Python 2.7.6 and 3.4.0 for this change and received no failures.\n",
      "issue_id": 3564,
      "issue_title": "For fixed with reader, make ``col_ends`` default sensible",
      "issue_body": "For the fixed width reader in io.ascii it would be nice to avoid having to do things like\n\n```\ncol_starts=[0,5,20,32,42,55], col_ends=[5,20,32,42,55,100]\n```\n\nBy default, one could assume that columns end where the next start, and have the last item default to infinity (or whatever means to the end of the line).\n",
      "issue_closed_at": "2015-04-03T15:38:08Z",
      "base_commit": "66d3a10569adb5de046a0ff458038293748228f7",
      "changes": [
        {
          "file": "astropy/io/ascii/fixedwidth.py",
          "type": "function",
          "name": "get_fixedwidth_params",
          "class_name": "FixedWidthHeader",
          "code": "def get_fixedwidth_params(self, line):\n        \"\"\"\n        Split ``line`` on the delimiter and determine column values and\n        column start and end positions.  This might include null columns with\n        zero length (e.g. for ``header row = \"| col1 || col2 | col3 |\"`` or\n        ``header2_row = \"----- ------- -----\"``).  The null columns are\n        stripped out.  Returns the values between delimiters and the\n        corresponding start and end positions.\n\n        Parameters\n        ----------\n        line : str\n            Input line\n\n        Returns\n        -------\n        vals : list\n            List of values.\n        starts : list\n            List of starting indices.\n        ends : list\n            List of ending indices.\n\n        \"\"\"\n\n        # If column positions are already specified then just use those, otherwise\n        # figure out positions between delimiters.\n        if self.col_starts is not None and self.col_ends is not None:\n            starts = list(self.col_starts)  # could be any iterable, e.g. np.array\n            ends = [x + 1 for x in self.col_ends]  # user supplies inclusive endpoint\n            if len(starts) != len(ends):\n                raise ValueError('Fixed width col_starts and col_ends must have the same length')\n            vals = [line[start:end].strip() for start, end in zip(starts, ends)]\n        else:\n            # There might be a cleaner way to do this but it works...\n            vals = line.split(self.splitter.delimiter)\n            starts = [0]\n            ends = []\n            for val in vals:\n                if val:\n                    ends.append(starts[-1] + len(val))\n                    starts.append(ends[-1] + 1)\n                else:\n                    starts[-1] += 1\n            starts = starts[:-1]\n            vals = [x.strip() for x in vals if x]\n            if len(vals) != len(starts) or len(vals) != len(ends):\n                raise InconsistentTableError('Error parsing fixed width header')\n\n        return vals, starts, ends"
        },
        {
          "file": "astropy/io/ascii/fixedwidth.py",
          "type": "function",
          "name": "get_fixedwidth_params",
          "class_name": "FixedWidthHeader",
          "code": "def get_fixedwidth_params(self, line):\n        \"\"\"\n        Split ``line`` on the delimiter and determine column values and\n        column start and end positions.  This might include null columns with\n        zero length (e.g. for ``header row = \"| col1 || col2 | col3 |\"`` or\n        ``header2_row = \"----- ------- -----\"``).  The null columns are\n        stripped out.  Returns the values between delimiters and the\n        corresponding start and end positions.\n\n        Parameters\n        ----------\n        line : str\n            Input line\n\n        Returns\n        -------\n        vals : list\n            List of values.\n        starts : list\n            List of starting indices.\n        ends : list\n            List of ending indices.\n\n        \"\"\"\n\n        # If column positions are already specified then just use those, otherwise\n        # figure out positions between delimiters.\n        if self.col_starts is not None and self.col_ends is not None:\n            starts = list(self.col_starts)  # could be any iterable, e.g. np.array\n            ends = [x + 1 for x in self.col_ends]  # user supplies inclusive endpoint\n            if len(starts) != len(ends):\n                raise ValueError('Fixed width col_starts and col_ends must have the same length')\n            vals = [line[start:end].strip() for start, end in zip(starts, ends)]\n        else:\n            # There might be a cleaner way to do this but it works...\n            vals = line.split(self.splitter.delimiter)\n            starts = [0]\n            ends = []\n            for val in vals:\n                if val:\n                    ends.append(starts[-1] + len(val))\n                    starts.append(ends[-1] + 1)\n                else:\n                    starts[-1] += 1\n            starts = starts[:-1]\n            vals = [x.strip() for x in vals if x]\n            if len(vals) != len(starts) or len(vals) != len(ends):\n                raise InconsistentTableError('Error parsing fixed width header')\n\n        return vals, starts, ends"
        }
      ]
    },
    {
      "pr_number": 2750,
      "pr_title": ".fz fits files KeyError: \"Keyword 'ZSIMPLE' not found in 0.3.2 versus 0.3",
      "pr_body": "I have some .fz fits files that are giving the error:\n\nKeyError: \"Keyword 'ZSIMPLE' not found.\"\n\nwhen I use astropy 0.3.2 and also 0.4rc3.dev9443 but 0.3 reads the header OK\n\nThe header looks like\n.\n.\n.\nZVAL2   =                    4 / bytes per pixel (1, 2, 4, or 8)  \nEXTNAME = 'COMPRESSED_IMAGE'  \nZSIMPLE =                    T / file does conform to FITS standard  \nZBITPIX =                  -32 / number of bits per data pixel  \n.\n.\n.\n",
      "issue_id": 2750,
      "issue_title": ".fz fits files KeyError: \"Keyword 'ZSIMPLE' not found in 0.3.2 versus 0.3",
      "issue_body": "I have some .fz fits files that are giving the error:\n\nKeyError: \"Keyword 'ZSIMPLE' not found.\"\n\nwhen I use astropy 0.3.2 and also 0.4rc3.dev9443 but 0.3 reads the header OK\n\nThe header looks like\n.\n.\n.\nZVAL2   =                    4 / bytes per pixel (1, 2, 4, or 8)  \nEXTNAME = 'COMPRESSED_IMAGE'  \nZSIMPLE =                    T / file does conform to FITS standard  \nZBITPIX =                  -32 / number of bits per data pixel  \n.\n.\n.\n",
      "issue_closed_at": "2014-09-16T14:07:45Z",
      "base_commit": "c027ce8414ab4eef47b2f5821960dc75b331d611",
      "changes": [
        {
          "file": "astropy/io/fits/hdu/compressed.py",
          "type": "function",
          "name": "header",
          "class_name": "CompImageHDU",
          "code": "def header(self):\n        # The header attribute is the header for the image data.  It\n        # is not actually stored in the object dictionary.  Instead,\n        # the _image_header is stored.  If the _image_header attribute\n        # has already been defined we just return it.  If not, we nust\n        # create it from the table header (the _header attribute).\n        if hasattr(self, '_image_header'):\n            return self._image_header\n\n        # Start with a copy of the table header.\n        image_header = self._header.copy()\n\n        # Delete cards that are related to the table.  And move\n        # the values of those cards that relate to the image from\n        # their corresponding table cards.  These include\n        # ZBITPIX -> BITPIX, ZNAXIS -> NAXIS, and ZNAXISn -> NAXISn.\n        for keyword in list(image_header):\n            if CompImageHeader._is_reserved_keyword(keyword, warn=False):\n                del image_header[keyword]\n\n        if 'ZSIMPLE' in self._header:\n            image_header.set('SIMPLE', self._header['ZSIMPLE'],\n                             self._header.comments['ZSIMPLE'], before=0)\n        elif 'ZTENSION' in self._header:\n            if self._header['ZTENSION'] != 'IMAGE':\n                warnings.warn(\"ZTENSION keyword in compressed \"\n                              \"extension != 'IMAGE'\", AstropyUserWarning)\n            image_header.set('XTENSION', 'IMAGE',\n                             self._header.comments['ZTENSION'], before=0)\n        else:\n            image_header.set('XTENSION', 'IMAGE', before=0)\n\n\n        image_header.set('BITPIX', self._header['ZBITPIX'],\n                         self._header.comments['ZBITPIX'], before=1)\n\n        image_header.set('NAXIS', self._header['ZNAXIS'],\n                         self._header.comments['ZNAXIS'], before=2)\n\n        last_naxis = 'NAXIS'\n        for idx in range(image_header['NAXIS']):\n            znaxis = 'ZNAXIS' + str(idx + 1)\n            naxis = znaxis[1:]\n            image_header.set(naxis, self._header[znaxis],\n                             self._header.comments[znaxis],\n                             after=last_naxis)\n            last_naxis = naxis\n\n        # Delete any other spurious NAXISn keywords:\n        naxis = image_header['NAXIS']\n        for keyword in list(image_header['NAXIS?*']):\n            try:\n                n = int(keyword[5:])\n            except:\n                continue\n\n            if n > naxis:\n                del image_header[keyword]\n\n        # Although PCOUNT and GCOUNT are considered mandatory for IMAGE HDUs,\n        # ZPCOUNT and ZGCOUNT are optional, probably because for IMAGE HDUs\n        # their values are always 0 and 1 respectively\n        if 'ZPCOUNT' in self._header:\n            image_header.set('PCOUNT', self._header['ZPCOUNT'],\n                             self._header.comments['ZPCOUNT'],\n                             after=last_naxis)\n        else:\n            image_header.set('PCOUNT', 0, after=last_naxis)\n\n        if 'ZGCOUNT' in self._header:\n            image_header.set('GCOUNT', self._header['ZGCOUNT'],\n                             self._header.comments['ZGCOUNT'],\n                             after='PCOUNT')\n        else:\n            image_header.set('GCOUNT', 1, after='PCOUNT')\n\n        if 'ZEXTEND' in self._header:\n            image_header.set('EXTEND', self._header['ZEXTEND'],\n                             self._header.comments['ZEXTEND'])\n\n        if 'ZBLOCKED' in self._header:\n            image_header.set('BLOCKED', self._header['ZBLOCKED'],\n                             self._header.comments['ZBLOCKED'])\n\n        # Move the ZHECKSUM and ZDATASUM cards to the image header\n        # as CHECKSUM and DATASUM\n        if 'ZHECKSUM' in self._header:\n            image_header.set('CHECKSUM', self._header['ZHECKSUM'],\n                             self._header.comments['ZHECKSUM'])\n\n        if 'ZDATASUM' in self._header:\n            image_header.set('DATASUM', self._header['ZDATASUM'],\n                             self._header.comments['ZDATASUM'])\n\n        # Remove the EXTNAME card if the value in the table header\n        # is the default value of COMPRESSED_IMAGE.\n        if ('EXTNAME' in self._header and\n                self._header['EXTNAME'] == 'COMPRESSED_IMAGE'):\n            del image_header['EXTNAME']\n\n        # Look to see if there are any blank cards in the table\n        # header.  If there are, there should be the same number\n        # of blank cards in the image header.  Add blank cards to\n        # the image header to make it so.\n        table_blanks = self._header._countblanks()\n        image_blanks = image_header._countblanks()\n\n        for _ in range(table_blanks - image_blanks):\n            image_header.append()\n\n        # Create the CompImageHeader that syncs with the table header, and save\n        # it off to self._image_header so it can be referenced later\n        # unambiguously\n        self._image_header = CompImageHeader(self._header, image_header)\n\n        return self._image_header"
        }
      ]
    },
    {
      "pr_number": 3530,
      "pr_title": "Define Time.__bool__ and add tests (closes #3520)",
      "pr_body": "This ensures any time object evaluates to `True` unless it is empty (see #3520). Also adds tests for both `__bool__` and `__len__`.\n",
      "issue_id": 3520,
      "issue_title": "time.Time __bool__ and __len__ unexpected behaviour",
      "issue_body": "I noticed this behaviour of time.Time in 0.4.4 and was able to replicate it in 1.0:\n\n``` python\n>>> t = time.Time('2015-01-01')\n>>> len(t)\nTraceback (most recent call last):\n  File \"<stdin>\", line 1, in <module>\n  File \"/Users/michele/Library/Enthought/Canopy_64bit/User/lib/python2.7/site-packages/astropy/time/core.py\", line 940, in __len__\n    return len(self.jd1)\nTypeError: object of type 'float' has no len()\n```\n\nFor behaviour consistent with the standard Python `datetime.datetime`, `len(t)` should instead throw an `AttributeError`.\n\nThis further comes in when one tries:\n\n``` python\n>>> if t:\n...     print(t)\nTraceback (most recent call last):\n  File \"<stdin>\", line 1, in <module>\n  File \"/Users/michele/Library/Enthought/Canopy_64bit/User/lib/python2.7/site-packages/astropy/time/core.py\", line 940, in __len__\n    return len(self.jd1)\nTypeError: object of type 'float' has no len()\n```\n\nFor behaviour consistent with `datetime`, `bool(t)` for a `time.Time` would always return `True`.\n",
      "issue_closed_at": "2015-02-24T15:21:23Z",
      "base_commit": "b3c2c761b9ec960b29cabf797c7d05faf08af7ed",
      "changes": [
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "shape",
          "class_name": "Time",
          "code": "def shape(self):\n        return self._time.jd1.shape"
        },
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "sidereal_time",
          "class_name": "Time",
          "code": "def sidereal_time(self, kind, longitude=None, model=None):\n        \"\"\"Calculate sidereal time.\n\n        Parameters\n        ---------------\n        kind : str\n            ``'mean'`` or ``'apparent'``, i.e., accounting for precession\n            only, or also for nutation.\n        longitude : `~astropy.units.Quantity`, `str`, or `None`; optional\n           The longitude on the Earth at which to compute the sidereal time.\n            Can be given as a `~astropy.units.Quantity` with angular units\n            (or an `~astropy.coordinates.Angle` or\n            `~astropy.coordinates.Longitude`), or as a name of an\n            observatory (currently, only ``'greenwich'`` is supported,\n            equivalent to 0 deg).  If `None` (default), the ``lon`` attribute of\n            the Time object is used.\n        model : str or `None`; optional\n            Precession (and nutation) model to use.  The available ones are:\n            - {0}: {1}\n            - {2}: {3}\n            If `None` (default), the last (most recent) one from the appropriate\n            list above is used.\n\n        Returns\n        -------\n        sidereal time : `~astropy.coordinates.Longitude`\n            Sidereal time as a quantity with units of hourangle\n        \"\"\"  # docstring is formatted below\n\n        from ..coordinates import Longitude\n\n        if kind.lower() not in SIDEREAL_TIME_MODELS.keys():\n            raise ValueError('The kind of sidereal time has to be {0}'.format(\n                ' or '.join(sorted(SIDEREAL_TIME_MODELS.keys()))))\n\n        available_models = SIDEREAL_TIME_MODELS[kind.lower()]\n\n        if model is None:\n            model = sorted(available_models.keys())[-1]\n        else:\n            if model.upper() not in available_models:\n                raise ValueError(\n                    'Model {0} not implemented for {1} sidereal time; '\n                    'available models are {2}'\n                    .format(model, kind, sorted(available_models.keys())))\n\n        if longitude is None:\n            if self.location is None:\n                raise ValueError('No longitude is given but the location for '\n                                 'the Time object is not set.')\n            longitude = self.location.longitude\n        elif longitude == 'greenwich':\n            longitude = Longitude(0., u.degree,\n                                  wrap_angle=180.*u.degree)\n        else:\n            # sanity check on input\n            longitude = Longitude(longitude, u.degree,\n                                  wrap_angle=180.*u.degree)\n\n        gst = self._erfa_sidereal_time(available_models[model.upper()])\n        return Longitude(gst + longitude, u.hourangle)"
        },
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "_set_delta_tdb_tt",
          "class_name": "Time",
          "code": "def _set_delta_tdb_tt(self, val):\n        self._delta_tdb_tt = self._match_shape(val)"
        }
      ]
    }
  ]
}