{
  "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\": \"9b3155ca-c3e1-445b-8065-256f9b8eece7\", \"workflow_run_id\": \"f5dea8a6-6b5c-4d02-b5c7-18386eb82597\", \"data\": {\"id\": \"f5dea8a6-6b5c-4d02-b5c7-18386eb82597\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/astropy/nddata/mixins/ndarithmetic.py b/astropy/nddata/mixins/ndarithmetic.py\\\\nindex 1234567..abcdef0 100644\\\\n--- a/astropy/nddata/mixins/ndarithmetic.py\\\\n+++ b/astropy/nddata/mixins/ndarithmetic.py\\\\n@@ -230,14 +230,15 @@ class NDArithmeticMixin:\\\\n     def _propagate_mask(self, operand1, operand2, result, handle_mask):\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"Propagate mask using the given function, if present\\\\\\\"\\\\\\\"\\\\\\\"\\\\n         mask1 = getattr(operand1, 'mask', None)\\\\n         mask2 = getattr(operand2, 'mask', None)\\\\n \\\\n+        if mask1 is None and mask2 is not None:\\\\n+            result.mask = mask2\\\\n+        elif mask2 is None and mask1 is not None:\\\\n+            result.mask = mask1\\\\n+        elif handle_mask is not None:\\\\n+            result.mask = handle_mask(mask1, mask2)\\\\n+\\\\n         # If handle_mask is None, assume no mask should be applied\\\\n         result.mask = mask1 if mask2 is None else mask2\\\\n-        if handle_mask is not None:\\\\n-            # Combine masks, assuming they can be combined via logical_or\\\\n-            result.mask = handle_mask(mask1, mask2)\\\\n\\\\n         # If none of the operands have a mask, no mask is applied to the result\\\\n-        if mask1 is None and mask2 is None:\\\\n-            result.mask = None\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch directly addresses the issue by checking if one operand has a mask and the other does not, and assigns the existing mask to the result. It avoids unnecessary operations and aligns well with the expected behavior. The risk of introducing new bugs is minimal.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/astropy/nddata/mixins/ndarithmetic.py b/astropy/nddata/mixins/ndarithmetic.py\\\\nindex 2345678..abcdef1 100644\\\\n--- a/astropy/nddata/mixins/ndarithmetic.py\\\\n+++ b/astropy/nddata/mixins/ndarithmetic.py\\\\n@@ -230,9 +230,13 @@ class NDArithmeticMixin:\\\\n     def _propagate_mask(self, operand1, operand2, result, handle_mask):\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"Propagate mask using the given function, if present\\\\\\\"\\\\\\\"\\\\\\\"\\\\n         mask1 = getattr(operand1, 'mask', None)\\\\n         mask2 = getattr(operand2, 'mask', None)\\\\n \\\\n-        # Combine masks, assuming they can be combined via logical_or\\\\n-        if handle_mask is not None:\\\\n-            result.mask = handle_mask(mask1, mask2)\\\\n+        if handle_mask is not None:\\\\n+            if mask1 is None and mask2 is not None:\\\\n+                result.mask = mask2\\\\n+            elif mask2 is None and mask1 is not None:\\\\n+                result.mask = mask1\\\\n+            else:\\\\n+                result.mask = handle_mask(mask1, mask2)\\\\n\\\\n         # If handle_mask is None, assume no mask should be applied\\\\n         result.mask = mask1 if mask2 is None else mask2\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue by checking for the presence of masks before applying the handle_mask function. It is slightly more complex than Patch 2 but still effective. The risk is slightly higher due to the added complexity.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/astropy/nddata/mixins/ndarithmetic.py b/astropy/nddata/mixins/ndarithmetic.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/astropy/nddata/mixins/ndarithmetic.py\\\\n+++ b/astropy/nddata/mixins/ndarithmetic.py\\\\n@@ -230,10 +230,12 @@ class NDArithmeticMixin:\\\\n     def _propagate_mask(self, operand1, operand2, result, handle_mask):\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"Propagate mask using the given function, if present\\\\\\\"\\\\\\\"\\\\\\\"\\\\n         mask1 = getattr(operand1, 'mask', None)\\\\n         mask2 = getattr(operand2, 'mask', None)\\\\n+        # Treat None masks as arrays of zeros\\\\n+        mask1 = np.zeros_like(result) if mask1 is None else mask1\\\\n+        mask2 = np.zeros_like(result) if mask2 is None else mask2\\\\n \\\\n         if handle_mask is not None:\\\\n-            # Combine masks, assuming they can be combined via logical_or\\\\n-            result.mask = handle_mask(mask1, mask2)\\\\n+            result.mask = handle_mask(mask1, mask2)\\\\n\\\\n         # If handle_mask is None, assume no mask should be applied\\\\n         result.mask = mask1 if mask2 is None else mask2\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch treats None masks as arrays of zeros, which may not be the most efficient or intuitive solution. It introduces a higher risk of unintended side effects, especially if zero is not an appropriate default value for masks.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the most effective solution, directly addressing the issue with minimal complexity and risk. Patch 3 is also a viable option but introduces slightly more complexity. Patch 1, while functional, uses a less intuitive approach that could lead to unintended side effects.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    import numpy as np\\\\n    from astropy.nddata import NDDataRef\\\\n\\\\n    array = np.array([[0, 1, 0], [1, 0, 1], [0, 1, 0]])\\\\n    mask = np.array([[0, 1, 64], [8, 0, 1], [2, 1, 0]])\\\\n\\\\n    nref_nomask = NDDataRef(array)\\\\n    nref_mask = NDDataRef(array, mask=mask)\\\\n\\\\n    try:\\\\n        nref_mask.multiply(1., handle_mask=np.bitwise_or).mask\\\\n    except TypeError:\\\\n        assert True\\\\n    else:\\\\n        assert False\\\\n\\\\n    try:\\\\n        nref_mask.multiply(nref_nomask, handle_mask=np.bitwise_or).mask\\\\n    except TypeError:\\\\n        assert True\\\\n    else:\\\\n        assert False\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue where mask propagation fails when one of the operands does not have a mask\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a TypeError due to unsupported operand type(s) for bitwise_or operation\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    import numpy as np\\\\n    from astropy.nddata import NDDataRef\\\\n\\\\n    array = np.array([[0, 1], [1, 0]])\\\\n    mask = np.array([[0, 1], [1, 0]])\\\\n\\\\n    nref_nomask = NDDataRef(array)\\\\n    nref_mask = NDDataRef(array, mask=mask)\\\\n\\\\n    # Test when both operands have no mask\\\\n    assert nref_nomask.multiply(2., handle_mask=np.bitwise_or).mask is None\\\\n\\\\n    # Test when one operand has a mask and the other does not\\\\n    assert np.array_equal(nref_mask.multiply(3., handle_mask=np.bitwise_or).mask, mask)\\\",\\n      \\\"description\\\": \\\"This test covers edge cases where both operands have no mask and when one operand has a mask while the other does not\\\",\\n      \\\"expected_behavior\\\": \\\"The tests should pass without raising any errors\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    import numpy as np\\\\n    from astropy.nddata import NDDataRef\\\\n\\\\n    array = np.array([[0, 1, 0], [1, 0, 1], [0, 1, 0]])\\\\n    mask = np.array([[0, 1, 64], [8, 0, 1], [2, 1, 0]])\\\\n\\\\n    nref_nomask = NDDataRef(array)\\\\n    nref_mask = NDDataRef(array, mask=mask)\\\\n\\\\n    # Apply Patch 3\\\\n    result_mask = nref_mask.multiply(1., handle_mask=np.bitwise_or).mask\\\\n    expected_result_mask = np.array([[ 0,  1, 64], [ 8,  0, 1], [ 2, 1, 0]])\\\\n    assert np.array_equal(result_mask, expected_result_mask)\\\\n\\\\n    result_no_mask = nref_nomask.multiply(nref_mask, handle_mask=np.bitwise_or).mask\\\\n    expected_result_no_mask = np.array([[ 0,  1, 64], [ 8,  0, 1], [ 2, 1, 0]])\\\\n    assert np.array_equal(result_no_mask, expected_result_no_mask)\\\",\\n      \\\"description\\\": \\\"This test validates that the patches work correctly by checking the mask propagation results after applying the patches\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated comprehensive test cases to reproduce the original issue, cover edge cases, and validate the patches for mask propagation in NDDataRef\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 404.933257, \"total_tokens\": 26837, \"total_steps\": 9, \"created_at\": 1753280136, \"finished_at\": 1753280541}}"
}