{
  "instance_id": "astropy__astropy-12907",
  "repo": "astropy/astropy",
  "created_at": "2022-03-03T15:14:54Z",
  "problem_statement": "Modeling's `separability_matrix` does not compute separability correctly for nested CompoundModels\nConsider the following model:\r\n\r\n```python\r\nfrom astropy.modeling import models as m\r\nfrom astropy.modeling.separable import separability_matrix\r\n\r\ncm = m.Linear1D(10) & m.Linear1D(5)\r\n```\r\n\r\nIt's separability matrix as you might expect is a diagonal:\r\n\r\n```python\r\n>>> separability_matrix(cm)\r\narray([[ True, False],\r\n       [False,  True]])\r\n```\r\n\r\nIf I make the model more complex:\r\n```python\r\n>>> separability_matrix(m.Pix2Sky_TAN() & m.Linear1D(10) & m.Linear1D(5))\r\narray([[ True,  True, False, False],\r\n       [ True,  True, False, False],\r\n       [False, False,  True, False],\r\n       [False, False, False,  True]])\r\n```\r\n\r\nThe output matrix is again, as expected, the outputs and inputs to the linear models are separable and independent of each other.\r\n\r\nIf however, I nest these compound models:\r\n```python\r\n>>> separability_matrix(m.Pix2Sky_TAN() & cm)\r\narray([[ True,  True, False, False],\r\n       [ True,  True, False, False],\r\n       [False, False,  True,  True],\r\n       [False, False,  True,  True]])\r\n```\r\nSuddenly the inputs and outputs are no longer separable?\r\n\r\nThis feels like a bug to me, but I might be missing something?\n",
  "patch": "diff --git a/astropy/modeling/separable.py b/astropy/modeling/separable.py\n--- a/astropy/modeling/separable.py\n+++ b/astropy/modeling/separable.py\n@@ -242,7 +242,7 @@ def _cstack(left, right):\n         cright = _coord_matrix(right, 'right', noutp)\n     else:\n         cright = np.zeros((noutp, right.shape[1]))\n-        cright[-right.shape[0]:, -right.shape[1]:] = 1\n+        cright[-right.shape[0]:, -right.shape[1]:] = right\n \n     return np.hstack([cleft, cright])\n \n",
  "similar_bug_items": [
    {
      "pr_number": 10094,
      "pr_title": "Apply space motion of SkyCoords for rv corrections",
      "pr_body": "<!-- This comments are hidden when you submit the pull request,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/master/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/master/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- If you are new or need to be re-acquainted with Astropy\r\ncontributing workflow, please see\r\nhttp://docs.astropy.org/en/latest/development/workflow/development_workflow.html .\r\nThere is even a practical example at\r\nhttps://docs.astropy.org/en/latest/development/workflow/git_edit_workflow_examples.html#astropy-fix-example . -->\r\n\r\n<!-- Astropy coding style guidelines can be found here:\r\nhttps://docs.astropy.org/en/latest/development/codeguide.html#coding-style-conventions\r\nOur testing infrastructure enforces to follow a subset of the PEP8 to be\r\nfollowed. You can check locally whether your changes have followed these by\r\nrunning the following command:\r\n\r\ntox -e codestyle\r\n\r\n-->\r\n\r\n<!-- Please just have a quick search on GitHub to see if a similar\r\npull request has already been posted.\r\nWe have old closed pull requests that might provide useful code or ideas\r\nthat directly tie in with your pull request. -->\r\n\r\n<!-- We have several automatic features that run when a pull request is open.\r\nThey can appear daunting but do not worry because maintainers will help\r\nyou navigate them, if necessary. -->\r\n\r\n### Description\r\n<!-- Provide a general description of what your pull request does.\r\nComplete the following sentence and add relevant details as you see fit. -->\r\n\r\nNow that we no longer throw an Exception when ```SkyCoord```s with velocity information are passed to ```SkyCoord.radial_velocity_correction``` we have to address a bug that is unfortunately baked into the API.\r\n\r\nCurrently if the passed ```SkyCoord``` had it's own obstime, and it was inconsistent with the passed obstime, we raised an Exception. However, the correct thing to do is to correct the ```SkyCoord``` for it's space motion so that it points to the proper location at the *passed* obstime. \r\n\r\nFailure to do so for nearby or fast moving objects (e.g tau Ceti below) leads to errors of metres/second:\r\n\r\n![RVs_before](https://user-images.githubusercontent.com/4570807/78016482-c518e400-7342-11ea-9833-93ecfe50bba3.png)\r\n\r\n<!-- If the pull request closes any open issues you can add this.\r\nIf you replace <Issue Number> with a number, GitHub will automatically link it.\r\nIf this pull request is unrelated to any issues, please remove\r\nthe following line. -->\r\n\r\nThis PR fixes #9979 by updating the object's position to the passed obstime, resulting in better than 10 mm/s agreement with TEMPO2 or BARYCORR.\r\n\r\n![RVs_after](https://user-images.githubusercontent.com/4570807/78016666-04dfcb80-7343-11ea-84b9-715124af3ea5.png)\r\n\r\nUnfortunately I don't think this can be considered a bug fix, since code that used to raise an exception will now run. \r\n\r\nNote that the test I have added for regression downloads the TEMPO2/BARYCORR corrections for tau Ceti from an external website. I have submitted a seperate PR (https://github.com/astropy/astropy-data/pull/88)  to astropy-data to host those corrections there, and then update this PR accordingly, so perhaps hold off until then to merge.",
      "issue_id": 9979,
      "issue_title": "radial_velocity_correction needs more careful handling of obstime for objects with high proper motion",
      "issue_body": "<!-- This comments are hidden when you submit the issue,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/master/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/master/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- Please have a search on our GitHub repository to see if a similar\r\nissue has already been posted.\r\nIf a similar issue is closed, have a quick look to see if you are satisfied\r\nby the resolution.\r\nIf not please go ahead and open an issue! -->\r\n\r\n<!-- Please check that the development version still produces the same bug.\r\nYou can install development version with\r\npip install git+https://github.com/astropy/astropy\r\ncommand. -->\r\n\r\n### Description\r\n<!-- Provide a general description of the bug. -->\r\nWhilst working on #9645 I noticed a subtlety in the radial velocity correction that is important at the metres/second level for objects with high proper motion (arcsecs/year).\r\n\r\nThe issue is that, in order to calculate the RV correction one needs to know the ICRS position of the source *at the time of observation*. To do this, one needs to apply space motion of the ```SkyCoord``` itself, in order to find the position of the source at the observation time. However, this requires knowing the epoch of the coordinates. \r\n\r\nAt the moment, we apply no space motion - i.e we assume the epoch of the coordinates is equal to the RV observation time. We need a way of knowing both the ```obstime``` of the RV measurement and the epoch of the coordinates.  However, at the moment, only one Time is allowed for the whole calculation, which is the time of observation of the radial velocity. In fact, if the user tries to fix this and supplies ```obstime``` argument when the ```SkyCoord``` already has an obstime set then we raise an error.\r\n\r\n### Expected behavior\r\n<!-- What did you expect to happen. -->\r\nWhat I think should happen is this:\r\n\r\n##### Case 1 (User supplies a single ```obstime``` argument, ```SkyCoord.obstime``` is set):\r\nAssume ```SkyCoord.obstime``` is the epoch of the coordinates, apply space motion to find position of source at ```obstime```.\r\n\r\n##### Case 2 (User supplies no ```obstime``` but ```SkyCoord.obstime``` is set):\r\nAssume epoch and RV observation time are the same. Do not apply space motion.\r\n\r\n##### Case 3 (User supplies ```obstime``` but ```SkyCoord.obstime``` is not set):\r\nThis is the only awkward case. We could assume the epoch is equal to the equinox and apply space motion, or we could assume case 2 and apply no space motion. Either way I think we ought to warn the user they are potentially doing something inaccurate.\r\n\r\nThe documentation for this method will also need changing to clarify what is happening.",
      "issue_closed_at": "2020-04-16T09:14:21Z",
      "base_commit": "60fb417789abd03269f4d3af412ee2004983e869",
      "changes": [
        {
          "file": "astropy/coordinates/sky_coordinate.py",
          "type": "function",
          "name": "radial_velocity_correction",
          "class_name": "SkyCoord",
          "code": "def radial_velocity_correction(self, kind='barycentric', obstime=None,\n                                   location=None):\n        \"\"\"\n        Compute the correction required to convert a radial velocity at a given\n        time and place on the Earth's Surface to a barycentric or heliocentric\n        velocity.\n\n        Parameters\n        ----------\n        kind : str\n            The kind of velocity correction.  Must be 'barycentric' or\n            'heliocentric'.\n        obstime : `~astropy.time.Time` or None, optional\n            The time at which to compute the correction.  If `None`, the\n            ``obstime`` frame attribute on the `SkyCoord` will be used.\n        location : `~astropy.coordinates.EarthLocation` or None, optional\n            The observer location at which to compute the correction.  If\n            `None`, the  ``location`` frame attribute on the passed-in\n            ``obstime`` will be used, and if that is None, the ``location``\n            frame attribute on the `SkyCoord` will be used.\n\n        Raises\n        ------\n        ValueError\n            If either ``obstime`` or ``location`` are passed in (not ``None``)\n            when the frame attribute is already set on this `SkyCoord`.\n        TypeError\n            If ``obstime`` or ``location`` aren't provided, either as arguments\n            or as frame attributes.\n\n        Returns\n        -------\n        vcorr : `~astropy.units.Quantity` with velocity units\n            The  correction with a positive sign.  I.e., *add* this\n            to an observed radial velocity to get the barycentric (or\n            heliocentric) velocity. If m/s precision or better is needed,\n            see the notes below.\n\n        Notes\n        -----\n        The barycentric correction is calculated to higher precision than the\n        heliocentric correction and includes additional physics (e.g time dilation).\n        Use barycentric corrections if m/s precision is required.\n\n        The algorithm here is sufficient to perform corrections at the mm/s level, but\n        care is needed in application. The barycentric correction returned uses the optical\n        approximation v = z * c. Strictly speaking, the barycentric correction is\n        multiplicative and should be applied as::\n\n          >>> from astropy.time import Time\n          >>> from astropy.coordinates import SkyCoord, EarthLocation\n          >>> from astropy.constants import c\n          >>> t = Time(56370.5, format='mjd', scale='utc')\n          >>> loc = EarthLocation('149d33m00.5s','-30d18m46.385s',236.87*u.m)\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> vcorr = sc.radial_velocity_correction(kind='barycentric', obstime=t, location=loc)  # doctest: +REMOTE_DATA\n          >>> rv = rv + vcorr + rv * vcorr / c  # doctest: +SKIP\n\n        Also note that this method returns the correction velocity in the so-called\n        *optical convention*::\n\n          >>> vcorr = zb * c  # doctest: +SKIP\n\n        where ``zb`` is the barycentric correction redshift as defined in section 3\n        of Wright & Eastman (2014). The application formula given above follows from their\n        equation (11) under assumption that the radial velocity ``rv`` has also been defined\n        using the same optical convention. Note, this can be regarded as a matter of\n        velocity definition and does not by itself imply any loss of accuracy, provided\n        sufficient care has been taken during interpretation of the results. If you need\n        the barycentric correction expressed as the full relativistic velocity (e.g., to provide\n        it as the input to another software which performs the application), the\n        following recipe can be used::\n\n          >>> zb = vcorr / c  # doctest: +REMOTE_DATA\n          >>> zb_plus_one_squared = (zb + 1) ** 2  # doctest: +REMOTE_DATA\n          >>> vcorr_rel = c * (zb_plus_one_squared - 1) / (zb_plus_one_squared + 1)  # doctest: +REMOTE_DATA\n\n        or alternatively using just equivalencies::\n\n          >>> vcorr_rel = vcorr.to(u.Hz, u.doppler_optical(1*u.Hz)).to(vcorr.unit, u.doppler_relativistic(1*u.Hz))  # doctest: +REMOTE_DATA\n\n        See also `~astropy.units.equivalencies.doppler_optical`,\n        `~astropy.units.equivalencies.doppler_radio`, and\n        `~astropy.units.equivalencies.doppler_relativistic` for more information on\n        the velocity conventions.\n\n        The default is for this method to use the builtin ephemeris for\n        computing the sun and earth location.  Other ephemerides can be chosen\n        by setting the `~astropy.coordinates.solar_system_ephemeris` variable,\n        either directly or via ``with`` statement.  For example, to use the JPL\n        ephemeris, do::\n\n          >>> from astropy.coordinates import solar_system_ephemeris\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> with solar_system_ephemeris.set('jpl'):  # doctest: +REMOTE_DATA\n          ...     rv += sc.radial_velocity_correction(obstime=t, location=loc)  # doctest: +SKIP\n\n        \"\"\"\n        # has to be here to prevent circular imports\n        from .solar_system import get_body_barycentric_posvel\n\n        # location validation\n        timeloc = getattr(obstime, 'location', None)\n        if location is None:\n            if self.location is not None:\n                location = self.location\n                if timeloc is not None:\n                    raise ValueError('`location` cannot be in both the '\n                                     'passed-in `obstime` and this `SkyCoord` '\n                                     'because it is ambiguous which is meant '\n                                     'for the radial_velocity_correction.')\n            elif timeloc is not None:\n                location = timeloc\n            else:\n                raise TypeError('Must provide a `location` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute, as an attribute on '\n                                'the passed in `obstime`, or in the method '\n                                'call.')\n\n        elif self.location is not None or timeloc is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`location` argument is passed in and there is '\n                             'also a  `location` attribute on this SkyCoord or '\n                             'the passed-in `obstime`.')\n\n        # obstime validation\n        if obstime is None:\n            obstime = self.obstime\n            if obstime is None:\n                raise TypeError('Must provide an `obstime` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute or in the method '\n                                'call.')\n        elif self.obstime is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`obstime` argument is passed in and it is '\n                             'inconsistent with the `obstime` frame '\n                             'attribute on the SkyCoord')\n\n        pos_earth, v_earth = get_body_barycentric_posvel('earth', obstime)\n        if kind == 'barycentric':\n            v_origin_to_earth = v_earth\n        elif kind == 'heliocentric':\n            v_sun = get_body_barycentric_posvel('sun', obstime)[1]\n            v_origin_to_earth = v_earth - v_sun\n        else:\n            raise ValueError(\"`kind` argument to radial_velocity_correction must \"\n                             \"be 'barycentric' or 'heliocentric', but got \"\n                             \"'{}'\".format(kind))\n\n        gcrs_p, gcrs_v = location.get_gcrs_posvel(obstime)\n        # transforming to GCRS is not the correct thing to do here, since we don't want to\n        # include aberration (or light deflection)? Instead, only apply parallax if necessary\n        icrs_cart = self.icrs.cartesian\n        icrs_cart_novel = icrs_cart.without_differentials()\n        if self.data.__class__ is UnitSphericalRepresentation:\n            targcart = icrs_cart_novel\n        else:\n            # skycoord has distances so apply parallax\n            obs_icrs_cart = pos_earth + gcrs_p\n            targcart = icrs_cart_novel - obs_icrs_cart\n            targcart /= targcart.norm()\n\n        if kind == 'barycentric':\n            beta_obs = (v_origin_to_earth + gcrs_v) / speed_of_light\n            gamma_obs = 1 / np.sqrt(1 - beta_obs.norm()**2)\n            gr = location.gravitational_redshift(obstime)\n            # barycentric redshift according to eq 28 in Wright & Eastmann (2014),\n            # neglecting Shapiro delay and effects of the star's own motion\n            zb = gamma_obs * (1 + beta_obs.dot(targcart)) / (1 + gr/speed_of_light)\n            # try and get terms corresponding to stellar motion.\n            if icrs_cart.differentials:\n                try:\n                    ro = self.icrs.cartesian\n                    beta_star = ro.differentials['s'].to_cartesian() / speed_of_light\n                    # ICRS unit vector at coordinate epoch\n                    ro = ro.without_differentials()\n                    ro /= ro.norm()\n                    zb *= (1 + beta_star.dot(ro)) / (1 + beta_star.dot(targcart))\n                except u.UnitConversionError:\n                    warnings.warn(\"SkyCoord contains some velocity information, but not enough to \"\n                                  \"calculate the full space motion of the source, and so this has \"\n                                  \"been ignored for the purposes of calculating the radial velocity \"\n                                  \"correction. This can lead to errors on the order of metres/second.\",\n                                  AstropyUserWarning)\n\n            zb = zb - 1\n            return zb * speed_of_light\n        else:\n            # do a simpler correction ignoring time dilation and gravitational redshift\n            # this is adequate since Heliocentric corrections shouldn't be used if\n            # cm/s precision is required.\n            return targcart.dot(v_origin_to_earth + gcrs_v)"
        },
        {
          "file": "astropy/coordinates/sky_coordinate.py",
          "type": "function",
          "name": "radial_velocity_correction",
          "class_name": "SkyCoord",
          "code": "def radial_velocity_correction(self, kind='barycentric', obstime=None,\n                                   location=None):\n        \"\"\"\n        Compute the correction required to convert a radial velocity at a given\n        time and place on the Earth's Surface to a barycentric or heliocentric\n        velocity.\n\n        Parameters\n        ----------\n        kind : str\n            The kind of velocity correction.  Must be 'barycentric' or\n            'heliocentric'.\n        obstime : `~astropy.time.Time` or None, optional\n            The time at which to compute the correction.  If `None`, the\n            ``obstime`` frame attribute on the `SkyCoord` will be used.\n        location : `~astropy.coordinates.EarthLocation` or None, optional\n            The observer location at which to compute the correction.  If\n            `None`, the  ``location`` frame attribute on the passed-in\n            ``obstime`` will be used, and if that is None, the ``location``\n            frame attribute on the `SkyCoord` will be used.\n\n        Raises\n        ------\n        ValueError\n            If either ``obstime`` or ``location`` are passed in (not ``None``)\n            when the frame attribute is already set on this `SkyCoord`.\n        TypeError\n            If ``obstime`` or ``location`` aren't provided, either as arguments\n            or as frame attributes.\n\n        Returns\n        -------\n        vcorr : `~astropy.units.Quantity` with velocity units\n            The  correction with a positive sign.  I.e., *add* this\n            to an observed radial velocity to get the barycentric (or\n            heliocentric) velocity. If m/s precision or better is needed,\n            see the notes below.\n\n        Notes\n        -----\n        The barycentric correction is calculated to higher precision than the\n        heliocentric correction and includes additional physics (e.g time dilation).\n        Use barycentric corrections if m/s precision is required.\n\n        The algorithm here is sufficient to perform corrections at the mm/s level, but\n        care is needed in application. The barycentric correction returned uses the optical\n        approximation v = z * c. Strictly speaking, the barycentric correction is\n        multiplicative and should be applied as::\n\n          >>> from astropy.time import Time\n          >>> from astropy.coordinates import SkyCoord, EarthLocation\n          >>> from astropy.constants import c\n          >>> t = Time(56370.5, format='mjd', scale='utc')\n          >>> loc = EarthLocation('149d33m00.5s','-30d18m46.385s',236.87*u.m)\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> vcorr = sc.radial_velocity_correction(kind='barycentric', obstime=t, location=loc)  # doctest: +REMOTE_DATA\n          >>> rv = rv + vcorr + rv * vcorr / c  # doctest: +SKIP\n\n        Also note that this method returns the correction velocity in the so-called\n        *optical convention*::\n\n          >>> vcorr = zb * c  # doctest: +SKIP\n\n        where ``zb`` is the barycentric correction redshift as defined in section 3\n        of Wright & Eastman (2014). The application formula given above follows from their\n        equation (11) under assumption that the radial velocity ``rv`` has also been defined\n        using the same optical convention. Note, this can be regarded as a matter of\n        velocity definition and does not by itself imply any loss of accuracy, provided\n        sufficient care has been taken during interpretation of the results. If you need\n        the barycentric correction expressed as the full relativistic velocity (e.g., to provide\n        it as the input to another software which performs the application), the\n        following recipe can be used::\n\n          >>> zb = vcorr / c  # doctest: +REMOTE_DATA\n          >>> zb_plus_one_squared = (zb + 1) ** 2  # doctest: +REMOTE_DATA\n          >>> vcorr_rel = c * (zb_plus_one_squared - 1) / (zb_plus_one_squared + 1)  # doctest: +REMOTE_DATA\n\n        or alternatively using just equivalencies::\n\n          >>> vcorr_rel = vcorr.to(u.Hz, u.doppler_optical(1*u.Hz)).to(vcorr.unit, u.doppler_relativistic(1*u.Hz))  # doctest: +REMOTE_DATA\n\n        See also `~astropy.units.equivalencies.doppler_optical`,\n        `~astropy.units.equivalencies.doppler_radio`, and\n        `~astropy.units.equivalencies.doppler_relativistic` for more information on\n        the velocity conventions.\n\n        The default is for this method to use the builtin ephemeris for\n        computing the sun and earth location.  Other ephemerides can be chosen\n        by setting the `~astropy.coordinates.solar_system_ephemeris` variable,\n        either directly or via ``with`` statement.  For example, to use the JPL\n        ephemeris, do::\n\n          >>> from astropy.coordinates import solar_system_ephemeris\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> with solar_system_ephemeris.set('jpl'):  # doctest: +REMOTE_DATA\n          ...     rv += sc.radial_velocity_correction(obstime=t, location=loc)  # doctest: +SKIP\n\n        \"\"\"\n        # has to be here to prevent circular imports\n        from .solar_system import get_body_barycentric_posvel\n\n        # location validation\n        timeloc = getattr(obstime, 'location', None)\n        if location is None:\n            if self.location is not None:\n                location = self.location\n                if timeloc is not None:\n                    raise ValueError('`location` cannot be in both the '\n                                     'passed-in `obstime` and this `SkyCoord` '\n                                     'because it is ambiguous which is meant '\n                                     'for the radial_velocity_correction.')\n            elif timeloc is not None:\n                location = timeloc\n            else:\n                raise TypeError('Must provide a `location` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute, as an attribute on '\n                                'the passed in `obstime`, or in the method '\n                                'call.')\n\n        elif self.location is not None or timeloc is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`location` argument is passed in and there is '\n                             'also a  `location` attribute on this SkyCoord or '\n                             'the passed-in `obstime`.')\n\n        # obstime validation\n        if obstime is None:\n            obstime = self.obstime\n            if obstime is None:\n                raise TypeError('Must provide an `obstime` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute or in the method '\n                                'call.')\n        elif self.obstime is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`obstime` argument is passed in and it is '\n                             'inconsistent with the `obstime` frame '\n                             'attribute on the SkyCoord')\n\n        pos_earth, v_earth = get_body_barycentric_posvel('earth', obstime)\n        if kind == 'barycentric':\n            v_origin_to_earth = v_earth\n        elif kind == 'heliocentric':\n            v_sun = get_body_barycentric_posvel('sun', obstime)[1]\n            v_origin_to_earth = v_earth - v_sun\n        else:\n            raise ValueError(\"`kind` argument to radial_velocity_correction must \"\n                             \"be 'barycentric' or 'heliocentric', but got \"\n                             \"'{}'\".format(kind))\n\n        gcrs_p, gcrs_v = location.get_gcrs_posvel(obstime)\n        # transforming to GCRS is not the correct thing to do here, since we don't want to\n        # include aberration (or light deflection)? Instead, only apply parallax if necessary\n        icrs_cart = self.icrs.cartesian\n        icrs_cart_novel = icrs_cart.without_differentials()\n        if self.data.__class__ is UnitSphericalRepresentation:\n            targcart = icrs_cart_novel\n        else:\n            # skycoord has distances so apply parallax\n            obs_icrs_cart = pos_earth + gcrs_p\n            targcart = icrs_cart_novel - obs_icrs_cart\n            targcart /= targcart.norm()\n\n        if kind == 'barycentric':\n            beta_obs = (v_origin_to_earth + gcrs_v) / speed_of_light\n            gamma_obs = 1 / np.sqrt(1 - beta_obs.norm()**2)\n            gr = location.gravitational_redshift(obstime)\n            # barycentric redshift according to eq 28 in Wright & Eastmann (2014),\n            # neglecting Shapiro delay and effects of the star's own motion\n            zb = gamma_obs * (1 + beta_obs.dot(targcart)) / (1 + gr/speed_of_light)\n            # try and get terms corresponding to stellar motion.\n            if icrs_cart.differentials:\n                try:\n                    ro = self.icrs.cartesian\n                    beta_star = ro.differentials['s'].to_cartesian() / speed_of_light\n                    # ICRS unit vector at coordinate epoch\n                    ro = ro.without_differentials()\n                    ro /= ro.norm()\n                    zb *= (1 + beta_star.dot(ro)) / (1 + beta_star.dot(targcart))\n                except u.UnitConversionError:\n                    warnings.warn(\"SkyCoord contains some velocity information, but not enough to \"\n                                  \"calculate the full space motion of the source, and so this has \"\n                                  \"been ignored for the purposes of calculating the radial velocity \"\n                                  \"correction. This can lead to errors on the order of metres/second.\",\n                                  AstropyUserWarning)\n\n            zb = zb - 1\n            return zb * speed_of_light\n        else:\n            # do a simpler correction ignoring time dilation and gravitational redshift\n            # this is adequate since Heliocentric corrections shouldn't be used if\n            # cm/s precision is required.\n            return targcart.dot(v_origin_to_earth + gcrs_v)"
        },
        {
          "file": "astropy/coordinates/sky_coordinate.py",
          "type": "function",
          "name": "radial_velocity_correction",
          "class_name": "SkyCoord",
          "code": "def radial_velocity_correction(self, kind='barycentric', obstime=None,\n                                   location=None):\n        \"\"\"\n        Compute the correction required to convert a radial velocity at a given\n        time and place on the Earth's Surface to a barycentric or heliocentric\n        velocity.\n\n        Parameters\n        ----------\n        kind : str\n            The kind of velocity correction.  Must be 'barycentric' or\n            'heliocentric'.\n        obstime : `~astropy.time.Time` or None, optional\n            The time at which to compute the correction.  If `None`, the\n            ``obstime`` frame attribute on the `SkyCoord` will be used.\n        location : `~astropy.coordinates.EarthLocation` or None, optional\n            The observer location at which to compute the correction.  If\n            `None`, the  ``location`` frame attribute on the passed-in\n            ``obstime`` will be used, and if that is None, the ``location``\n            frame attribute on the `SkyCoord` will be used.\n\n        Raises\n        ------\n        ValueError\n            If either ``obstime`` or ``location`` are passed in (not ``None``)\n            when the frame attribute is already set on this `SkyCoord`.\n        TypeError\n            If ``obstime`` or ``location`` aren't provided, either as arguments\n            or as frame attributes.\n\n        Returns\n        -------\n        vcorr : `~astropy.units.Quantity` with velocity units\n            The  correction with a positive sign.  I.e., *add* this\n            to an observed radial velocity to get the barycentric (or\n            heliocentric) velocity. If m/s precision or better is needed,\n            see the notes below.\n\n        Notes\n        -----\n        The barycentric correction is calculated to higher precision than the\n        heliocentric correction and includes additional physics (e.g time dilation).\n        Use barycentric corrections if m/s precision is required.\n\n        The algorithm here is sufficient to perform corrections at the mm/s level, but\n        care is needed in application. The barycentric correction returned uses the optical\n        approximation v = z * c. Strictly speaking, the barycentric correction is\n        multiplicative and should be applied as::\n\n          >>> from astropy.time import Time\n          >>> from astropy.coordinates import SkyCoord, EarthLocation\n          >>> from astropy.constants import c\n          >>> t = Time(56370.5, format='mjd', scale='utc')\n          >>> loc = EarthLocation('149d33m00.5s','-30d18m46.385s',236.87*u.m)\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> vcorr = sc.radial_velocity_correction(kind='barycentric', obstime=t, location=loc)  # doctest: +REMOTE_DATA\n          >>> rv = rv + vcorr + rv * vcorr / c  # doctest: +SKIP\n\n        Also note that this method returns the correction velocity in the so-called\n        *optical convention*::\n\n          >>> vcorr = zb * c  # doctest: +SKIP\n\n        where ``zb`` is the barycentric correction redshift as defined in section 3\n        of Wright & Eastman (2014). The application formula given above follows from their\n        equation (11) under assumption that the radial velocity ``rv`` has also been defined\n        using the same optical convention. Note, this can be regarded as a matter of\n        velocity definition and does not by itself imply any loss of accuracy, provided\n        sufficient care has been taken during interpretation of the results. If you need\n        the barycentric correction expressed as the full relativistic velocity (e.g., to provide\n        it as the input to another software which performs the application), the\n        following recipe can be used::\n\n          >>> zb = vcorr / c  # doctest: +REMOTE_DATA\n          >>> zb_plus_one_squared = (zb + 1) ** 2  # doctest: +REMOTE_DATA\n          >>> vcorr_rel = c * (zb_plus_one_squared - 1) / (zb_plus_one_squared + 1)  # doctest: +REMOTE_DATA\n\n        or alternatively using just equivalencies::\n\n          >>> vcorr_rel = vcorr.to(u.Hz, u.doppler_optical(1*u.Hz)).to(vcorr.unit, u.doppler_relativistic(1*u.Hz))  # doctest: +REMOTE_DATA\n\n        See also `~astropy.units.equivalencies.doppler_optical`,\n        `~astropy.units.equivalencies.doppler_radio`, and\n        `~astropy.units.equivalencies.doppler_relativistic` for more information on\n        the velocity conventions.\n\n        The default is for this method to use the builtin ephemeris for\n        computing the sun and earth location.  Other ephemerides can be chosen\n        by setting the `~astropy.coordinates.solar_system_ephemeris` variable,\n        either directly or via ``with`` statement.  For example, to use the JPL\n        ephemeris, do::\n\n          >>> from astropy.coordinates import solar_system_ephemeris\n          >>> sc = SkyCoord(1*u.deg, 2*u.deg)\n          >>> with solar_system_ephemeris.set('jpl'):  # doctest: +REMOTE_DATA\n          ...     rv += sc.radial_velocity_correction(obstime=t, location=loc)  # doctest: +SKIP\n\n        \"\"\"\n        # has to be here to prevent circular imports\n        from .solar_system import get_body_barycentric_posvel\n\n        # location validation\n        timeloc = getattr(obstime, 'location', None)\n        if location is None:\n            if self.location is not None:\n                location = self.location\n                if timeloc is not None:\n                    raise ValueError('`location` cannot be in both the '\n                                     'passed-in `obstime` and this `SkyCoord` '\n                                     'because it is ambiguous which is meant '\n                                     'for the radial_velocity_correction.')\n            elif timeloc is not None:\n                location = timeloc\n            else:\n                raise TypeError('Must provide a `location` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute, as an attribute on '\n                                'the passed in `obstime`, or in the method '\n                                'call.')\n\n        elif self.location is not None or timeloc is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`location` argument is passed in and there is '\n                             'also a  `location` attribute on this SkyCoord or '\n                             'the passed-in `obstime`.')\n\n        # obstime validation\n        if obstime is None:\n            obstime = self.obstime\n            if obstime is None:\n                raise TypeError('Must provide an `obstime` to '\n                                'radial_velocity_correction, either as a '\n                                'SkyCoord frame attribute or in the method '\n                                'call.')\n        elif self.obstime is not None:\n            raise ValueError('Cannot compute radial velocity correction if '\n                             '`obstime` argument is passed in and it is '\n                             'inconsistent with the `obstime` frame '\n                             'attribute on the SkyCoord')\n\n        pos_earth, v_earth = get_body_barycentric_posvel('earth', obstime)\n        if kind == 'barycentric':\n            v_origin_to_earth = v_earth\n        elif kind == 'heliocentric':\n            v_sun = get_body_barycentric_posvel('sun', obstime)[1]\n            v_origin_to_earth = v_earth - v_sun\n        else:\n            raise ValueError(\"`kind` argument to radial_velocity_correction must \"\n                             \"be 'barycentric' or 'heliocentric', but got \"\n                             \"'{}'\".format(kind))\n\n        gcrs_p, gcrs_v = location.get_gcrs_posvel(obstime)\n        # transforming to GCRS is not the correct thing to do here, since we don't want to\n        # include aberration (or light deflection)? Instead, only apply parallax if necessary\n        icrs_cart = self.icrs.cartesian\n        icrs_cart_novel = icrs_cart.without_differentials()\n        if self.data.__class__ is UnitSphericalRepresentation:\n            targcart = icrs_cart_novel\n        else:\n            # skycoord has distances so apply parallax\n            obs_icrs_cart = pos_earth + gcrs_p\n            targcart = icrs_cart_novel - obs_icrs_cart\n            targcart /= targcart.norm()\n\n        if kind == 'barycentric':\n            beta_obs = (v_origin_to_earth + gcrs_v) / speed_of_light\n            gamma_obs = 1 / np.sqrt(1 - beta_obs.norm()**2)\n            gr = location.gravitational_redshift(obstime)\n            # barycentric redshift according to eq 28 in Wright & Eastmann (2014),\n            # neglecting Shapiro delay and effects of the star's own motion\n            zb = gamma_obs * (1 + beta_obs.dot(targcart)) / (1 + gr/speed_of_light)\n            # try and get terms corresponding to stellar motion.\n            if icrs_cart.differentials:\n                try:\n                    ro = self.icrs.cartesian\n                    beta_star = ro.differentials['s'].to_cartesian() / speed_of_light\n                    # ICRS unit vector at coordinate epoch\n                    ro = ro.without_differentials()\n                    ro /= ro.norm()\n                    zb *= (1 + beta_star.dot(ro)) / (1 + beta_star.dot(targcart))\n                except u.UnitConversionError:\n                    warnings.warn(\"SkyCoord contains some velocity information, but not enough to \"\n                                  \"calculate the full space motion of the source, and so this has \"\n                                  \"been ignored for the purposes of calculating the radial velocity \"\n                                  \"correction. This can lead to errors on the order of metres/second.\",\n                                  AstropyUserWarning)\n\n            zb = zb - 1\n            return zb * speed_of_light\n        else:\n            # do a simpler correction ignoring time dilation and gravitational redshift\n            # this is adequate since Heliocentric corrections shouldn't be used if\n            # cm/s precision is required.\n            return targcart.dot(v_origin_to_earth + gcrs_v)"
        }
      ]
    },
    {
      "pr_number": 10415,
      "pr_title": "Allow fitting of a wider range of compound models ",
      "pr_body": "Fixes #10414\r\n",
      "issue_id": 10414,
      "issue_title": "Fitting of compound models too restrictive",
      "issue_body": "### Description\r\n<!-- Provide a general description of the bug. -->\r\n@karllark reported on Slack:\r\n\r\nI'm getting an interesting error when fitting a compound model.  The error is:\r\n```\r\nValueError: Fitting a compound model without units can only be performed on\r\ncompound models that only use the arithmetic operators + and -\r\n```\r\nThe model I'm fitting has units for a summed set of models including BlackBody, Drude1D, Gaussian1D models with units of MJy/sr that are then multiplied by a dust attenuation model that does not have units.\r\n\r\nAn example to reproduce the error:\r\n```\r\nimport numpy as np\r\nfrom astropy.modeling import fitting, models\r\n\r\nfitter = fitting.LevMarLSQFitter()\r\nmodel = models.BlackBody(temperature=3000 * u.K) * models.Const1D(amplitude=1.0)\r\n    \r\nx = [1.0, 2.0, 3.0] * u.micron\r\nn = np.random.normal(3)\r\ny = model(x) * (1.0 + n)\r\nres = fitter(model, x, y)\r\n\r\n```\r\n\r\n",
      "issue_closed_at": "2020-06-15T18:50:02Z",
      "base_commit": "1a065d5ce403e226799cfb3d606fda33be0a6c08",
      "changes": [
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "without_units_for_data",
          "class_name": "Model",
          "code": "def without_units_for_data(self, **kwargs):\n        \"\"\"\n        Return an instance of the model for which the parameter values have\n        been converted to the right units for the data, then the units have\n        been stripped away.\n\n        The input and output Quantity objects should be given as keyword\n        arguments.\n\n        Notes\n        -----\n\n        This method is needed in order to be able to fit models with units in\n        the parameters, since we need to temporarily strip away the units from\n        the model during the fitting (which might be done by e.g. scipy\n        functions).\n\n        The units that the parameters should be converted to are not\n        necessarily the units of the input data, but are derived from them.\n        Model subclasses that want fitting to work in the presence of\n        quantities need to define a ``_parameter_units_for_data_units`` method\n        that takes the input and output units (as two dictionaries) and\n        returns a dictionary giving the target units for each parameter.\n\n        For compound models this will only work when the expression only\n        involves the addition or subtraction operators.\n        \"\"\"\n        if isinstance(self, CompoundModel):\n            self._make_opset()\n            if not self._opset.issubset(set(('+', '-'))):\n                raise ValueError(\n                    \"Fitting a compound model without units can only be performed on\"\n                    \"compound models that only use the arithmetic operators + and -\")\n\n        model = self.copy()\n\n        inputs_unit = {inp: getattr(kwargs[inp], 'unit', dimensionless_unscaled)\n                       for inp in self.inputs if kwargs[inp] is not None}\n\n        outputs_unit = {out: getattr(kwargs[out], 'unit', dimensionless_unscaled)\n                        for out in self.outputs if kwargs[out] is not None}\n        parameter_units = self._parameter_units_for_data_units(inputs_unit,\n                                                               outputs_unit)\n        for name, unit in parameter_units.items():\n            parameter = getattr(model, name)\n            if parameter.unit is not None:\n                parameter.value = parameter.quantity.to(unit).value\n                parameter._set_unit(None, force=True)\n\n        if isinstance(model, CompoundModel):\n            model.strip_units_from_tree()\n\n        return model"
        },
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CompoundModel",
          "code": "def __init__(self, op, left, right, name=None, inverse=None):\n        self.__dict__['_param_names'] = None\n        self._n_submodels = None\n        self.op = op\n        self.left = left\n        self.right = right\n        self._bounding_box = None\n        self._user_bounding_box = None\n        self._leaflist = None\n        self._opset = None\n        self._tdict = None\n        self._parameters = None\n        self._parameters_ = None\n        self._param_metrics = None\n\n        if inverse:\n            warnings.warn(\n                \"The 'inverse' argument is deprecated.  Instead, set the inverse \"\n                \"property after CompoundModel is initialized.\",\n                AstropyDeprecationWarning\n            )\n            self.inverse = inverse\n\n        if op != 'fix_inputs' and len(left) != len(right):\n            raise ValueError(\n                'Both operands must have equal values for n_models')\n        self._n_models = len(left)\n\n        if op != 'fix_inputs' and ((left.model_set_axis != right.model_set_axis)\n                                   or left.model_set_axis):  # not False and not 0\n            raise ValueError(\"model_set_axis must be False or 0 and consistent for operands\")\n        self._model_set_axis = left.model_set_axis\n\n        if op in ['+', '-', '*', '/', '**'] or op in SPECIAL_OPERATORS:\n            if (left.n_inputs != right.n_inputs) or \\\n               (left.n_outputs != right.n_outputs):\n                raise ModelDefinitionError(\n                    'Both operands must match numbers of inputs and outputs')\n            self.n_inputs = left.n_inputs\n            self.n_outputs = left.n_outputs\n            self.inputs = left.inputs\n            self.outputs = left.outputs\n        elif op == '&':\n            self.n_inputs = left.n_inputs + right.n_inputs\n            self.n_outputs = left.n_outputs + right.n_outputs\n            self.inputs = combine_labels(left.inputs, right.inputs)\n            self.outputs = combine_labels(left.outputs, right.outputs)\n        elif op == '|':\n            if left.n_outputs != right.n_inputs:\n                raise ModelDefinitionError(\n                    \"Unsupported operands for |: {0} (n_inputs={1}, \"\n                    \"n_outputs={2}) and {3} (n_inputs={4}, n_outputs={5}); \"\n                    \"n_outputs for the left-hand model must match n_inputs \"\n                    \"for the right-hand model.\".format(\n                        left.name, left.n_inputs, left.n_outputs, right.name,\n                        right.n_inputs, right.n_outputs))\n\n            self.n_inputs = left.n_inputs\n            self.n_outputs = right.n_outputs\n            self.inputs = left.inputs\n            self.outputs = right.outputs\n        elif op == 'fix_inputs':\n            if not isinstance(left, Model):\n                raise ValueError('First argument to \"fix_inputs\" must be an instance of an astropy Model.')\n            if not isinstance(right, dict):\n                raise ValueError('Expected a dictionary for second argument of \"fix_inputs\".')\n\n            # Dict keys must match either possible indices\n            # for model on left side, or names for inputs.\n            self.n_inputs = left.n_inputs - len(right)\n            # Assign directly to the private attribute (instead of using the setter)\n            # to avoid asserting the new number of outputs matches the old one.\n            self._outputs = left.outputs\n            self.n_outputs = left.n_outputs\n            newinputs = list(left.inputs)\n            keys = right.keys()\n            input_ind = []\n            for key in keys:\n                if isinstance(key, int):\n                    if key >= left.n_inputs or key < 0:\n                        raise ValueError(\n                            'Substitution key integer value '\n                            'not among possible input choices.')\n                    if key in input_ind:\n                        raise ValueError(\"Duplicate specification of \"\n                                         \"same input (index/name).\")\n                    input_ind.append(key)\n                elif isinstance(key, str):\n                    if key not in left.inputs:\n                        raise ValueError(\n                            'Substitution key string not among possible '\n                            'input choices.')\n                    # Check to see it doesn't match positional\n                    # specification.\n                    ind = left.inputs.index(key)\n                    if ind in input_ind:\n                        raise ValueError(\"Duplicate specification of \"\n                                         \"same input (index/name).\")\n                    input_ind.append(ind)\n            # Remove substituted inputs\n            input_ind.sort()\n            input_ind.reverse()\n            for ind in input_ind:\n                del newinputs[ind]\n            self.inputs = tuple(newinputs)\n            # Now check to see if the input model has bounding_box defined.\n            # If so, remove the appropriate dimensions and set it for this\n            # instance.\n            try:\n                bounding_box = self.left.bounding_box\n                self._fix_input_bounding_box(input_ind)\n            except NotImplementedError:\n                pass\n\n        else:\n            raise ModelDefinitionError('Illegal operator: ', self.op)\n        self.name = name\n        self._fittable = None\n        self.fit_deriv = None\n        self.col_fit_deriv = None\n        if op in ('|', '+', '-'):\n            self.linear = left.linear and right.linear\n        else:\n            self.linear = False\n        self.eqcons = []\n        self.ineqcons = []\n        self._map_parameters()"
        },
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "_make_leaflist",
          "class_name": "CompoundModel",
          "code": "def _make_leaflist(self):\n        tdict = {}\n        leaflist = []\n        make_subtree_dict(self, '', tdict, leaflist)\n        self._leaflist = leaflist\n        self._tdict = tdict"
        },
        {
          "file": "astropy/modeling/functional_models.py",
          "type": "function",
          "name": "fit_deriv",
          "class_name": "Exponential1D",
          "code": "def fit_deriv(x, amplitude, tau):\n        ''' Derivative with respect to parameters'''\n        d_amplitude = np.exp(x / tau)\n        d_tau = -amplitude * (x / tau**2) * np.exp(x / tau)\n        return [d_amplitude, d_tau]"
        },
        {
          "file": "astropy/modeling/functional_models.py",
          "type": "function",
          "name": "fit_deriv",
          "class_name": "Exponential1D",
          "code": "def fit_deriv(x, amplitude, tau):\n        ''' Derivative with respect to parameters'''\n        d_amplitude = np.exp(x / tau)\n        d_tau = -amplitude * (x / tau**2) * np.exp(x / tau)\n        return [d_amplitude, d_tau]"
        }
      ]
    },
    {
      "pr_number": 10158,
      "pr_title": "fix units of compound models",
      "pr_body": "Fixes #9921\r\n\r\nMap `input_units` and `return_units` of compound models correctly.",
      "issue_id": 9921,
      "issue_title": "CompoundModel.return_units assumes that outputs will have have same names as inputs",
      "issue_body": "### Description\r\nWhen a CompoundModel is constructed from a component model that has different input and output names, its return_units method fails with a KeyError.\r\n\r\n### Steps to Reproduce\r\n```python\r\nfrom astropy.modeling import Model\r\nfrom astropy import units as u\r\n\r\nclass ExampleModel(Model):\r\n    n_inputs = 1\r\n    n_outputs = 1\r\n    def __init__(self):\r\n        self._input_units = {\"x\": u.m}\r\n        self._return_units = {\"y\": u.s}\r\n        super().__init__()\r\n \r\n    def evaluate(self, input):\r\n        return u.Quantity(1, u.s)\r\n\r\nm = ExampleModel()\r\n\r\n(m & m).return_units\r\n```\r\nResult:\r\n```\r\nKeyError                                  Traceback (most recent call last)\r\n<ipython-input-1-6d0cb817f1ae> in <module>\r\n     15 m = ExampleModel()\r\n     16\r\n---> 17 (m & m).return_units\r\n\r\n~/repos/astropy/astropy/modeling/core.py in return_units(self)\r\n   3129         inputs_map = self.inputs_map()\r\n   3130         return {key: inputs_map[key][0].return_units[orig_key]\r\n-> 3131                 for key, (mod, orig_key) in inputs_map.items()\r\n   3132                 if inputs_map[key][0].return_units is not None}\r\n   3133\r\n\r\n~/repos/astropy/astropy/modeling/core.py in <dictcomp>(.0)\r\n   3130         return {key: inputs_map[key][0].return_units[orig_key]\r\n   3131                 for key, (mod, orig_key) in inputs_map.items()\r\n-> 3132                 if inputs_map[key][0].return_units is not None}\r\n   3133\r\n   3134     def outputs_map(self):\r\n\r\nKeyError: 'x'\r\n```\r\n\r\n### System Details\r\n```\r\nDarwin-17.7.0-x86_64-i386-64bit\r\nPython 3.7.6 | packaged by conda-forge | (default, Jan  7 2020, 22:05:27)\r\n[Clang 9.0.1 ]\r\nNumpy 1.18.1\r\nastropy 4.1.dev503+g1cb6c81e8.d20200129\r\nScipy 1.4.1\r\n```",
      "issue_closed_at": "2020-04-21T18:50:33Z",
      "base_commit": "59df4c077c3a0530af4735c44be3db8f2083ad6e",
      "changes": [
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "inputs_map",
          "class_name": "CompoundModel",
          "code": "def inputs_map(self):\n        \"\"\"\n        Map the names of the inputs to this ExpressionTree to the inputs to the leaf models.\n        \"\"\"\n        inputs_map = {}\n        if not isinstance(self.op, str):  # If we don't have an operator the mapping is trivial\n            return {inp: (self, inp) for inp in self.inputs}\n\n        elif self.op == '|':\n            if isinstance(self.left, CompoundModel):\n                l_inputs_map = self.left.inputs_map()\n            for inp in self.inputs:\n                if isinstance(self.left, CompoundModel):\n                    inputs_map[inp] = l_inputs_map[inp]\n                else:\n                    inputs_map[inp] = self.left, inp\n        elif self.op == '&':\n            if isinstance(self.left, CompoundModel):\n                l_inputs_map = self.left.inputs_map()\n            if isinstance(self.right, CompoundModel):\n                r_inputs_map = self.right.inputs_map()\n            for i, inp in enumerate(self.inputs):\n                if i < len(self.left.inputs):  # Get from left\n                    if isinstance(self.left, CompoundModel):\n                        inputs_map[inp] = l_inputs_map[self.left.inputs[i]]\n                    else:\n                        inputs_map[inp] = self.left, self.left.inputs[i]\n                else:  # Get from right\n                    if isinstance(self.right, CompoundModel):\n                        inputs_map[inp] = r_inputs_map[self.right.inputs[i - len(self.left.inputs)]]\n                    else:\n                        inputs_map[inp] = self.right, self.right.inputs[i - len(self.left.inputs)]\n        else:\n            if isinstance(self.left, CompoundModel):\n                l_inputs_map = self.left.inputs_map()\n            for inp in self.left.inputs:\n                if isinstance(self.left, CompoundModel):\n                    inputs_map[inp] = l_inputs_map[inp]\n                else:\n                    inputs_map[inp] = self.left, inp\n        return inputs_map"
        },
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "input_units_strict",
          "class_name": "CompoundModel",
          "code": "def input_units_strict(self):\n        inputs_map = self.inputs_map()\n        return {key: inputs_map[key][0].input_units_strict[orig_key]\n                for key, (mod, orig_key) in inputs_map.items()}"
        },
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "outputs_map",
          "class_name": "CompoundModel",
          "code": "def outputs_map(self):\n        \"\"\"\n        Map the names of the outputs to this ExpressionTree to the outputs to the leaf models.\n        \"\"\"\n        outputs_map = {}\n        if not isinstance(self.op, str):  # If we don't have an operator the mapping is trivial\n            return {out: (self, out) for out in self.outputs}\n\n        elif self.op == '|':\n            r_outputs_map = self.right.outputs_map()\n            for out in self.outputs:\n                if isinstance(self.right, CompoundModel):\n                    outputs_map[out] = r_outputs_map[out]\n                else:\n                    outputs_map[out] = self, out\n\n        elif self.op == '&':\n            l_outputs_map = self.left.outputs_map()\n            r_outputs_map = self.right.outputs_map()\n            for i, out in enumerate(self.outputs):\n                if i < len(self.left.outputs):  # Get from left\n                    if isinstance(self.left, CompoundModel):\n                        outputs_map[out] = l_outputs_map[self.left.outputs[i]]\n                    else:\n                        outputs_map[out] = self.left, self.left.outputs[i]\n                else:  # Get from right\n                    if isinstance(self.right, CompoundModel):\n                        outputs_map[out] = r_outputs_map[self.right.outputs[i - len(self.left.outputs)]]\n                    else:\n                        outputs_map[out] = self.right, self.right.outputs[i - len(self.left.outputs)]\n\n        else:\n            if isinstance(self.left, CompoundModel):\n                l_outputs_map = self.left.outputs_map()\n            for out in self.left.outputs:\n                if isinstance(self.left, CompoundModel):\n                    outputs_map[out] = l_outputs_map()[out]\n                else:\n                    outputs_map[out] = self.left, out\n        return outputs_map"
        },
        {
          "file": "astropy/modeling/core.py",
          "type": "function",
          "name": "outputs_map",
          "class_name": "CompoundModel",
          "code": "def outputs_map(self):\n        \"\"\"\n        Map the names of the outputs to this ExpressionTree to the outputs to the leaf models.\n        \"\"\"\n        outputs_map = {}\n        if not isinstance(self.op, str):  # If we don't have an operator the mapping is trivial\n            return {out: (self, out) for out in self.outputs}\n\n        elif self.op == '|':\n            r_outputs_map = self.right.outputs_map()\n            for out in self.outputs:\n                if isinstance(self.right, CompoundModel):\n                    outputs_map[out] = r_outputs_map[out]\n                else:\n                    outputs_map[out] = self, out\n\n        elif self.op == '&':\n            l_outputs_map = self.left.outputs_map()\n            r_outputs_map = self.right.outputs_map()\n            for i, out in enumerate(self.outputs):\n                if i < len(self.left.outputs):  # Get from left\n                    if isinstance(self.left, CompoundModel):\n                        outputs_map[out] = l_outputs_map[self.left.outputs[i]]\n                    else:\n                        outputs_map[out] = self.left, self.left.outputs[i]\n                else:  # Get from right\n                    if isinstance(self.right, CompoundModel):\n                        outputs_map[out] = r_outputs_map[self.right.outputs[i - len(self.left.outputs)]]\n                    else:\n                        outputs_map[out] = self.right, self.right.outputs[i - len(self.left.outputs)]\n\n        else:\n            if isinstance(self.left, CompoundModel):\n                l_outputs_map = self.left.outputs_map()\n            for out in self.left.outputs:\n                if isinstance(self.left, CompoundModel):\n                    outputs_map[out] = l_outputs_map()[out]\n                else:\n                    outputs_map[out] = self.left, out\n        return outputs_map"
        }
      ]
    },
    {
      "pr_number": 8876,
      "pr_title": "Let scalar Quantity.value return a numpy scalar",
      "pr_body": "Benefits: more consistent with `ndarray`, fixes odd problems like #8614.\r\n\r\nDisadvantages: numpy scalars do not always behave the same as python ones. E.g., `(-4.) ** 0.5` returns a complex number (`0+2j`), but `np.float_(-4.) ** 0.5` returns `nan`.\r\n\r\nfixes #8614 \r\n\r\nEDIT: @astrofrog - requested your review since this goes back a *long* time...\r\n\r\nEDIT: partially addresses #6389 as well; but fully addressing it would require returning an array scalar.",
      "issue_id": 8614,
      "issue_title": "Implicit dtype conversion happening when indexing CartesianRepresentation",
      "issue_body": "When a `CartesianRepresentation` object is created from a `np.float32` object, indexing it automatically upgrades it to `np.float64`.\r\n\r\nNormal behavior:\r\n\r\n```\r\nIn [83]: xyz = np.array([[1, 0, 0], [0.9, 0.1, 0]])\r\n\r\nIn [84]: CartesianRepresentation(xyz.astype(np.float32), xyz_axis=-1, unit=\"km\").xyz.dtype\r\nOut[84]: dtype('float32')\r\n\r\nIn [85]: CartesianRepresentation(xyz.astype(np.float32), xyz_axis=-1, unit=\"km\").xyz[0].dtype\r\nOut[85]: dtype('float32')\r\n```\r\n\r\nHowever, when indexing the representation object itself:\r\n\r\n```\r\nIn [86]: CartesianRepresentation(xyz.astype(np.float32), xyz_axis=-1, unit=\"km\")[0].xyz.dtype\r\nOut[86]: dtype('float64')\r\n```\r\n\r\n*Not to be confused* with #8613, which also changes the dtype but probably because of the internal conversion to `Quantity`:\r\n\r\n```\r\nIn [87]: CartesianRepresentation(xyz.astype(np.float16), xyz_axis=-1, unit=\"km\").xyz.dtype\r\nOut[87]: dtype('float64')\r\n```",
      "issue_closed_at": "2019-07-03T11:39:55Z",
      "base_commit": "872a7a1d7fdea083f8936c4ecbb5a2f5f7d1b020",
      "changes": [
        {
          "file": "astropy/units/quantity.py",
          "type": "function",
          "name": "to_value",
          "class_name": "Quantity",
          "code": "def to_value(self, unit=None, equivalencies=[]):\n        \"\"\"\n        The numerical value, possibly in a different unit.\n\n        Parameters\n        ----------\n        unit : `~astropy.units.UnitBase` instance or str, optional\n            The unit in which the value should be given. If not given or `None`,\n            use the current unit.\n\n        equivalencies : list of equivalence pairs, optional\n            A list of equivalence pairs to try if the units are not directly\n            convertible (see :ref:`unit_equivalencies`). If not provided or\n            ``[]``, class default equivalencies will be used (none for\n            `~astropy.units.Quantity`, but may be set for subclasses).\n            If `None`, no equivalencies will be applied at all, not even any\n            set globally or within a context.\n\n        Returns\n        -------\n        value : `~numpy.ndarray` or scalar\n            The value in the units specified. For arrays, this will be a view\n            of the data if no unit conversion was necessary.\n\n        See also\n        --------\n        to : Get a new instance in a different unit.\n        \"\"\"\n        if unit is None or unit is self.unit:\n            value = self.view(np.ndarray)\n        else:\n            unit = Unit(unit)\n            # We want a view if the unit does not change.  One could check\n            # with \"==\", but that calculates the scale that we need anyway.\n            # TODO: would be better for `unit.to` to have an in-place flag.\n            try:\n                scale = self.unit._to(unit)\n            except Exception:\n                # Short-cut failed; try default (maybe equivalencies help).\n                value = self._to_value(unit, equivalencies)\n            else:\n                value = self.view(np.ndarray)\n                if not is_effectively_unity(scale):\n                    # not in-place!\n                    value = value * scale\n\n        return value if self.shape else (value[()] if self.dtype.fields\n                                         else value.item())"
        },
        {
          "file": "astropy/units/utils.py",
          "type": "function",
          "name": "sanitize_scale",
          "class_name": null,
          "code": "def sanitize_scale(scale):\n    if is_effectively_unity(scale):\n        return 1.0\n\n    # Maximum speed for regular case where scale is a float.\n    if scale.__class__ is float:\n        return scale\n\n    # All classes that scale can be (int, float, complex, Fraction)\n    # have an \"imag\" attribute.\n    if scale.imag:\n        if abs(scale.real) > abs(scale.imag):\n            if is_effectively_unity(scale.imag/scale.real + 1):\n                return scale.real\n\n        elif is_effectively_unity(scale.real/scale.imag + 1):\n            return complex(0., scale.imag)\n\n        return scale\n\n    else:\n        return scale.real"
        }
      ]
    },
    {
      "pr_number": 11157,
      "pr_title": "BUG: VOTable NumericArray to broadcast mask",
      "pr_body": "<!-- This comments are hidden when you submit the pull request,\r\nso you do not need to remove them! -->\r\n\r\n<!-- Please be sure to check out our contributing guidelines,\r\nhttps://github.com/astropy/astropy/blob/master/CONTRIBUTING.md .\r\nPlease be sure to check out our code of conduct,\r\nhttps://github.com/astropy/astropy/blob/master/CODE_OF_CONDUCT.md . -->\r\n\r\n<!-- If you are new or need to be re-acquainted with Astropy\r\ncontributing workflow, please see\r\nhttp://docs.astropy.org/en/latest/development/workflow/development_workflow.html .\r\nThere is even a practical example at\r\nhttps://docs.astropy.org/en/latest/development/workflow/git_edit_workflow_examples.html#astropy-fix-example . -->\r\n\r\n<!-- Astropy coding style guidelines can be found here:\r\nhttps://docs.astropy.org/en/latest/development/codeguide.html#coding-style-conventions\r\nOur testing infrastructure enforces to follow a subset of the PEP8 to be\r\nfollowed. You can check locally whether your changes have followed these by\r\nrunning the following command:\r\n\r\ntox -e codestyle\r\n\r\n-->\r\n\r\n<!-- Please just have a quick search on GitHub to see if a similar\r\npull request has already been posted.\r\nWe have old closed pull requests that might provide useful code or ideas\r\nthat directly tie in with your pull request. -->\r\n\r\n<!-- We have several automatic features that run when a pull request is open.\r\nThey can appear daunting but do not worry because maintainers will help\r\nyou navigate them, if necessary. -->\r\n\r\n### Description\r\n<!-- Provide a general description of what your pull request does.\r\nComplete the following sentence and add relevant details as you see fit. -->\r\n\r\n<!-- In addition please ensure that the pull request title is descriptive\r\nand allows maintainers to infer the applicable subpackage(s). -->\r\n\r\nThis pull request is to handle mask broadcasting for `NumericArray` converter in `io.votable`.\r\n\r\n<!-- If the pull request closes any open issues you can add this.\r\nIf you replace <Issue Number> with a number, GitHub will automatically link it.\r\nIf this pull request is unrelated to any issues, please remove\r\nthe following line. -->\r\n\r\nFixes #11087 \r\n\r\ncc @druzmieres  ",
      "issue_id": 11087,
      "issue_title": "Saving a VOTable with an array of fixed size to a file only saves first element",
      "issue_body": "### Description\r\n<!-- Provide a general description of the bug. -->\r\nI am trying to add metadata to a VOTable that I'm creating. This metadata corresponds to a tuple of known length. The data seems to be ok when inserted into the table but, when I save the table to a file, only the first element is saved.\r\n\r\n### Expected behavior\r\n<!-- What did you expect to happen. -->\r\nWhat I expect to see is the entire tuple saved in the file. The tuple is `(1.0, 2.0, 3.0)`, what I see when I print `params` is  `[<PARAM ID=\"sampling\" arraysize=\"3\" datatype=\"double\" name=\"sampling\" value=\"(1.0, 2.0, 3.0)\"/>]`\r\n\r\n### Actual behavior\r\n<!-- What actually happened. -->\r\n<!-- Was the output confusing or poorly described? -->\r\nOnly the first element is saved in the file. When I open it, I see: `PARAM` section I see: `<PARAM ID=\"sampling\" arraysize=\"3\" datatype=\"double\" name=\"sampling\" value=\"1\"/>`\r\n\r\n### Steps to Reproduce\r\n<!-- Ideally a code example could be provided so we can run it ourselves. -->\r\n<!-- If you are pasting code, use triple backticks (```) around\r\nyour code snippet. -->\r\n<!-- If necessary, sanitize your screen output to be pasted so you do not\r\nreveal secrets like tokens and passwords. -->\r\n\r\nWhat I'm doing to create the table and add the metadata is the following:\r\n```\r\n# Create a new VOTable file\r\nvotable = VOTableFile()\r\n# Add a resource\r\nresource = Resource(); votable.resources.append(resource)\r\n# Add a table for the spectra (and add the sampling as metadata)\r\nspectra_table = Table(votable); resource.tables.append(spectra_table)\r\n# Add sampling as param\r\nsampling = tuple([1.0, 2.0, 3.0])\r\n# I'm using a list because, later, I could have more than one param\r\nparams = [Param(votable, name='sampling', datatype='double', arraysize='3', value=sampling)]\r\nspectra_table.params.extend(params)\r\n```\r\n\r\nWhen I print `params`, I see `[<PARAM ID=\"sampling\" arraysize=\"3\" datatype=\"double\" name=\"sampling\" value=\"(1.0, 2.0, 3.0)\"/>]` which seems to be ok. However, when I save this to a file: \r\n```\r\nvotable.to_xml('table_with_params.xml')\r\n```\r\nIn the `PARAM` section I see: `<PARAM ID=\"sampling\" arraysize=\"3\" datatype=\"double\" name=\"sampling\" value=\"1\"/>`.\r\n\r\nI tried sampling being a tuple, list and numpy array, but they all lead to the same result. If I use `*` instead of the fixed known length, the entire array is saved.\r\n\r\n### System Details\r\n<!-- Even if you do not think this is necessary, it is useful information for the maintainers.\r\nPlease run the following snippet and paste the output below:\r\nimport platform; print(platform.platform())\r\nimport sys; print(\"Python\", sys.version)\r\nimport numpy; print(\"Numpy\", numpy.__version__)\r\nimport astropy; print(\"astropy\", astropy.__version__)\r\nimport scipy; print(\"Scipy\", scipy.__version__)\r\nimport matplotlib; print(\"Matplotlib\", matplotlib.__version__)\r\n-->\r\nLinux-5.4.0-52-generic-x86_64-with-glibc2.29\r\nPython 3.8.5 (default, Jul 28 2020, 12:59:40) \r\n[GCC 9.3.0]\r\nNumpy 1.18.2\r\nastropy 4.3.dev268+g883a7c4f5\r\nScipy 1.4.1\r\nMatplotlib 3.2.1\r\n",
      "issue_closed_at": "2021-01-05T17:05:25Z",
      "base_commit": "9c79f08d72cdd7982ea990d9b01553ac9c4d657b",
      "changes": [
        {
          "file": "astropy/io/votable/converters.py",
          "type": "function",
          "name": "output",
          "class_name": "Boolean",
          "code": "def output(self, value, mask):\n        if mask:\n            return '?'\n        if value:\n            return 'T'\n        return 'F'"
        }
      ]
    }
  ]
}