{
  "id": "astropy__astropy-7166",
  "question": "InheritDocstrings metaclass doesn't work for properties\nInside the InheritDocstrings metaclass it uses `inspect.isfunction` which returns `False` for properties.\n",
  "answer": "{\"task_id\": \"24ae09c0-2610-438f-a657-1b2107da9440\", \"workflow_run_id\": \"79e686a8-f4ec-4837-9ee8-37af07269324\", \"data\": {\"id\": \"79e686a8-f4ec-4837-9ee8-37af07269324\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,7 +250,7 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n             if (inspect.isfunction(val) and\\n                 getattr(val, '__doc__', None) is None):\\n-                for base in bases:\\n+                _inherit_function_docstring(val, key, bases)\\n+            elif isinstance(val, property) and getattr(val, '__doc__', None) is None:\\n+                _inherit_property_docstring(val, key, bases, dct)\\n+\\n+        def _inherit_function_docstring(func, key, bases):\\n+            for base in bases:\\n+                if hasattr(base, key):\\n+                    base_func = getattr(base, key)\\n+                    if inspect.isfunction(base_func) and base_func.__doc__:\\n+                        func.__doc__ = base_func.__doc__\\n+                        break\\n+\\n+        def _inherit_property_docstring(prop, key, bases, dct):\\n+            for base in bases:\\n+                if hasattr(base, key):\\n+                    base_attr = getattr(base, key)\\n+                    if isinstance(base_attr, property) and base_attr.__doc__:\\n+                        # Create a new property with inherited docstring\\n+                        dct[key] = property(prop.fget, prop.fset, prop.fdel, base_attr.__doc__)\\n+                        break\", \"--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,7 +250,8 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n-                getattr(val, '__doc__', None) is None):\\n+            if ((inspect.isfunction(val) or isinstance(val, property)) and\\n+                getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n+                        if ((inspect.isfunction(base_func) or isinstance(base_func, property)) and \\n+                            base_func.__doc__):\\n+                            if isinstance(val, property):\\n+                                # Replace property with new one that has inherited docstring\\n+                                dct[key] = property(val.fget, val.fset, val.fdel, base_func.__doc__)\\n+                            else:\\n+                                val.__doc__ = base_func.__doc__\\n                             break\", \"--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,7 +250,7 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n             if (inspect.isfunction(val) and\\n                 getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n                         if inspect.isfunction(base_func) and base_func.__doc__:\\n                             val.__doc__ = base_func.__doc__\\n                             break\\n+            elif isinstance(val, property) and getattr(val, '__doc__', None) is None:\\n+                for base in bases:\\n+                    if hasattr(base, key):\\n+                        base_attr = getattr(base, key)\\n+                        if isinstance(base_attr, property) and base_attr.__doc__:\\n+                            # Create new property with inherited docstring\\n+                            dct[key] = property(\\n+                                fget=val.fget,\\n+                                fset=val.fset, \\n+                                fdel=val.fdel,\\n+                                doc=base_attr.__doc__\\n+                            )\\n+                            break\", \"--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -247,6 +247,23 @@ class InheritDocstrings(type):\\n     def __new__(cls, name, bases, dct):\\n         # Loop over attributes and see if they are functions that\\n         # don't have docstrings\\n+        \\n+        def _should_inherit_docstring(attr):\\n+            \\\"\\\"\\\"Check if attribute should inherit docstring.\\\"\\\"\\\"\\n+            return ((inspect.isfunction(attr) or isinstance(attr, property)) and\\n+                    getattr(attr, '__doc__', None) is None)\\n+        \\n+        def _inherit_docstring(attr, key, bases, dct):\\n+            \\\"\\\"\\\"Inherit docstring from base classes.\\\"\\\"\\\"\\n+            for base in bases:\\n+                if hasattr(base, key):\\n+                    base_attr = getattr(base, key)\\n+                    if base_attr.__doc__:\\n+                        if isinstance(attr, property) and isinstance(base_attr, property):\\n+                            dct[key] = property(attr.fget, attr.fset, attr.fdel, base_attr.__doc__)\\n+                        elif inspect.isfunction(attr) and inspect.isfunction(base_attr):\\n+                            attr.__doc__ = base_attr.__doc__\\n+                        break\\n+\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n-                getattr(val, '__doc__', None) is None):\\n-                for base in bases:\\n-                    if hasattr(base, key):\\n-                        base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n-                            break\\n+            if _should_inherit_docstring(val):\\n+                _inherit_docstring(val, key, bases, dct)\", \"--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,7 +250,8 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n+            if ((inspect.isfunction(val) or isinstance(val, property)) and\\n                 getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n-                            break\\n+                        if base_func.__doc__:\\n+                            if isinstance(val, property) and isinstance(base_func, property):\\n+                                dct[key] = property(val.fget, val.fset, val.fdel, base_func.__doc__)\\n+                                break\\n+                            elif inspect.isfunction(val) and inspect.isfunction(base_func):\\n+                                val.__doc__ = base_func.__doc__\\n+                                break\", \"diff\\n--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -247,13 +247,18 @@ class InheritDocstrings(type):\\n     def __new__(cls, name, bases, dct):\\n         # Loop over attributes and see if they are functions that\\n         # don't have docstrings\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n+            if ((inspect.isfunction(val) or isinstance(val, property)) and\\n                 getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n-                            break\\n+                        if base_func.__doc__:\\n+                            if isinstance(val, property) and isinstance(base_func, property):\\n+                                # Create new property with inherited docstring\\n+                                dct[key] = property(val.fget, val.fset, val.fdel, base_func.__doc__)\\n+                                break\\n+                            elif inspect.isfunction(val) and inspect.isfunction(base_func):\\n+                                val.__doc__ = base_func.__doc__\\n+                                break\\n         return super(InheritDocstrings, cls).__new__(cls, name, bases, dct)\\n\", \"diff\\n--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,11 +250,15 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n+            if ((inspect.isfunction(val) or isinstance(val, property)) and\\n                 getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n-                            break\\n+                        if base_func.__doc__:\\n+                            if isinstance(val, property) and isinstance(base_func, property):\\n+                                # Create new property with inherited docstring\\n+                                dct[key] = property(val.fget, val.fset, val.fdel, base_func.__doc__)\\n+                                break\\n+                            elif inspect.isfunction(val) and inspect.isfunction(base_func):\\n+                                val.__doc__ = base_func.__doc__\\n+                                break\\n         return super(InheritDocstrings, cls).__new__(cls, name, bases, dct)\\n\", \"diff\\n--- a/astropy/utils/misc.py\\n+++ b/astropy/utils/misc.py\\n@@ -250,11 +250,15 @@ class InheritDocstrings(type):\\n         for key, val in dct.items():\\n-            if (inspect.isfunction(val) and\\n+            if ((inspect.isfunction(val) or isinstance(val, property)) and\\n                 getattr(val, '__doc__', None) is None):\\n                 for base in bases:\\n                     if hasattr(base, key):\\n                         base_func = getattr(base, key)\\n-                        if inspect.isfunction(base_func) and base_func.__doc__:\\n-                            val.__doc__ = base_func.__doc__\\n-                            break\\n+                        if base_func.__doc__:\\n+                            if isinstance(val, property) and isinstance(base_func, property):\\n+                                # Create new property with inherited docstring\\n+                                dct[key] = property(val.fget, val.fset, val.fdel, base_func.__doc__)\\n+                                break\\n+                            elif inspect.isfunction(val) and inspect.isfunction(base_func):\\n+                                val.__doc__ = base_func.__doc__\\n+                                break\\n         return super(InheritDocstrings, cls).__new__(cls, name, bases, dct)\\n\"]}, \"error\": \"\", \"elapsed_time\": 109.444167, \"total_tokens\": 39717, \"total_steps\": 30, \"created_at\": 1754645804, \"finished_at\": 1754645913}}"
}