{
  "instance_id": "astropy__astropy-14995",
  "repo": "astropy/astropy",
  "created_at": "2023-06-27T19:48:18Z",
  "problem_statement": "In v5.3, NDDataRef mask propagation fails when one of the operand does not have a mask\n### Description\n\nThis applies to v5.3. \r\n\r\nIt looks like when one of the operand does not have a mask, the mask propagation when doing arithmetic, in particular with `handle_mask=np.bitwise_or` fails.  This is not a problem in v5.2.\r\n\r\nI don't know enough about how all that works, but it seems from the error that the operand without a mask is set as a mask of None's and then the bitwise_or tries to operate on an integer and a None and fails.\n\n### Expected behavior\n\nWhen one of the operand does not have mask, the mask that exists should just be copied over to the output.  Or whatever was done in that situation in v5.2 where there's no problem.\n\n### How to Reproduce\n\nThis is with v5.3.   With v5.2, there are no errors.\r\n\r\n```\r\n>>> import numpy as np\r\n>>> from astropy.nddata import NDDataRef\r\n\r\n>>> array = np.array([[0, 1, 0], [1, 0, 1], [0, 1, 0]])\r\n>>> mask = np.array([[0, 1, 64], [8, 0, 1], [2, 1, 0]])\r\n\r\n>>> nref_nomask = NDDataRef(array)\r\n>>> nref_mask = NDDataRef(array, mask=mask)\r\n\r\n# multiply no mask by constant (no mask * no mask)\r\n>>> nref_nomask.multiply(1., handle_mask=np.bitwise_or).mask   # returns nothing, no mask,  OK\r\n\r\n# multiply no mask by itself (no mask * no mask)\r\n>>> nref_nomask.multiply(nref_nomask, handle_mask=np.bitwise_or).mask # return nothing, no mask, OK\r\n\r\n# multiply mask by constant (mask * no mask)\r\n>>> nref_mask.multiply(1., handle_mask=np.bitwise_or).mask\r\n...\r\nTypeError: unsupported operand type(s) for |: 'int' and 'NoneType'\r\n\r\n# multiply mask by itself (mask * mask)\r\n>>> nref_mask.multiply(nref_mask, handle_mask=np.bitwise_or).mask\r\narray([[ 0,  1, 64],\r\n       [ 8,  0,  1],\r\n       [ 2,  1,  0]])\r\n\r\n# multiply mask by no mask (mask * no mask)\r\n>>> nref_mask.multiply(nref_nomask, handle_mask=np.bitwise_or).mask\r\n...\r\nTypeError: unsupported operand type(s) for |: 'int' and 'NoneType'\r\n```\r\n\n\n### Versions\n\n>>> import sys; print(\"Python\", sys.version)\r\nPython 3.10.11 | packaged by conda-forge | (main, May 10 2023, 19:07:22) [Clang 14.0.6 ]\r\n>>> import astropy; print(\"astropy\", astropy.__version__)\r\nastropy 5.3\r\n>>> import numpy; print(\"Numpy\", numpy.__version__)\r\nNumpy 1.24.3\r\n>>> import erfa; print(\"pyerfa\", erfa.__version__)\r\npyerfa 2.0.0.3\r\n>>> import scipy; print(\"Scipy\", scipy.__version__)\r\nScipy 1.10.1\r\n>>> import matplotlib; print(\"Matplotlib\", matplotlib.__version__)\r\nMatplotlib 3.7.1\r\n\n",
  "patch": "diff --git a/astropy/nddata/mixins/ndarithmetic.py b/astropy/nddata/mixins/ndarithmetic.py\n--- a/astropy/nddata/mixins/ndarithmetic.py\n+++ b/astropy/nddata/mixins/ndarithmetic.py\n@@ -520,10 +520,10 @@ def _arithmetic_mask(self, operation, operand, handle_mask, axis=None, **kwds):\n         elif self.mask is None and operand is not None:\n             # Make a copy so there is no reference in the result.\n             return deepcopy(operand.mask)\n-        elif operand is None:\n+        elif operand.mask is None:\n             return deepcopy(self.mask)\n         else:\n-            # Now lets calculate the resulting mask (operation enforces copy)\n+            # Now let's calculate the resulting mask (operation enforces copy)\n             return handle_mask(self.mask, operand.mask, **kwds)\n \n     def _arithmetic_wcs(self, operation, operand, compare_wcs, **kwds):\n",
  "similar_bug_items": [
    {
      "pr_number": 2796,
      "pr_title": "Add check for np.ma.nomask in NDData mask setter",
      "pr_body": "Adds a check for a passed value of `np.ma.nomask` in the NDData mask setter.  This prevents a `ValueError` being raised when passing a numpy masked array which has no masked entries to `NDData()` (issue #2784).\n",
      "issue_id": 2784,
      "issue_title": "Initialising NDData objects with a numpy masked array object fails if there are no masked entries",
      "issue_body": "If you attempt to initialise an NDData object using a numpy masked array object which happens to have no masked entries it raises an exception, e.g.\n\n```\nmaskedarray = numpy.ma.array(numpy.ones((10,10)))\nastropy.nddata.NDData(maskedarray)\n```\n\nresults in `ValueError: dimensions of mask do not match data` (confirmed with Python 3.4.1, numpy 1.8.1 and astropy 0.4).\n\nThe problem is the mask setter code in nddata.py:\n\n```\n@mask.setter\ndef mask(self, value):\n    if value is not None:\n        mask = np.array(value, dtype=np.bool_, copy=False)\n        if mask.shape != self.shape:\n            raise ValueError(\"dimensions of mask do not match data\")\n        else:\n            self._mask = mask\n    else:\n        # internal representation should be one numpy understands\n        self._mask = np.ma.nomask\n```\n\nThe 3rd line tests whether the provided mask `value` is equal to `None` but the mask of a numpy masked array with no masked elements is not `None`, it is `numpy.ma.nomask` (which evaluates to `False`).  As a result the expression in the if statement evaluates as `True` and the setter goes ahead to convert `numpy.ma.nomask` to a single element boolean array `[False,]` before raising an exception because the shape doesn't match that of the data.\n\nI think all that is required here is to replace \n\n`if value is not None:` \n\nwith \n\n`if (value is not None) and (value != np.ma.nomask):`\n",
      "issue_closed_at": "2014-08-01T15:42:33Z",
      "base_commit": "ecaa91e59e3acc314f197e2abbfd4c43df8f10ad",
      "changes": [
        {
          "file": "astropy/nddata/nddata.py",
          "type": "function",
          "name": "mask",
          "class_name": "NDData",
          "code": "def mask(self, value):\n        if value is not None:\n            mask = np.array(value, dtype=np.bool_, copy=False)\n            if mask.shape != self.shape:\n                raise ValueError(\"dimensions of mask do not match data\")\n            else:\n                self._mask = mask\n        else:\n            # internal representation should be one numpy understands\n            self._mask = np.ma.nomask"
        }
      ]
    },
    {
      "pr_number": 889,
      "pr_title": "Add TimeAstropyTime format to instantiate from list of Times",
      "pr_body": "This will close #886.  (Sorry, don't have hub setup on my mac to attach code to that issue).\n",
      "issue_id": 886,
      "issue_title": "creating a Time with multiple Time(format=datetime) objects gives an error",
      "issue_body": "I've noticed some unexpected behavior with the `datetime` support in `time` recently added by @taldcroft in #860.  It may not actually be related to that (that is, it might be true for all formats), but I first noticed it after the #860 merge, so it _might_ be connected.\n\nWhen I do `Time([dt1,dt2],scale='utc')` (`dt1` and `dt2` are `datetime.datetime` objects), I get exactly what I expect: A `Time` object that contains an array with two elements.  Similarly, when I do `Time(t1)` (where `t1` is a time object) I get out the natural expectation: a copy of `t1`.  However, if I do this:\n\n```\nt1 = Time(dt1, scale='utc')\nt2 = Time(dt2, scale='utc')\ntarr = Time([t1, t2])\n```\n\nI get an error about it being an invalid format.  I would have instead expected to get back a new `Time` object with a two-element array that looks just like `Time([dt1,dt2],scale='utc')`\n",
      "issue_closed_at": "2013-03-22T06:27:44Z",
      "base_commit": "dad4904adda976bcb01a1216ba20cb05dbd4bb21",
      "changes": [
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "_init_from_vals",
          "class_name": "Time",
          "code": "def _init_from_vals(self, val, val2, format, scale, copy):\n        \"\"\"\n        Set the internal _format, scale, and _time attrs from user\n        inputs.  This handles coercion into the correct shapes and\n        some basic input validation.\n        \"\"\"\n\n        # Coerce val into a 1-d array\n        val, val_ndim = _make_1d_array(val, copy)\n\n        # If val2 is None then replace with zeros of the same length\n        if val2 is None:\n            val2 = np.zeros(len(val), dtype=np.double)\n            val2_ndim = val_ndim\n        else:\n            val2, val2_ndim = _make_1d_array(val2, copy)\n\n        # Consistency checks\n        if len(val) != len(val2):\n            raise ValueError('Input val and val2 must match in length')\n\n        self.is_scalar = (val_ndim == 0)\n        if val_ndim != val2_ndim:\n            raise ValueError('Input val and val2 must have same dimensions')\n\n        if scale is not None and scale not in self.SCALES:\n            raise ScaleValueError(\"Scale {0} is not in the allowed scales {1}\"\n                                  .format(repr(scale), sorted(self.SCALES)))\n\n        # Parse / convert input values into internal jd1, jd2 based on format\n        self._format, self._time = self._get_time_fmt(val, val2, format, scale)"
        },
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "_get_time_fmt",
          "class_name": "Time",
          "code": "def _get_time_fmt(self, val, val2, format, scale):\n        \"\"\"\n        Given the supplied val, val2, format and scale try to instantiate\n        the corresponding TimeFormat class to convert the input values into\n        the internal jd1 and jd2.\n\n        If format is None and the input is a string-type or object array then guess\n        available formats and stop when one matches.\n        \"\"\"\n\n        if format is None and val.dtype.kind in ('S', 'U', 'O'):\n            formats = [(name, cls) for name, cls in self.FORMATS.items()\n                       if issubclass(cls, TimeUnique)]\n            err_msg = 'any of the formats where the format keyword is optional {0}'.format(\n                [name for name, cls in formats])\n        elif format not in self.FORMATS:\n            if format is None:\n                raise ValueError(\"No time format was given, and the input is not unique\")\n            else:\n                raise ValueError(\"Format {0} is not one of the allowed \"\n                                 \"formats {1}\".format(repr(format), sorted(self.FORMATS)))\n        else:\n            formats = [(format, self.FORMATS[format])]\n            err_msg = 'the format class {0}'.format(format)\n\n        for format, FormatClass in formats:\n            try:\n                return format, FormatClass(val, val2, scale, self.precision,\n                                           self.in_subfmt, self.out_subfmt)\n            except (ValueError, TypeError):\n                pass\n        else:\n            raise ValueError('Input values did not match {0}'.format(err_msg))"
        },
        {
          "file": "astropy/time/core.py",
          "type": "class",
          "name": "TimeUnique",
          "code": "class TimeUnique(TimeFormat):\n    \"\"\"\n    Base class for time formats that can uniquely create a time object\n    without requiring an explicit format specifier.  This class does\n    nothing but provide inheritance to identify a class as unique.\n    \"\"\"\n    pass"
        }
      ]
    },
    {
      "pr_number": 13068,
      "pr_title": "Issue #13008, Time() precision should be 0-9",
      "pr_body": "<!-- This comments are hidden when you submit the pull request,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/main/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/main/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- If you are new or need to be re-acquainted with Astropy\r\ncontributing workflow, please see\r\nhttp://docs.astropy.org/en/latest/development/workflow/development_workflow.html .\r\nThere is even a practical example at\r\nhttps://docs.astropy.org/en/latest/development/workflow/git_edit_workflow_examples.html#astropy-fix-example . -->\r\n\r\n<!-- Astropy coding style guidelines can be found here:\r\nhttps://docs.astropy.org/en/latest/development/codeguide.html#coding-style-conventions\r\nOur testing infrastructure enforces to follow a subset of the PEP8 to be\r\nfollowed. You can check locally whether your changes have followed these by\r\nrunning the following command:\r\n\r\ntox -e codestyle\r\n\r\n-->\r\n\r\n<!-- Please just have a quick search on GitHub to see if a similar\r\npull request has already been posted.\r\nWe have old closed pull requests that might provide useful code or ideas\r\nthat directly tie in with your pull request. -->\r\n\r\n<!-- We have several automatic features that run when a pull request is open.\r\nThey can appear daunting but do not worry because maintainers will help\r\nyou navigate them, if necessary. -->\r\n\r\n### Description\r\n<!-- Provide a general description of what your pull request does.\r\nComplete the following sentence and add relevant details as you see fit. -->\r\n\r\n<!-- In addition please ensure that the pull request title is descriptive\r\nand allows maintainers to infer the applicable subpackage(s). -->\r\n\r\n<!-- READ THIS FOR MANUAL BACKPORT FROM A MAINTAINER:\r\nApply \"skip-basebranch-check\" label **before** you open the PR! -->\r\n\r\nThis pull request is to address: `astropy.time.Time()` has a precision attribute that should be an int in the range 0-9 (inclusive). The setter has already been coded to raise an exception if the precision is outside that range (see [here](https://github.com/astropy/astropy/blob/main/astropy/time/core.py#L610-L611)). However, `Time()` can still be _initialized_ with a precision outside of this range. Thus,\r\n```\r\n>>> from astropy.time import Time\r\n>>> t1 = Time(123, fromat=\"jd\")\r\n>>> t1.precision = 10\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/brett/env/lib/python3.8/site-packages/astropy/time/core.py\", line 610, in precision\r\n    raise ValueError('precision attribute must be an int between '\r\nValueError: precision attribute must be an int between 0 and 9\r\n```\r\nproduces an exception, but \r\n```\r\n>>> t2 = Time(123, format=\"jd\", precision=10)\r\n>>> \r\n```\r\ndoes not.\r\n\r\nApologies if this PR is unnecessary or if I didn't follow the contributing guidelines. I'm at work rn, but I can certainly update this PR if needed.\r\n\r\n<!-- If the pull request closes any open issues you can add this.\r\nIf you replace <Issue Number> with a number, GitHub will automatically link it.\r\nIf this pull request is unrelated to any issues, please remove\r\nthe following line. -->\r\n\r\nFixes #13008\r\n\r\n### Checklist for package maintainer(s)\r\n<!-- This section is to be filled by package maintainer(s) who will\r\nreview this pull request. -->\r\n\r\nThis checklist is meant to remind the package maintainer(s) who will review this pull request of some common things to look for. This list is not exhaustive.\r\n\r\n- [x] Do the proposed changes actually accomplish desired goals?\r\n- [x] Do the proposed changes follow the [Astropy coding guidelines](https://docs.astropy.org/en/latest/development/codeguide.html)?\r\n- [x] Are tests added/updated as required? If so, do they follow the [Astropy testing guidelines](https://docs.astropy.org/en/latest/development/testguide.html)?\r\n- [x] Are docs added/updated as required? If so, do they follow the [Astropy documentation guidelines](https://docs.astropy.org/en/latest/development/docguide.html#astropy-documentation-rules-and-guidelines)?\r\n- [x] Is rebase and/or squash necessary? If so, please provide the author with appropriate instructions. Also see [\"When to rebase and squash commits\"](https://docs.astropy.org/en/latest/development/when_to_rebase.html).\r\n- [x] Did the CI pass? If no, are the failures related? If you need to run daily and weekly cron jobs as part of the PR, please apply the `Extra CI` label.\r\n- [x] Is a change log needed? If yes, did the change log check pass? If no, add the `no-changelog-entry-needed` label. If this is a manual backport, use the `skip-changelog-checks` label unless special changelog handling is necessary.\r\n- [x] Is this a big PR that makes a \"What's new?\" entry worthwhile and if so, is (1) a \"what's new\" entry included in this PR and (2) the \"whatsnew-needed\" label applied?\r\n- [x] Is a milestone set? Milestone must be set but `astropy-bot` check might be missing; do not let the green checkmark fool you.\r\n- [x] At the time of adding the milestone, if the milestone set requires a backport to release branch(es), apply the appropriate `backport-X.Y.x` label(s) *before* merge.\r\n",
      "issue_id": 13008,
      "issue_title": "Time from astropy.time not precise",
      "issue_body": "Hello,\r\n\r\nI encounter difficulties with Time. I'm working on a package to perform photometry and occultation. \r\n\r\nFor this last case, data need times values accurately estimated. Of course, data coming from different camera will will have different time format in the header.\r\n\r\nto manage this without passing long time to build a time parser, i decided to use Time object which do exactly what i need. The problem is, i dont arrive to make accurate conversion between different format using Time.\r\n\r\nlet's do an exemple:\r\n\r\n```\r\nt1 = '2022-03-24T23:13:41.390999'\r\nt1 = Time(t1, format = 'isot', precision = len(t1.split('.')[-1]))\r\nt2 = t1.to_value('jd')\r\n# result is 2459663.4678401737\r\n```\r\nnow let's do reverse\r\n\r\n```\r\nt2 = Time(t2, format = 'jd', precision = len(str(t2).split('.')[-1]))\r\nt3 = t2.to_value('isot')\r\n# result is 2022-03-24T23:13:41.0551352177\r\n```\r\nas you can see i don't fall back on the same value and the difference is quite high. I would like to fall back on the original one.\r\n\r\nthank you in advance\r\n",
      "issue_closed_at": "2022-06-27T13:08:16Z",
      "base_commit": "2288ecd4e9c4d3722d72b7f4a6555a34f4f04fc7",
      "changes": [
        {
          "file": "astropy/time/core.py",
          "type": "function",
          "name": "precision",
          "class_name": "TimeBase",
          "code": "def precision(self, val):\n        del self.cache\n        if not isinstance(val, int) or val < 0 or val > 9:\n            raise ValueError('precision attribute must be an int between '\n                             '0 and 9')\n        self._time.precision = val"
        },
        {
          "file": "astropy/time/formats.py",
          "type": "function",
          "name": "masked",
          "class_name": "TimeFormat",
          "code": "def masked(self):\n        if 'masked' not in self.cache:\n            self.cache['masked'] = bool(np.any(self.mask))\n        return self.cache['masked']"
        }
      ]
    },
    {
      "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": 11449,
      "pr_title": "Bug Fix #5857: Match catalog frames should use exact catalog frame",
      "pr_body": "Transformation to catalog frame is done twice. Once in matching.py and then in sky_coordinate.py.\r\n\r\nWe remove the transformation from sky_coordinate.py.\r\nInside matching.py, it checks if matchcoord is an instance of ``SkyCoord``, if it is, then we set the ``merge_attribute`` to ``False`` while calling ``transform_to``.\r\n\r\n<!-- This comments are hidden when you submit the pull request,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/master/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/master/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- If you are new or need to be re-acquainted with Astropy\r\ncontributing workflow, please see\r\nhttp://docs.astropy.org/en/latest/development/workflow/development_workflow.html .\r\nThere is even a practical example at\r\nhttps://docs.astropy.org/en/latest/development/workflow/git_edit_workflow_examples.html#astropy-fix-example . -->\r\n\r\n<!-- Astropy coding style guidelines can be found here:\r\nhttps://docs.astropy.org/en/latest/development/codeguide.html#coding-style-conventions\r\nOur testing infrastructure enforces to follow a subset of the PEP8 to be\r\nfollowed. You can check locally whether your changes have followed these by\r\nrunning the following command:\r\n\r\ntox -e codestyle\r\n\r\n-->\r\n\r\n<!-- Please just have a quick search on GitHub to see if a similar\r\npull request has already been posted.\r\nWe have old closed pull requests that might provide useful code or ideas\r\nthat directly tie in with your pull request. -->\r\n\r\n<!-- We have several automatic features that run when a pull request is open.\r\nThey can appear daunting but do not worry because maintainers will help\r\nyou navigate them, if necessary. -->\r\n\r\n### Description\r\n<!-- Provide a general description of what your pull request does.\r\nComplete the following sentence and add relevant details as you see fit. -->\r\n\r\n<!-- In addition please ensure that the pull request title is descriptive\r\nand allows maintainers to infer the applicable subpackage(s). -->\r\n\r\nThis pull request is to address #5857.\r\n\r\n<!-- If the pull request closes any open issues you can add this.\r\nIf you replace <Issue Number> with a number, GitHub will automatically link it.\r\nIf this pull request is unrelated to any issues, please remove\r\nthe following line. -->\r\n\r\nFixes #5857.\r\n",
      "issue_id": 5857,
      "issue_title": "Match catalog frames should use exact catalog frame",
      "issue_body": "In #5722 it was noted that the fact that `transform_to` overrides default frame attributes is sometimes problematic. The fix for position angle and separation is in #5762, but there @taldcroft realised that catalog matching would also be affected. It turns out the transformation to the catalog frame is done twice, both in the `SkyCoord.match*` method, and in `matching.py`. So, one has to be chosen and that one should gain a `merge_attributes=False` flag.\r\n\r\nIdeally, we still do this for 1.3.1, but not essential, so milestone can be moved if need be.",
      "issue_closed_at": "2021-04-12T19:37:50Z",
      "base_commit": "522d5cd8f5b77fd1c42f80b1d57b68ae46aef44f",
      "changes": [
        {
          "file": "astropy/coordinates/matching.py",
          "type": "line",
          "name": "line 9",
          "code": "from .representation import UnitSphericalRepresentation\nfrom astropy import units as u\nfrom . import Angle\n\n__all__ = ['match_coordinates_3d', 'match_coordinates_sky', 'search_around_3d',\n           'search_around_sky']"
        },
        {
          "file": "astropy/coordinates/matching.py",
          "type": "function",
          "name": "match_coordinates_3d",
          "class_name": null,
          "code": "def match_coordinates_3d(matchcoord, catalogcoord, nthneighbor=1, storekdtree='kdtree_3d'):\n    \"\"\"\n    Finds the nearest 3-dimensional matches of a coordinate or coordinates in\n    a set of catalog coordinates.\n\n    This finds the 3-dimensional closest neighbor, which is only different\n    from the on-sky distance if ``distance`` is set in either ``matchcoord``\n    or ``catalogcoord``.\n\n    Parameters\n    ----------\n    matchcoord : `~astropy.coordinates.BaseCoordinateFrame` or `~astropy.coordinates.SkyCoord`\n        The coordinate(s) to match to the catalog.\n    catalogcoord : `~astropy.coordinates.BaseCoordinateFrame` or `~astropy.coordinates.SkyCoord`\n        The base catalog in which to search for matches. Typically this will\n        be a coordinate object that is an array (i.e.,\n        ``catalogcoord.isscalar == False``)\n    nthneighbor : int, optional\n        Which closest neighbor to search for.  Typically ``1`` is desired here,\n        as that is correct for matching one set of coordinates to another.\n        The next likely use case is ``2``, for matching a coordinate catalog\n        against *itself* (``1`` is inappropriate because each point will find\n        itself as the closest match).\n    storekdtree : bool or str, optional\n        If a string, will store the KD-Tree used for the computation\n        in the ``catalogcoord``, as in ``catalogcoord.cache`` with the\n        provided name.  This dramatically speeds up subsequent calls with the\n        same catalog. If False, the KD-Tree is discarded after use.\n\n    Returns\n    -------\n    idx : integer array\n        Indices into ``catalogcoord`` to get the matched points for each\n        ``matchcoord``. Shape matches ``matchcoord``.\n    sep2d : `~astropy.coordinates.Angle`\n        The on-sky separation between the closest match for each ``matchcoord``\n        and the ``matchcoord``. Shape matches ``matchcoord``.\n    dist3d : `~astropy.units.Quantity`\n        The 3D distance between the closest match for each ``matchcoord`` and\n        the ``matchcoord``. Shape matches ``matchcoord``.\n\n    Notes\n    -----\n    This function requires `SciPy <https://www.scipy.org/>`_ to be installed\n    or it will fail.\n    \"\"\"\n    if catalogcoord.isscalar or len(catalogcoord) < 1:\n        raise ValueError('The catalog for coordinate matching cannot be a '\n                         'scalar or length-0.')\n\n    kdt = _get_cartesian_kdtree(catalogcoord, storekdtree)\n\n    # make sure coordinate systems match\n    matchcoord = matchcoord.transform_to(catalogcoord)\n\n    # make sure units match\n    catunit = catalogcoord.cartesian.x.unit\n    matchxyz = matchcoord.cartesian.xyz.to(catunit)\n\n    matchflatxyz = matchxyz.reshape((3, np.prod(matchxyz.shape) // 3))\n    # Querying NaN returns garbage\n    if np.isnan(matchflatxyz.value).any():\n        raise ValueError(\"Matching coordinates cannot contain NaN entries.\")\n    dist, idx = kdt.query(matchflatxyz.T, nthneighbor)\n\n    if nthneighbor > 1:  # query gives 1D arrays if k=1, 2D arrays otherwise\n        dist = dist[:, -1]\n        idx = idx[:, -1]\n\n    sep2d = catalogcoord[idx].separation(matchcoord)\n    return idx.reshape(matchxyz.shape[1:]), sep2d, dist.reshape(matchxyz.shape[1:]) * catunit"
        },
        {
          "file": "astropy/coordinates/matching.py",
          "type": "function",
          "name": "match_coordinates_sky",
          "class_name": null,
          "code": "def match_coordinates_sky(matchcoord, catalogcoord, nthneighbor=1, storekdtree='kdtree_sky'):\n    \"\"\"\n    Finds the nearest on-sky matches of a coordinate or coordinates in\n    a set of catalog coordinates.\n\n    This finds the on-sky closest neighbor, which is only different from the\n    3-dimensional match if ``distance`` is set in either ``matchcoord``\n    or ``catalogcoord``.\n\n    Parameters\n    ----------\n    matchcoord : `~astropy.coordinates.BaseCoordinateFrame` or `~astropy.coordinates.SkyCoord`\n        The coordinate(s) to match to the catalog.\n    catalogcoord : `~astropy.coordinates.BaseCoordinateFrame` or `~astropy.coordinates.SkyCoord`\n        The base catalog in which to search for matches. Typically this will\n        be a coordinate object that is an array (i.e.,\n        ``catalogcoord.isscalar == False``)\n    nthneighbor : int, optional\n        Which closest neighbor to search for.  Typically ``1`` is desired here,\n        as that is correct for matching one set of coordinates to another.\n        The next likely use case is ``2``, for matching a coordinate catalog\n        against *itself* (``1`` is inappropriate because each point will find\n        itself as the closest match).\n    storekdtree : bool or str, optional\n        If a string, will store the KD-Tree used for the computation\n        in the ``catalogcoord`` in ``catalogcoord.cache`` with the\n        provided name.  This dramatically speeds up subsequent calls with the\n        same catalog. If False, the KD-Tree is discarded after use.\n\n    Returns\n    -------\n    idx : integer array\n        Indices into ``catalogcoord`` to get the matched points for each\n        ``matchcoord``. Shape matches ``matchcoord``.\n    sep2d : `~astropy.coordinates.Angle`\n        The on-sky separation between the closest match for each\n        ``matchcoord`` and the ``matchcoord``. Shape matches ``matchcoord``.\n    dist3d : `~astropy.units.Quantity`\n        The 3D distance between the closest match for each ``matchcoord`` and\n        the ``matchcoord``. Shape matches ``matchcoord``.  If either\n        ``matchcoord`` or ``catalogcoord`` don't have a distance, this is the 3D\n        distance on the unit sphere, rather than a true distance.\n\n    Notes\n    -----\n    This function requires `SciPy <https://www.scipy.org/>`_ to be installed\n    or it will fail.\n    \"\"\"\n    if catalogcoord.isscalar or len(catalogcoord) < 1:\n        raise ValueError('The catalog for coordinate matching cannot be a '\n                         'scalar or length-0.')\n\n    # send to catalog frame\n    newmatch = matchcoord.transform_to(catalogcoord)\n\n    # strip out distance info\n    match_urepr = newmatch.data.represent_as(UnitSphericalRepresentation)\n    newmatch_u = newmatch.realize_frame(match_urepr)\n\n    cat_urepr = catalogcoord.data.represent_as(UnitSphericalRepresentation)\n    newcat_u = catalogcoord.realize_frame(cat_urepr)\n\n    # Check for a stored KD-tree on the passed-in coordinate. Normally it will\n    # have a distinct name from the \"3D\" one, so it's safe to use even though\n    # it's based on UnitSphericalRepresentation.\n    storekdtree = catalogcoord.cache.get(storekdtree, storekdtree)\n\n    idx, sep2d, sep3d = match_coordinates_3d(newmatch_u, newcat_u, nthneighbor, storekdtree)\n    # sep3d is *wrong* above, because the distance information was removed,\n    # unless one of the catalogs doesn't have a real distance\n    if not (isinstance(catalogcoord.data, UnitSphericalRepresentation) or\n            isinstance(newmatch.data, UnitSphericalRepresentation)):\n        sep3d = catalogcoord[idx].separation_3d(newmatch)\n\n    # update the kdtree on the actual passed-in coordinate\n    if isinstance(storekdtree, str):\n        catalogcoord.cache[storekdtree] = newcat_u.cache[storekdtree]\n    elif storekdtree is True:\n        # the old backwards-compatible name\n        catalogcoord.cache['kdtree'] = newcat_u.cache['kdtree']\n\n    return idx, sep2d, sep3d"
        },
        {
          "file": "astropy/coordinates/sky_coordinate.py",
          "type": "function",
          "name": "match_to_catalog_sky",
          "class_name": "SkyCoord",
          "code": "def match_to_catalog_sky(self, catalogcoord, nthneighbor=1):\n        \"\"\"\n        Finds the nearest on-sky matches of this coordinate in a set of\n        catalog coordinates.\n\n        For more on how to use this (and related) functionality, see the\n        examples in :doc:`/coordinates/matchsep`.\n\n        Parameters\n        ----------\n        catalogcoord : `~astropy.coordinates.SkyCoord` or `~astropy.coordinates.BaseCoordinateFrame`\n            The base catalog in which to search for matches. Typically this\n            will be a coordinate object that is an array (i.e.,\n            ``catalogcoord.isscalar == False``)\n        nthneighbor : int, optional\n            Which closest neighbor to search for.  Typically ``1`` is\n            desired here, as that is correct for matching one set of\n            coordinates to another. The next likely use case is ``2``,\n            for matching a coordinate catalog against *itself* (``1``\n            is inappropriate because each point will find itself as the\n            closest match).\n\n        Returns\n        -------\n        idx : integer array\n            Indices into ``catalogcoord`` to get the matched points for\n            each of this object's coordinates. Shape matches this\n            object.\n        sep2d : `~astropy.coordinates.Angle`\n            The on-sky separation between the closest match for each\n            element in this object in ``catalogcoord``. Shape matches\n            this object.\n        dist3d : `~astropy.units.Quantity`\n            The 3D distance between the closest match for each element\n            in this object in ``catalogcoord``. Shape matches this\n            object. Unless both this and ``catalogcoord`` have associated\n            distances, this quantity assumes that all sources are at a\n            distance of 1 (dimensionless).\n\n        Notes\n        -----\n        This method requires `SciPy <https://www.scipy.org/>`_ to be\n        installed or it will fail.\n\n        See Also\n        --------\n        astropy.coordinates.match_coordinates_sky\n        SkyCoord.match_to_catalog_3d\n        \"\"\"\n        from .matching import match_coordinates_sky\n\n        if (isinstance(catalogcoord, (SkyCoord, BaseCoordinateFrame))\n                and catalogcoord.has_data):\n            self_in_catalog_frame = self.transform_to(catalogcoord)\n        else:\n            raise TypeError('Can only get separation to another SkyCoord or a '\n                            'coordinate frame with data')\n\n        res = match_coordinates_sky(self_in_catalog_frame, catalogcoord,\n                                    nthneighbor=nthneighbor,\n                                    storekdtree='_kdtree_sky')\n        return res"
        },
        {
          "file": "astropy/coordinates/sky_coordinate.py",
          "type": "function",
          "name": "match_to_catalog_3d",
          "class_name": "SkyCoord",
          "code": "def match_to_catalog_3d(self, catalogcoord, nthneighbor=1):\n        \"\"\"\n        Finds the nearest 3-dimensional matches of this coordinate to a set\n        of catalog coordinates.\n\n        This finds the 3-dimensional closest neighbor, which is only different\n        from the on-sky distance if ``distance`` is set in this object or the\n        ``catalogcoord`` object.\n\n        For more on how to use this (and related) functionality, see the\n        examples in :doc:`/coordinates/matchsep`.\n\n        Parameters\n        ----------\n        catalogcoord : `~astropy.coordinates.SkyCoord` or `~astropy.coordinates.BaseCoordinateFrame`\n            The base catalog in which to search for matches. Typically this\n            will be a coordinate object that is an array (i.e.,\n            ``catalogcoord.isscalar == False``)\n        nthneighbor : int, optional\n            Which closest neighbor to search for.  Typically ``1`` is\n            desired here, as that is correct for matching one set of\n            coordinates to another.  The next likely use case is\n            ``2``, for matching a coordinate catalog against *itself*\n            (``1`` is inappropriate because each point will find\n            itself as the closest match).\n\n        Returns\n        -------\n        idx : integer array\n            Indices into ``catalogcoord`` to get the matched points for\n            each of this object's coordinates. Shape matches this\n            object.\n        sep2d : `~astropy.coordinates.Angle`\n            The on-sky separation between the closest match for each\n            element in this object in ``catalogcoord``. Shape matches\n            this object.\n        dist3d : `~astropy.units.Quantity`\n            The 3D distance between the closest match for each element\n            in this object in ``catalogcoord``. Shape matches this\n            object.\n\n        Notes\n        -----\n        This method requires `SciPy <https://www.scipy.org/>`_ to be\n        installed or it will fail.\n\n        See Also\n        --------\n        astropy.coordinates.match_coordinates_3d\n        SkyCoord.match_to_catalog_sky\n        \"\"\"\n        from .matching import match_coordinates_3d\n\n        if (isinstance(catalogcoord, (SkyCoord, BaseCoordinateFrame))\n                and catalogcoord.has_data):\n            self_in_catalog_frame = self.transform_to(catalogcoord)\n        else:\n            raise TypeError('Can only get separation to another SkyCoord or a '\n                            'coordinate frame with data')\n\n        res = match_coordinates_3d(self_in_catalog_frame, catalogcoord,\n                                   nthneighbor=nthneighbor,\n                                   storekdtree='_kdtree_3d')\n\n        return res"
        }
      ]
    }
  ]
}