{
  "id": "astropy__astropy-14096",
  "question": "Subclassed SkyCoord gives misleading attribute access message\nI'm trying to subclass `SkyCoord`, and add some custom properties. This all seems to be working fine, but when I have a custom property (`prop` below) that tries to access a non-existent attribute (`random_attr`) below, the error message is misleading because it says `prop` doesn't exist, where it should say `random_attr` doesn't exist.\r\n\r\n```python\r\nimport astropy.coordinates as coord\r\n\r\n\r\nclass custom_coord(coord.SkyCoord):\r\n    @property\r\n    def prop(self):\r\n        return self.random_attr\r\n\r\n\r\nc = custom_coord('00h42m30s', '+41d12m00s', frame='icrs')\r\nc.prop\r\n```\r\n\r\nraises\r\n```\r\nTraceback (most recent call last):\r\n  File \"test.py\", line 11, in <module>\r\n    c.prop\r\n  File \"/Users/dstansby/miniconda3/lib/python3.7/site-packages/astropy/coordinates/sky_coordinate.py\", line 600, in __getattr__\r\n    .format(self.__class__.__name__, attr))\r\nAttributeError: 'custom_coord' object has no attribute 'prop'\r\n```\n",
  "answer": "{\"task_id\": \"92bed3b9-e194-4948-9cc2-bf055575aaf5\", \"workflow_run_id\": \"edf3f1d0-37ff-4cda-a4ef-f08a8408fd2e\", \"data\": {\"id\": \"edf3f1d0-37ff-4cda-a4ef-f08a8408fd2e\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -592,6 +592,15 @@ class SkyCoord(ShapedLikeNDArray):\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n+            \\n+            # Check if the attribute exists as a descriptor/property on the class\\n+            for cls in type(self).__mro__:\\n+                if attr in cls.__dict__:\\n+                    descriptor = cls.__dict__[attr]\\n+                    if hasattr(descriptor, '__get__'):\\n+                        # It's a descriptor (property), let the original AttributeError propagate\\n+                        _currently_getting_frame_attr = False\\n+                        raise\\n             try:\\n                 return getattr(self._sky_coord_frame, attr)\\n             except AttributeError:\", \"--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -592,11 +592,18 @@ class SkyCoord(ShapedLikeNDArray):\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n+            original_exception = None\\n             try:\\n                 return getattr(self._sky_coord_frame, attr)\\n-            except AttributeError:\\n+            except AttributeError as e:\\n+                # Store the frame AttributeError\\n+                original_exception = e\\n                 pass\\n             finally:\\n                 _currently_getting_frame_attr = False\\n+            \\n+            # If we have an original exception from frame access, it means the attribute\\n+            # doesn't exist on the frame either, so raise our custom message\\n+            if original_exception is not None:\\n+                pass  # Continue to raise the custom AttributeError below\\n \\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\", \"--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,7 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        import sys\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n@@ -597,6 +598,16 @@ class SkyCoord(ShapedLikeNDArray):\\n             finally:\\n                 _currently_getting_frame_attr = False\\n \\n+        # Check if this AttributeError came from within a property/descriptor\\n+        # by examining the current exception context\\n+        exc_info = sys.exc_info()\\n+        if exc_info[0] is AttributeError and exc_info[2] is not None:\\n+            # Walk up the traceback to see if we're inside a property getter\\n+            tb = exc_info[2]\\n+            while tb.tb_next is not None:\\n+                tb = tb.tb_next\\n+            if tb.tb_frame.f_code.co_name in ('__get__', 'fget') or 'property' in str(tb.tb_frame.f_code):\\n+                raise exc_info[1].with_traceback(exc_info[2])\\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\", \"--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,14 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        \\n+        # First check if this attribute exists in our class hierarchy\\n+        # If it does, then any AttributeError should be from within that attribute\\n+        for cls in type(self).__mro__:\\n+            if hasattr(cls, attr):\\n+                # Attribute exists in class, so AttributeError is from within it\\n+                # Let the original AttributeError propagate\\n+                raise\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\", \"--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,7 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        import sys\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n@@ -597,6 +598,15 @@ class SkyCoord(ShapedLikeNDArray):\\n             finally:\\n                 _currently_getting_frame_attr = False\\n \\n+        # Check if we're handling an AttributeError that came from a property\\n+        # by looking at the current exception context\\n+        current_exception = sys.exc_info()[1]\\n+        if (isinstance(current_exception, AttributeError) and \\n+            current_exception.args and \\n+            attr not in str(current_exception.args[0])):\\n+            # The AttributeError is about a different attribute (from within a property)\\n+            # Re-raise the original exception\\n+            raise\\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\", \"diff\\n--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,7 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        import sys\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n@@ -597,6 +598,13 @@ class SkyCoord(ShapedLikeNDArray):\\n             finally:\\n                 _currently_getting_frame_attr = False\\n \\n+        # Check if we're currently handling an AttributeError from within a property\\n+        # If so, let the original error propagate instead of masking it\\n+        exc_info = sys.exc_info()\\n+        if (exc_info[0] is AttributeError and exc_info[1] is not None and \\n+            hasattr(type(self), attr)):\\n+            # Re-raise the original AttributeError\\n+            raise exc_info[1].with_traceback(exc_info[2])\\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\\n\", \"diff\\n--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,7 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        import sys\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n@@ -597,6 +598,15 @@ class SkyCoord(ShapedLikeNDArray):\\n             finally:\\n                 _currently_getting_frame_attr = False\\n \\n+        # Check if we're currently handling an AttributeError that originated\\n+        # from within a property/descriptor of this class\\n+        exc_info = sys.exc_info()\\n+        if (exc_info[0] is AttributeError and exc_info[1] is not None and \\n+            hasattr(type(self), attr)):\\n+            # The attribute exists in our class hierarchy, so the AttributeError\\n+            # must have come from within that attribute's getter. Re-raise it.\\n+            raise exc_info[1].with_traceback(exc_info[2])\\n+            \\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\\n\", \"diff\\n--- a/astropy/coordinates/sky_coordinate.py\\n+++ b/astropy/coordinates/sky_coordinate.py\\n@@ -587,6 +587,7 @@ class SkyCoord(ShapedLikeNDArray):\\n \\n     def __getattr__(self, attr):\\n         # See if we can get the attribute from the frame\\n+        import sys\\n         # Only do this if we're not already getting from the frame.\\n         if not _currently_getting_frame_attr:\\n             _currently_getting_frame_attr = True\\n@@ -597,6 +598,15 @@ class SkyCoord(ShapedLikeNDArray):\\n             finally:\\n                 _currently_getting_frame_attr = False\\n \\n+        # Check if we're currently handling an AttributeError that originated\\n+        # from within a property/descriptor of this class\\n+        exc_info = sys.exc_info()\\n+        if (exc_info[0] is AttributeError and exc_info[1] is not None and \\n+            hasattr(type(self), attr)):\\n+            # The attribute exists in our class hierarchy, so the AttributeError\\n+            # must have come from within that attribute's getter. Re-raise it.\\n+            raise exc_info[1].with_traceback(exc_info[2])\\n+            \\n         raise AttributeError(\\\"'{0}' object has no attribute '{1}'\\\"\\n                              .format(self.__class__.__name__, attr))\\n\"]}, \"error\": \"\", \"elapsed_time\": 132.17744, \"total_tokens\": 40492, \"total_steps\": 30, \"created_at\": 1754644697, \"finished_at\": 1754644829}}"
}