{
  "Selected_candidate": {
    "pr_number": 19091,
    "pr_title": "Fixed tensor contraction problem",
    "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\" (see\r\nhttps://tinyurl.com/auto-closing for more information). Also, please\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #18465 \r\n\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nTensor contractions using `replace_with_arrays` would prematurely contract the data arrays before applying the metric—yielding incorrect results. The solution implemented here is a rather trivial one in that it simply separates out an existing function for doing matrix multiplication with an existing data array with the metric and utilizes it in `Tensor._extract_data` to apply the metric before dummy indices are contracted.\r\n\r\n#### Other comments\r\n\r\nThis is my first pull request with Sympy. Let me know if I should make any adjustments to my code.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* tensor\r\n  * Fixed a bug where tensor contractions in calls to `replace_with_arrays` would fail to apply the metric.\r\n<!-- END RELEASE NOTES -->",
    "issue_id": 18465,
    "issue_title": "Tensor contractions are wrong",
    "issue_body": "This is essentially a generalization of #17328.\r\n\r\nThe problem in the current implementation is that contractions are handled before applications of the metric, which leads to incorrect results such as in #17328.\r\n\r\nIn `tensor/tensor.py`:\r\n```python\r\nclass Tensor(TensExpr):\r\n# ...\r\n    def _extract_data(self, replacement_dict):\r\n    # ...\r\n        if len(dum1) > 0:\r\n            indices2 = other.get_indices()\r\n            repl = {}\r\n            for p1, p2 in dum1:\r\n                repl[indices2[p2]] = -indices2[p1]\r\n            other = other.xreplace(repl).doit()\r\n            array = _TensorDataLazyEvaluator.data_contract_dum([array], dum1, len(indices2))\r\n\r\n        free_ind1 = self.get_free_indices()\r\n        free_ind2 = other.get_free_indices()\r\n\r\n        return self._match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict)\r\n```\r\nAnd thus, the issue is that `_TensorDataLazyEvaluator.data_contract_dum` is being called prior to `self._match_indices_with_other_tensor` (where the metric is applied).\r\n\r\nThe reason that this ordering matters is because tensor contraction is itself the abstraction of applying the metric to the tensors that represent psuedo-riemannian manifolds. In essence, it means that we must have it that ![equation](https://latex.codecogs.com/svg.latex?T^\\mu_\\mu=g_{\\mu\\nu}T^{\\mu\\nu}); however, this isn't the case here.\r\n\r\nI've tried tampering with the code above, but by the way tensors have been designed, this bug is essentially unavoidable. As a consequence, the tensor module needs to be refactored in order to get accurate results. (Also, I couldn't help but notice that the last argument to `_TensorDataLazyEvaluator.data_contract_dum` isn't used).\r\n\r\n@drybalka had mentioned that he had this sort of refactoring in the works, but based on his fork, progress seems to be slow. I think discussions should be in order for reorganizing how tensors actually represent their components in this module.",
    "issue_closed_at": "2020-04-20T15:15:50Z",
    "base_commit": "64d28fe0534f6993695d11244ea740f783958dc8",
    "changes": [
      {
        "file": "sympy/tensor/tensor.py",
        "type": "function",
        "name": "recursor",
        "class_name": "TensExpr",
        "code": "def recursor(expr, pos):\n            if isinstance(expr, TensorIndex):\n                yield (expr, pos)\n            elif isinstance(expr, (Tuple, TensExpr)):\n                for p, arg in enumerate(expr.args):\n                    for i in recursor(arg, pos+(p,)):\n                        yield i"
      },
      {
        "file": "sympy/tensor/tensor.py",
        "type": "function",
        "name": "_match_indices_with_other_tensor",
        "class_name": "TensExpr",
        "code": "def _match_indices_with_other_tensor(array, free_ind1, free_ind2, replacement_dict):\n        from .array import tensorcontraction, tensorproduct, permutedims\n\n        index_types1 = [i.tensor_index_type for i in free_ind1]\n\n        # Check if variance of indices needs to be fixed:\n        pos2up = []\n        pos2down = []\n        free2remaining = free_ind2[:]\n        for pos1, index1 in enumerate(free_ind1):\n            if index1 in free2remaining:\n                pos2 = free2remaining.index(index1)\n                free2remaining[pos2] = None\n                continue\n            if -index1 in free2remaining:\n                pos2 = free2remaining.index(-index1)\n                free2remaining[pos2] = None\n                free_ind2[pos2] = index1\n                if index1.is_up:\n                    pos2up.append(pos2)\n                else:\n                    pos2down.append(pos2)\n            else:\n                index2 = free2remaining[pos1]\n                if index2 is None:\n                    raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n                free2remaining[pos1] = None\n                free_ind2[pos1] = index1\n                if index1.is_up ^ index2.is_up:\n                    if index1.is_up:\n                        pos2up.append(pos1)\n                    else:\n                        pos2down.append(pos1)\n\n        if len(set(free_ind1) & set(free_ind2)) < len(free_ind1):\n            raise ValueError(\"incompatible indices: %s and %s\" % (free_ind1, free_ind2))\n\n        # TODO: add possibility of metric after (spinors)\n        def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)\n\n        # Raise indices:\n        for pos in pos2up:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            metric_inverse = _TensorDataLazyEvaluator.inverse_matrix(metric)\n            array = contract_and_permute(metric_inverse, array, pos)\n        # Lower indices:\n        for pos in pos2down:\n            index_type_pos = index_types1[pos]  # type: TensorIndexType\n            if index_type_pos not in replacement_dict:\n                raise ValueError(\"No metric provided to lower index\")\n            metric = replacement_dict[index_type_pos]\n            array = contract_and_permute(metric, array, pos)\n\n        if free_ind1:\n            permutation = TensExpr._get_indices_permutation(free_ind2, free_ind1)\n            array = permutedims(array, permutation)\n\n        if hasattr(array, \"rank\") and array.rank() == 0:\n            array = array[()]\n\n        return free_ind2, array"
      },
      {
        "file": "sympy/tensor/tensor.py",
        "type": "function",
        "name": "contract_and_permute",
        "class_name": "TensExpr",
        "code": "def contract_and_permute(metric, array, pos):\n            array = tensorcontraction(tensorproduct(metric, array), (1, 2+pos))\n            permu = list(range(len(free_ind1)))\n            permu[0], permu[pos] = permu[pos], permu[0]\n            return permutedims(array, permu)"
      },
      {
        "file": "sympy/tensor/tensor.py",
        "type": "function",
        "name": "_extract_data",
        "class_name": "TensorElement",
        "code": "def _extract_data(self, replacement_dict):\n        ret_indices, array = self.expr._extract_data(replacement_dict)\n        index_map = self.index_map\n        slice_tuple = tuple(index_map.get(i, slice(None)) for i in ret_indices)\n        ret_indices = [i for i in ret_indices if i not in index_map]\n        array = array.__getitem__(slice_tuple)\n        return ret_indices, array"
      }
    ]
  },
  "Justification": "Candidate A is most useful because it involves Tensor operations similar to the CURRENT bug report, which also focuses on Tensor computations. Both reports deal with expansion and manipulation of tensor structures, and the fixes proposed in Candidate A are in the same module (tensor/tensor.py) as the CURRENT bug's operations. Additionally, the Candidate A bug report addresses issues with order and processing in tensor applications which is highly relevant to the current issue of incorrect behavior during tensor expansion.",
  "instance_id": "sympy__sympy-24152",
  "repo": "sympy/sympy",
  "created_at": "2022-10-21T13:47:03Z",
  "problem_statement": "Bug in expand of TensorProduct + Workaround + Fix\n### Error description\r\nThe expansion of a TensorProduct object stops incomplete if summands in the tensor product factors have (scalar) factors, e.g.\r\n```\r\nfrom sympy import *\r\nfrom sympy.physics.quantum import *\r\nU = Operator('U')\r\nV = Operator('V')\r\nP = TensorProduct(2*U - V, U + V)\r\nprint(P) \r\n# (2*U - V)x(U + V)\r\nprint(P.expand(tensorproduct=True)) \r\n#result: 2*Ux(U + V) - Vx(U + V) #expansion has missed 2nd tensor factor and is incomplete\r\n```\r\nThis is clearly not the expected behaviour. It also effects other functions that rely on .expand(tensorproduct=True), as e.g. qapply() .\r\n\r\n### Work around\r\nRepeat .expand(tensorproduct=True) as may times as there are tensor factors, resp. until the expanded term does no longer change. This is however only reasonable in interactive session and not in algorithms.\r\n\r\n### Code Fix\r\n.expand relies on the method TensorProduct._eval_expand_tensorproduct(). The issue arises from an inprecise check in TensorProduct._eval_expand_tensorproduct() whether a recursive call is required; it fails when the creation of a TensorProduct object returns commutative (scalar) factors up front: in that case the constructor returns a Mul(c_factors, TensorProduct(..)).\r\nI thus propose the following  code fix in TensorProduct._eval_expand_tensorproduct() in quantum/tensorproduct.py.  I have marked the four lines to be added / modified:\r\n```\r\n    def _eval_expand_tensorproduct(self, **hints):\r\n                ...\r\n                for aa in args[i].args:\r\n                    tp = TensorProduct(*args[:i] + (aa,) + args[i + 1:])\r\n                    c_part, nc_part = tp.args_cnc() #added\r\n                    if len(nc_part)==1 and isinstance(nc_part[0], TensorProduct): #modified\r\n                        nc_part = (nc_part[0]._eval_expand_tensorproduct(), ) #modified\r\n                    add_args.append(Mul(*c_part)*Mul(*nc_part)) #modified\r\n                break\r\n                ...\r\n```\r\nThe fix splits of commutative (scalar) factors from the tp returned. The TensorProduct object will be the one nc factor in nc_part (see TensorProduct.__new__ constructor), if any. Note that the constructor will return 0 if a tensor factor is 0, so there is no guarantee that tp contains a TensorProduct object (e.g. TensorProduct(U-U, U+V).\r\n\r\n\r\n\n",
  "patch": "diff --git a/sympy/physics/quantum/tensorproduct.py b/sympy/physics/quantum/tensorproduct.py\n--- a/sympy/physics/quantum/tensorproduct.py\n+++ b/sympy/physics/quantum/tensorproduct.py\n@@ -246,9 +246,12 @@ def _eval_expand_tensorproduct(self, **hints):\n             if isinstance(args[i], Add):\n                 for aa in args[i].args:\n                     tp = TensorProduct(*args[:i] + (aa,) + args[i + 1:])\n-                    if isinstance(tp, TensorProduct):\n-                        tp = tp._eval_expand_tensorproduct()\n-                    add_args.append(tp)\n+                    c_part, nc_part = tp.args_cnc()\n+                    # Check for TensorProduct object: is the one object in nc_part, if any:\n+                    # (Note: any other object type to be expanded must be added here)\n+                    if len(nc_part) == 1 and isinstance(nc_part[0], TensorProduct):\n+                        nc_part = (nc_part[0]._eval_expand_tensorproduct(), )\n+                    add_args.append(Mul(*c_part)*Mul(*nc_part))\n                 break\n \n         if add_args:\n"
}