{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful bug report overall because it addresses similar functionality within the `nddata` module where the CURRENT bug is localized. The nature of the mask handling failing in the case of `NDData` initialization due to `None` values aligns closely with the issue description in the CURRENT bug. This shared context and focus on mask management could provide crucial insights and relevant patches to resolve the ongoing issue.",
  "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"
}