{
  "id": "astropy__astropy-14995",
  "question": "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",
  "answer": "{\"task_id\": \"52eb7a13-259e-4af2-8687-b5118c434d70\", \"workflow_run_id\": \"ff6f9ebc-6ff6-473c-89ed-5de01cbb5bd8\", \"data\": {\"id\": \"ff6f9ebc-6ff6-473c-89ed-5de01cbb5bd8\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,20 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Handle cases where one or both masks are None\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        elif mask1 is None:\\n+            # If mask1 is None, create a zero mask with the same shape as mask2\\n+            mask1 = np.zeros_like(mask2, dtype=bool)\\n+        elif mask2 is None:\\n+            # If mask2 is None, create a zero mask with the same shape as mask1\\n+            mask2 = np.zeros_like(mask1, dtype=bool)\\n+        \\n+        # Ensure masks are boolean arrays for bitwise operations\\n+        mask1 = np.asarray(mask1, dtype=bool)\\n+        mask2 = np.asarray(mask2, dtype=bool)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\", \"--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,16 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Handle None masks before applying the mask function\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        elif mask1 is None:\\n+            return deepcopy(mask2)\\n+        elif mask2 is None:\\n+            return deepcopy(mask1)\\n+        \\n+        # Both masks are not None, proceed with normal handling\\n+        \\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\", \"--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,25 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Early return for cases with None masks\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        \\n+        # If one mask is None, we need to handle it based on the operation\\n+        if mask1 is None or mask2 is None:\\n+            non_none_mask = mask1 if mask1 is not None else mask2\\n+            \\n+            # For bitwise operations, create a zero mask for the None operand\\n+            if handle_mask in (np.bitwise_or, np.bitwise_and, np.bitwise_xor):\\n+                zero_mask = np.zeros_like(non_none_mask, dtype=bool)\\n+                if mask1 is None:\\n+                    mask1 = zero_mask\\n+                else:\\n+                    mask2 = zero_mask\\n+            else:\\n+                # For other operations, just return the existing mask\\n+                return deepcopy(non_none_mask)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\", \"--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -300,6 +300,18 @@ class NDArithmeticMixin:\\n         # Determine the output mask\\n         if handle_mask is not None:\\n             if mask1 is not None or mask2 is not None:\\n+                # Normalize None masks to zero arrays for consistent handling\\n+                if mask1 is None and mask2 is not None:\\n+                    mask1 = np.zeros_like(mask2, dtype=bool)\\n+                elif mask2 is None and mask1 is not None:\\n+                    mask2 = np.zeros_like(mask1, dtype=bool)\\n+                \\n+                # Ensure masks are boolean arrays for bitwise operations\\n+                if mask1 is not None:\\n+                    mask1 = np.asarray(mask1, dtype=bool)\\n+                if mask2 is not None:\\n+                    mask2 = np.asarray(mask2, dtype=bool)\\n+                    \\n                 result_mask = self._combine_masks(mask1, mask2, handle_mask)\\n             else:\\n                 result_mask = None\", \"--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,18 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Special handling for bitwise operations with None masks\\n+        if handle_mask in (np.bitwise_or, np.bitwise_and, np.bitwise_xor):\\n+            if mask1 is None and mask2 is None:\\n+                return None\\n+            elif mask1 is None:\\n+                return deepcopy(mask2)\\n+            elif mask2 is None:\\n+                return deepcopy(mask1)\\n+            # Convert to boolean arrays to ensure bitwise operations work\\n+            mask1 = np.asarray(mask1, dtype=bool)\\n+            mask2 = np.asarray(mask2, dtype=bool)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\", \"diff\\n--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,25 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Early return for cases with None masks\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        \\n+        # If one mask is None, we need to handle it based on the operation\\n+        if mask1 is None or mask2 is None:\\n+            non_none_mask = mask1 if mask1 is not None else mask2\\n+            \\n+            # For bitwise operations, create a zero mask for the None operand\\n+            if handle_mask in (np.bitwise_or, np.bitwise_and, np.bitwise_xor):\\n+                zero_mask = np.zeros_like(non_none_mask, dtype=bool)\\n+                if mask1 is None:\\n+                    mask1 = zero_mask\\n+                else:\\n+                    mask2 = zero_mask\\n+            else:\\n+                # For other operations, just return the existing mask\\n+                return deepcopy(non_none_mask)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\\n\", \"diff\\n--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,18 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Handle None mask cases early to avoid type errors\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        elif mask1 is None:\\n+            return deepcopy(mask2)\\n+        elif mask2 is None:\\n+            return deepcopy(mask1)\\n+        \\n+        # Ensure masks are boolean arrays for bitwise operations\\n+        mask1 = np.asarray(mask1, dtype=bool)\\n+        mask2 = np.asarray(mask2, dtype=bool)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\\n\", \"diff\\n--- a/astropy/nddata/mixins/arithmetics.py\\n+++ b/astropy/nddata/mixins/arithmetics.py\\n@@ -450,6 +450,18 @@ class NDArithmeticMixin:\\n         if handle_mask is None:\\n             return None\\n \\n+        # Handle None mask cases early to avoid type errors\\n+        if mask1 is None and mask2 is None:\\n+            return None\\n+        elif mask1 is None:\\n+            return deepcopy(mask2)\\n+        elif mask2 is None:\\n+            return deepcopy(mask1)\\n+        \\n+        # Ensure masks are boolean arrays for bitwise operations\\n+        mask1 = np.asarray(mask1, dtype=bool)\\n+        mask2 = np.asarray(mask2, dtype=bool)\\n+\\n         # If only one mask is present we need not bother about any type checks\\n         if mask1 is None:\\n             return deepcopy(mask2)\\n\"]}, \"error\": \"\", \"elapsed_time\": 119.794573, \"total_tokens\": 45838, \"total_steps\": 30, \"created_at\": 1754645684, \"finished_at\": 1754645804}}"
}