{
    "Selected_candidate": {
        "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]"
            }
        ]
    },
    "Justification": "Candidate B addresses the fitting of compound models, which directly relates to the CURRENT bug involving nested CompoundModels and their expected separability matrix. Both reports discuss behaviors of compound models and their properties, suggesting that knowledge gained through the resolution of Candidate B could be highly relevant for understanding and fixing the CURRENT issue. Moreover, the error message and the complex nature of both issues place them in the same conceptual space, making Candidate B the most useful for debugging the CURRENT bug."
}