{
  "Selected_candidate": {
    "pr_number": 19705,
    "pr_title": "Fix handling of addition and multiplication of Dimensions for get_dimensional_dependencies",
    "pr_body": "#### References to other Issues or PRs\r\nFixes #18738\r\n\r\n#### Brief description of what is fixed or changed\r\nFixes some cases where the method get_dimensional_dependencies of DimensionSystem raises exceptions when handling adding and multiplying dimensions. This addresses issue #18738 (at least the first half dealing with Dimensions, the second comment dealing with Quantities still doesn't work).\r\n\r\nThe following example previously raised an exception and now works properly:\r\n```\r\nIn [3]: from sympy.physics.units import length, mass, acceleration, time, pressure, force\r\nIn [4]: from sympy.physics.units.systems.si import dimsys_SI\r\nIn [5]: dimsys_SI.get_dimensional_dependencies(mass * length / time**2 + force - pressure * length**2)\r\nOut[5]: {'mass': 1, 'length': 1, 'time': -2}\r\n```\r\n\r\n#### Other comments\r\nAs part of the fix, the following two lines needed to be moved from the beginning of get_dimensional_dependencies to the beginning of _get_dimensional_dependencies_from_name:\r\n```\r\n        if isinstance(name, Dimension):\r\n            name = name.name\r\n\r\n        if isinstance(name, str):\r\n            name = Symbol(name)\r\n```\r\n\r\nThis had to be done to allow the recursion of the multiplication and addition operators to work properly. This is what fixed #18738 even though I was just trying to get the addition case above to work.  Additionally, an exception now gets raised if _def_dimensional_dependencies_for_name does not handle the input so that more useful error messages are provided then what is shown in #18738.\r\n\r\nIf the user attempts to add incompatible dimensions and then calls get_dimensional_dependencies, an exception is raised.  For example the following will raise an exception:\r\n\r\n`In [4]: dimsys_SI.get_dimensional_dependencies(mass * length / time**2 + pressure)`\r\n\r\nTests have been added for issue #18738 and for the addition cases listed above.\r\n\r\n#### Release Notes\r\n<!-- BEGIN RELEASE NOTES -->\r\n* physics.units\r\n  * Fixed some dimensional analysis bugs with the addition and multiplication operators.\r\n<!-- END RELEASE NOTES -->",
    "issue_id": 18738,
    "issue_title": "Dimensional analysis: AttributeError in get_dimensional_dependencies with simple example",
    "issue_body": "Hi all, \r\n\r\nI am trying to get a simple example of a dimensional analysis (as described in [this tutorial](https://docs.sympy.org/latest/modules/physics/units/examples.html#dimensional-analysis) ) to work. Unfortunately, an AttributeError is raised inside the method `get_dimensional_dependencies`.\r\n\r\nHere is a minimal example:\r\n```\r\nfrom sympy import init_printing, symbols, sqrt\r\nfrom sympy.physics.units.systems.si import dimsys_SI\r\nfrom sympy.physics.units import length\r\ninit_printing()\r\n\r\na, b = symbols(\"a b\")\r\nc = sqrt(a**2 + b**2)\r\nc_dim = c.subs({a: length, b: length})\r\n\r\nprint(c_dim) \r\n# >> sqrt(2)*Dimension(sqrt(length**2))\r\n\r\ndimsys_SI.equivalent_dims(c_dim, length) \r\n# >> AttributeError: 'NoneType' object has no attribute 'items' \r\n```\r\n\r\nIt seems that `get_dimensional_dependencies` has problems handling non-dimensional terms (i.e. the `sqrt(2)` term in `c_dim` in the above example). Oddly, when I try to do the substitution 'manually', this term is eliminated:\r\n```\r\nc_dim = sqrt(length**2 + length**2)\r\nprint(c_dim) \r\n# >> Dimension(sqrt(length**2))\r\n```\r\n\r\nIs this intended behavior? It does make sense that multiplying a dimension with a dimensionless value should just give back the dimension as result. However, how come the `sqrt(2)`-term is not eliminated when substituting symbols for dimensions using `subs`?\r\n\r\nI hope I haven't just misunderstood the tutorial/docs. Happy for any help or hints. Also, I would be willing to dive into the code if someone would point me into the right direction.\r\n\r\nThis may be related to the bug discussed [here](https://gitter.im/sympy/sympy?at=5dc0c052fb4dab784a65dfb1).\r\n\r\nThe above example was tested with Python 2.7 and 3.7 and sympy 1.5.1.",
    "issue_closed_at": "2020-07-17T11:37:09Z",
    "base_commit": "f46f4488c7cb7aff9666e26ce1bfb980f98e97a7",
    "changes": [
      {
        "file": "sympy/physics/units/dimensions.py",
        "type": "function",
        "name": "dimensional_dependencies",
        "class_name": "DimensionSystem",
        "code": "def dimensional_dependencies(self):\n        return self.args[2]"
      },
      {
        "file": "sympy/physics/units/dimensions.py",
        "type": "function",
        "name": "_get_dimensional_dependencies_for_name",
        "class_name": "DimensionSystem",
        "code": "def _get_dimensional_dependencies_for_name(self, name):\n\n        if name.is_Symbol:\n            # Dimensions not included in the dependencies are considered\n            # as base dimensions:\n            return dict(self.dimensional_dependencies.get(name, {name: 1}))\n\n        if name.is_Number:\n            return {}\n\n        get_for_name = self._get_dimensional_dependencies_for_name\n\n        if name.is_Mul:\n            ret = collections.defaultdict(int)\n            dicts = [get_for_name(i) for i in name.args]\n            for d in dicts:\n                for k, v in d.items():\n                    ret[k] += v\n            return {k: v for (k, v) in ret.items() if v != 0}\n\n        if name.is_Pow:\n            dim = get_for_name(name.base)\n            return {k: v*name.exp for (k, v) in dim.items()}\n\n        if name.is_Function:\n            args = (Dimension._from_dimensional_dependencies(\n                get_for_name(arg)) for arg in name.args)\n            result = name.func(*args)\n\n            if isinstance(result, Dimension):\n                return self.get_dimensional_dependencies(result)\n            elif result.func == name.func:\n                return {}\n            else:\n                return get_for_name(result)"
      },
      {
        "file": "sympy/physics/units/dimensions.py",
        "type": "function",
        "name": "_get_dimensional_dependencies_for_name",
        "class_name": "DimensionSystem",
        "code": "def _get_dimensional_dependencies_for_name(self, name):\n\n        if name.is_Symbol:\n            # Dimensions not included in the dependencies are considered\n            # as base dimensions:\n            return dict(self.dimensional_dependencies.get(name, {name: 1}))\n\n        if name.is_Number:\n            return {}\n\n        get_for_name = self._get_dimensional_dependencies_for_name\n\n        if name.is_Mul:\n            ret = collections.defaultdict(int)\n            dicts = [get_for_name(i) for i in name.args]\n            for d in dicts:\n                for k, v in d.items():\n                    ret[k] += v\n            return {k: v for (k, v) in ret.items() if v != 0}\n\n        if name.is_Pow:\n            dim = get_for_name(name.base)\n            return {k: v*name.exp for (k, v) in dim.items()}\n\n        if name.is_Function:\n            args = (Dimension._from_dimensional_dependencies(\n                get_for_name(arg)) for arg in name.args)\n            result = name.func(*args)\n\n            if isinstance(result, Dimension):\n                return self.get_dimensional_dependencies(result)\n            elif result.func == name.func:\n                return {}\n            else:\n                return get_for_name(result)"
      }
    ]
  },
  "Justification": "Candidate A addresses the general concept of evaluating expressions that may raise exceptions in similar contexts. Although the specific types of evaluations differ, they both highlight issues with unexpected results from the evaluation context. The structural and patch similarities could provide valuable insights, potentially guiding adjustments in the CURRENT bug’s handling of spatial evaluation with `Point2D` and `evaluate(False)`.",
  "instance_id": "sympy__sympy-22714",
  "repo": "sympy/sympy",
  "created_at": "2021-12-19T18:54:36Z",
  "problem_statement": "simpify gives `Imaginary coordinates are not permitted.` with evaluate(False)\n## Issue\r\n`with evaluate(False)` crashes unexpectedly with `Point2D`\r\n\r\n## Code\r\n```python\r\nimport sympy as sp\r\nwith sp.evaluate(False):\r\n  sp.S('Point2D(Integer(1),Integer(2))')\r\n```\r\n\r\n## Error\r\n```\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/core/sympify.py\", line 472, in sympify\r\n    expr = parse_expr(a, local_dict=locals, transformations=transformations, evaluate=evaluate)\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/parsing/sympy_parser.py\", line 1026, in parse_expr\r\n    raise e from ValueError(f\"Error from parse_expr with transformed code: {code!r}\")\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/parsing/sympy_parser.py\", line 1017, in parse_expr\r\n    rv = eval_expr(code, local_dict, global_dict)\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/parsing/sympy_parser.py\", line 911, in eval_expr\r\n    expr = eval(\r\n  File \"<string>\", line 1, in <module>\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/geometry/point.py\", line 912, in __new__\r\n    args = Point(*args, **kwargs)\r\n  File \"/home/avinash/.local/lib/python3.8/site-packages/sympy/geometry/point.py\", line 153, in __new__\r\n    raise ValueError('Imaginary coordinates are not permitted.')\r\nValueError: Imaginary coordinates are not permitted.\r\n```\r\n\r\nHowever, it works without `with evaluate(False)`. Both of following commands work\r\n```python\r\nsp.S('Point2D(Integer(1),Integer(2))')\r\nsp.S('Point2D(Integer(1),Integer(2))', evaluate=False)\r\n```\n",
  "patch": "diff --git a/sympy/geometry/point.py b/sympy/geometry/point.py\n--- a/sympy/geometry/point.py\n+++ b/sympy/geometry/point.py\n@@ -152,7 +152,7 @@ def __new__(cls, *args, **kwargs):\n                         'warn' or 'ignore'.'''))\n         if any(coords[dim:]):\n             raise ValueError('Nonzero coordinates cannot be removed.')\n-        if any(a.is_number and im(a) for a in coords):\n+        if any(a.is_number and im(a).is_zero is False for a in coords):\n             raise ValueError('Imaginary coordinates are not permitted.')\n         if not all(isinstance(a, Expr) for a in coords):\n             raise TypeError('Coordinates must be valid SymPy expressions.')\n"
}