{
  "Selected_candidate": {
    "pr_number": 16482,
    "pr_title": "Change return type of sort_key function",
    "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://github.com/blog/1506-closing-issues-via-pull-requests . Please also\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #9608\r\n\r\n#### Brief description of what is fixed or changed\r\nThis function defined in `partitions.py` was returning a list which is not hashable.\r\nSo I changed the return value into a tuple (since they are immutable).\r\nI wrote a test in `test_partitions.py` and it passed when I ran it with `bin/test`.\r\n\r\n#### Other comments\r\n\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\nNO ENTRY\r\n<!-- END RELEASE NOTES -->\r\n",
    "issue_id": 9608,
    "issue_title": "Partition can't be ordered",
    "issue_body": "``` python\nIn [13]: from sympy.combinatorics.partitions import Partition\n\nIn [14]: a = Partition([1, 2, 3], [4])\n\nIn [15]: b = Partition([1, 2], [3, 4])\n\nIn [16]: args = [a, b]\n```\n\nBelow is the Traceback:\n\n```\nIn [17]: args = list(ordered(args, Set._infimum_key))\n---------------------------------------------------------------------------\nTypeError                                 Traceback (most recent call last)\n<ipython-input-17-092459052e8d> in <module>()\n----> 1 args = list(ordered(args, Set._infimum_key))\n\n/home/amit/Desktop/myrepo/sympy/sympy/core/compatibility.pyc in ordered(seq, keys, default, warn)\n    676                     raise ValueError(\n    677                         'not enough keys to break ties: %s' % u)\n--> 678         for v in d[k]:\n    679             yield v\n    680         d.pop(k)\n\n/home/amit/Desktop/myrepo/sympy/sympy/core/compatibility.pyc in ordered(seq, keys, default, warn)\n    676                     raise ValueError(\n    677                         'not enough keys to break ties: %s' % u)\n--> 678         for v in d[k]:\n    679             yield v\n    680         d.pop(k)\n\n/home/amit/Desktop/myrepo/sympy/sympy/core/compatibility.pyc in ordered(seq, keys, default, warn)\n    657         f = keys.pop(0)\n    658         for a in seq:\n--> 659             d[f(a)].append(a)\n    660     else:\n    661         if not default:\n\nTypeError: unhashable type: 'list'\n```\n",
    "issue_closed_at": "2019-03-29T10:43:25Z",
    "base_commit": "4cdcbd914d66733eee1de77d592bbad3d693b049",
    "changes": [
      {
        "file": "sympy/combinatorics/partitions.py",
        "type": "function",
        "name": "sort_key",
        "class_name": "Partition",
        "code": "def sort_key(self, order=None):\n        \"\"\"Return a canonical key that can be used for sorting.\n\n        Ordering is based on the size and sorted elements of the partition\n        and ties are broken with the rank.\n\n        Examples\n        ========\n\n        >>> from sympy.utilities.iterables import default_sort_key\n        >>> from sympy.combinatorics.partitions import Partition\n        >>> from sympy.abc import x\n        >>> a = Partition([1, 2])\n        >>> b = Partition([3, 4])\n        >>> c = Partition([1, x])\n        >>> d = Partition(list(range(4)))\n        >>> l = [d, b, a + 1, a, c]\n        >>> l.sort(key=default_sort_key); l\n        [{{1, 2}}, {{1}, {2}}, {{1, x}}, {{3, 4}}, {{0, 1, 2, 3}}]\n        \"\"\"\n        if order is None:\n            members = self.members\n        else:\n            members = tuple(sorted(self.members,\n                             key=lambda w: default_sort_key(w, order)))\n        return list(map(default_sort_key, (self.size, members, self.rank)))"
      }
    ]
  },
  "Justification": "Candidate E addresses issues directly related to the `partitions` module, which is the context of the CURRENT bug. It deals with similar output behaviors and management of partitions, making it highly relevant for identifying and fixing the bug related to dictionary reuse in the output of the `partitions()` iterator.",
  "instance_id": "sympy__sympy-20154",
  "repo": "sympy/sympy",
  "created_at": "2020-09-26T22:49:04Z",
  "problem_statement": "partitions() reusing the output dictionaries\nThe partitions() iterator in sympy.utilities.iterables reuses the output dictionaries. There is a caveat about it in the docstring. \r\n\r\nI'm wondering if it's really that important for it to do this. It shouldn't be that much of a performance loss to copy the dictionary before yielding it. This behavior is very confusing. It means that something as simple as list(partitions()) will give an apparently wrong result. And it can lead to much more subtle bugs if the partitions are used in a nontrivial way. \n",
  "patch": "diff --git a/sympy/utilities/iterables.py b/sympy/utilities/iterables.py\n--- a/sympy/utilities/iterables.py\n+++ b/sympy/utilities/iterables.py\n@@ -1738,21 +1738,6 @@ def partitions(n, m=None, k=None, size=False):\n     {2: 1, 4: 1}\n     {3: 2}\n \n-    Note that the _same_ dictionary object is returned each time.\n-    This is for speed:  generating each partition goes quickly,\n-    taking constant time, independent of n.\n-\n-    >>> [p for p in partitions(6, k=2)]\n-    [{1: 6}, {1: 6}, {1: 6}, {1: 6}]\n-\n-    If you want to build a list of the returned dictionaries then\n-    make a copy of them:\n-\n-    >>> [p.copy() for p in partitions(6, k=2)]  # doctest: +SKIP\n-    [{2: 3}, {1: 2, 2: 2}, {1: 4, 2: 1}, {1: 6}]\n-    >>> [(M, p.copy()) for M, p in partitions(6, k=2, size=True)]  # doctest: +SKIP\n-    [(3, {2: 3}), (4, {1: 2, 2: 2}), (5, {1: 4, 2: 1}), (6, {1: 6})]\n-\n     References\n     ==========\n \n@@ -1802,9 +1787,9 @@ def partitions(n, m=None, k=None, size=False):\n         keys.append(r)\n     room = m - q - bool(r)\n     if size:\n-        yield sum(ms.values()), ms\n+        yield sum(ms.values()), ms.copy()\n     else:\n-        yield ms\n+        yield ms.copy()\n \n     while keys != [1]:\n         # Reuse any 1's.\n@@ -1842,9 +1827,9 @@ def partitions(n, m=None, k=None, size=False):\n             break\n         room -= need\n         if size:\n-            yield sum(ms.values()), ms\n+            yield sum(ms.values()), ms.copy()\n         else:\n-            yield ms\n+            yield ms.copy()\n \n \n def ordered_partitions(n, m=None, sort=True):\n"
}