{
  "Selected_candidate": {
    "pr_number": 16753,
    "pr_title": "Fix Quantity._collect_factor_and_dimension for exp().",
    "pr_body": "Fixes #16751\r\n\r\nReturn `dim` instead of `dim.name` in error message of `Quantity._collect_factor_and_dimension()` to avoid error if `dim` is actually an expression, such as in `Quantity._collect_factor_and_dimension(1 - exp(u / v))`.\r\n<!-- BEGIN RELEASE NOTES -->\r\nNO ENTRY\r\n<!-- END RELEASE NOTES -->",
    "issue_id": 16751,
    "issue_title": "Bug in Quantity._collect_factor_and_dimension",
    "issue_body": "This should give an error, but it fails in the wrong place:\r\n```python\r\nfrom sympy import S, exp\r\nfrom sympy.physics.units import length, mass, Quantity\r\n\r\nu = Quantity(\"u\")\r\nv = Quantity(\"v\")\r\nu.set_dimension(length)\r\nv.set_dimension(mass)\r\nQuantity._collect_factor_and_dimension(1 - exp(u/v))\r\n```\r\nreturns:\r\n\r\n> AttributeError: 'Mul' object has no attribute 'name'\r\nThis is because the code expects a quantity but encounters an expression in the exponent.",
    "issue_closed_at": "2019-05-01T04:55:02Z",
    "base_commit": "9f6e77b6cf0025201d5d783f24236ca07f82ab99",
    "changes": [
      {
        "file": "sympy/physics/units/quantities.py",
        "type": "function",
        "name": "_collect_factor_and_dimension",
        "class_name": "Quantity",
        "code": "def _collect_factor_and_dimension(expr):\n        \"\"\"Return tuple with factor expression and dimension expression.\"\"\"\n        if isinstance(expr, Quantity):\n            return expr.scale_factor, expr.dimension\n        elif isinstance(expr, Mul):\n            factor = 1\n            dimension = Dimension(1)\n            for arg in expr.args:\n                arg_factor, arg_dim = Quantity._collect_factor_and_dimension(arg)\n                factor *= arg_factor\n                dimension *= arg_dim\n            return factor, dimension\n        elif isinstance(expr, Pow):\n            factor, dim = Quantity._collect_factor_and_dimension(expr.base)\n            exp_factor, exp_dim = Quantity._collect_factor_and_dimension(expr.exp)\n            if exp_dim.is_dimensionless:\n               exp_dim = 1\n            return factor ** exp_factor, dim ** (exp_factor * exp_dim)\n        elif isinstance(expr, Add):\n            factor, dim = Quantity._collect_factor_and_dimension(expr.args[0])\n            for addend in expr.args[1:]:\n                addend_factor, addend_dim = \\\n                    Quantity._collect_factor_and_dimension(addend)\n                if dim != addend_dim:\n                    raise ValueError(\n                        'Dimension of \"{0}\" is {1}, '\n                        'but it should be {2}'.format(\n                            addend, addend_dim.name, dim.name))\n                factor += addend_factor\n            return factor, dim\n        elif isinstance(expr, Derivative):\n            factor, dim = Quantity._collect_factor_and_dimension(expr.args[0])\n            for independent, count in expr.variable_count:\n                ifactor, idim = Quantity._collect_factor_and_dimension(independent)\n                factor /= ifactor**count\n                dim /= idim**count\n            return factor, dim\n        elif isinstance(expr, Function):\n            fds = [Quantity._collect_factor_and_dimension(\n                arg) for arg in expr.args]\n            return (expr.func(*(f[0] for f in fds)),\n                    expr.func(*(d[1] for d in fds)))\n        elif isinstance(expr, Dimension):\n            return 1, expr\n        else:\n            return expr, Dimension(1)"
      }
    ]
  },
  "Justification": "Candidate A has a high structural similarity to the CURRENT bug report, as both involve the `_collect_factor_and_dimension` function within the `Quantity` class in the `sympy` framework. The error messages provided relate to erroneous dimensional calculations, which is the core of the CURRENT bug. The fix in Candidate A focuses directly on improving error handling in cases where the dimensions do not align correctly, which is crucial for addressing the CURRENT issue effectively. Thus, it will likely provide pertinent insights or a direct solution to the problem facing the CURRENT bug report.",
  "instance_id": "sympy__sympy-24213",
  "repo": "sympy/sympy",
  "created_at": "2022-11-03T14:00:09Z",
  "problem_statement": "collect_factor_and_dimension does not detect equivalent dimensions in addition\nCode to reproduce:\r\n```python\r\nfrom sympy.physics import units\r\nfrom sympy.physics.units.systems.si import SI\r\n\r\nv1 = units.Quantity('v1')\r\nSI.set_quantity_dimension(v1, units.velocity)\r\nSI.set_quantity_scale_factor(v1, 2 * units.meter / units.second)\r\n\r\na1 = units.Quantity('a1')\r\nSI.set_quantity_dimension(a1, units.acceleration)\r\nSI.set_quantity_scale_factor(a1, -9.8 * units.meter / units.second**2)\r\n\r\nt1 = units.Quantity('t1')\r\nSI.set_quantity_dimension(t1, units.time)\r\nSI.set_quantity_scale_factor(t1, 5 * units.second)\r\n\r\nexpr1 = a1*t1 + v1\r\nSI._collect_factor_and_dimension(expr1)\r\n```\r\nResults in:\r\n```\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"C:\\Python\\Python310\\lib\\site-packages\\sympy\\physics\\units\\unitsystem.py\", line 179, in _collect_factor_and_dimension\r\n    raise ValueError(\r\nValueError: Dimension of \"v1\" is Dimension(velocity), but it should be Dimension(acceleration*time)\r\n```\n",
  "patch": "diff --git a/sympy/physics/units/unitsystem.py b/sympy/physics/units/unitsystem.py\n--- a/sympy/physics/units/unitsystem.py\n+++ b/sympy/physics/units/unitsystem.py\n@@ -175,7 +175,7 @@ def _collect_factor_and_dimension(self, expr):\n             for addend in expr.args[1:]:\n                 addend_factor, addend_dim = \\\n                     self._collect_factor_and_dimension(addend)\n-                if dim != addend_dim:\n+                if not self.get_dimension_system().equivalent_dims(dim, addend_dim):\n                     raise ValueError(\n                         'Dimension of \"{}\" is {}, '\n                         'but it should be {}'.format(\n"
}