{
  "instance_id": "sympy__sympy-23117",
  "repo": "sympy/sympy",
  "created_at": "2022-02-19T13:15:18Z",
  "problem_statement": "sympy.Array([]) fails, while sympy.Matrix([]) works\nSymPy 1.4 does not allow to construct empty Array (see code below). Is this the intended behavior?\r\n\r\n```\r\n>>> import sympy\r\nKeyboardInterrupt\r\n>>> import sympy\r\n>>> from sympy import Array\r\n>>> sympy.__version__\r\n'1.4'\r\n>>> a = Array([])\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"/Users/hcui7/miniconda3/envs/a/lib/python3.7/site-packages/sympy/tensor/array/dense_ndim_array.py\", line 130, in __new__\r\n    return cls._new(iterable, shape, **kwargs)\r\n  File \"/Users/hcui7/miniconda3/envs/a/lib/python3.7/site-packages/sympy/tensor/array/dense_ndim_array.py\", line 136, in _new\r\n    shape, flat_list = cls._handle_ndarray_creation_inputs(iterable, shape, **kwargs)\r\n  File \"/Users/hcui7/miniconda3/envs/a/lib/python3.7/site-packages/sympy/tensor/array/ndim_array.py\", line 142, in _handle_ndarray_creation_inputs\r\n    iterable, shape = cls._scan_iterable_shape(iterable)\r\n  File \"/Users/hcui7/miniconda3/envs/a/lib/python3.7/site-packages/sympy/tensor/array/ndim_array.py\", line 127, in _scan_iterable_shape\r\n    return f(iterable)\r\n  File \"/Users/hcui7/miniconda3/envs/a/lib/python3.7/site-packages/sympy/tensor/array/ndim_array.py\", line 120, in f\r\n    elems, shapes = zip(*[f(i) for i in pointer])\r\nValueError: not enough values to unpack (expected 2, got 0)\r\n```\r\n\r\n@czgdp1807 \n",
  "patch": "diff --git a/sympy/tensor/array/ndim_array.py b/sympy/tensor/array/ndim_array.py\n--- a/sympy/tensor/array/ndim_array.py\n+++ b/sympy/tensor/array/ndim_array.py\n@@ -145,10 +145,12 @@ def __new__(cls, iterable, shape=None, **kwargs):\n \n     def _parse_index(self, index):\n         if isinstance(index, (SYMPY_INTS, Integer)):\n-            raise ValueError(\"Only a tuple index is accepted\")\n+            if index >= self._loop_size:\n+                raise ValueError(\"Only a tuple index is accepted\")\n+            return index\n \n         if self._loop_size == 0:\n-            raise ValueError(\"Index not valide with an empty array\")\n+            raise ValueError(\"Index not valid with an empty array\")\n \n         if len(index) != self._rank:\n             raise ValueError('Wrong number of array axes')\n@@ -194,6 +196,9 @@ def f(pointer):\n             if not isinstance(pointer, Iterable):\n                 return [pointer], ()\n \n+            if len(pointer) == 0:\n+                return [], (0,)\n+\n             result = []\n             elems, shapes = zip(*[f(i) for i in pointer])\n             if len(set(shapes)) != 1:\n@@ -567,11 +572,11 @@ def _check_special_bounds(cls, flat_list, shape):\n \n     def _check_index_for_getitem(self, index):\n         if isinstance(index, (SYMPY_INTS, Integer, slice)):\n-            index = (index, )\n+            index = (index,)\n \n         if len(index) < self.rank():\n-            index = tuple([i for i in index] + \\\n-                          [slice(None) for i in range(len(index), self.rank())])\n+            index = tuple(index) + \\\n+                          tuple(slice(None) for i in range(len(index), self.rank()))\n \n         if len(index) > self.rank():\n             raise ValueError('Dimension of index greater than rank of array')\n",
  "similar_bug_items": [
    {
      "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)"
        }
      ]
    },
    {
      "pr_number": 22973,
      "pr_title": "Add convenience methods for constructing `AlgebraicField`",
      "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\n\r\nFixes #22954 \r\n\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nAdds a couple of convenience methods for constructing `AlgebraicField`.\r\n\r\nServes users who don't care about the complex embedding, by choosing an embedding for them.\r\n\r\nIn particular, makes constructing a cyclotomic field much easier.\r\n\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 between the BEGIN and END\r\nstatements. The basic format is a bulleted list with the name of the subpackage\r\nand the release note for this PR. For example:\r\n\r\n* solvers\r\n  * Added a new solver for logarithmic equations.\r\n\r\n* functions\r\n  * Fixed a bug with log of integers.\r\n\r\nor if no release note(s) should be included use:\r\n\r\nNO ENTRY\r\n\r\nSee https://github.com/sympy/sympy/wiki/Writing-Release-Notes for more\r\ninformation on how to write release notes. The bot will check your release\r\nnotes automatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* polys\r\n  * Add convenience constructors for `AlgebraicField`\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 22954,
      "issue_title": "polys/numberfields: Convenience constructors for `AlgebraicField`",
      "issue_body": "I'd like to propose a pair of new constructor methods, to support a couple of common use cases.\r\n\r\n\r\n## Use Case 1\r\n\r\n* You want a number field `Q(a)`,\r\n* AND you have the minimal polynomial `f(x)` for `a`,\r\n* AND you don't care _which_ root of `f` `a` is. (E.g. it's a Galois field, so it doesn't matter.),\r\n* AND, OPTIONALLY you want an `alias` for `a` so that field elements print nicely.\r\n\r\n\r\n## Use Case 2\r\n\r\n* You want the `n`th cyclotomic field.\r\n* You probably want the primitive element to be aliased either `\"zeta\"` or `f\"zeta{n}\"`.\r\n\r\n\r\n\r\n## The Current Situation\r\n\r\nFor Use Case 1, you have to do this:\r\n```python\r\nalpha = AlgebraicNumber(CRootOf(f, -1), alias=\"alpha\")\r\nK = QQ.algebraic_field(alpha)\r\n\r\n```\r\nwhich has several problems:\r\n\r\n* It's too long.\r\n* You shouldn't have to import `CRootOf`.\r\n* You shouldn't have to choose an index for `CRootOf`.\r\n\r\n\r\nUse Case 2 is similar, and you also have to import `cyclotomic_poly`:\r\n\r\n```python\r\nzeta = AlgebraicNumber(CRootOf(cyclotomic_poly(19), -1), alias=\"zeta\")\r\nK = QQ.algebraic_field(zeta)\r\n```\r\n\r\n\r\n\r\n## Proposal\r\n\r\nAlongside the `QQ.algebraic_field()` method (which takes one arg, being an `Expr`) I would propose two additional methods:\r\n\r\n```python\r\n    def alg_field_from_poly(self, poly, alias=None, root_index=-1):\r\n        alpha = AlgebraicNumber(CRootOf(poly, root_index), alias=alias)\r\n        return self.algebraic_field(alpha)\r\n\r\n    def cyclotomic_field(self, n, alias=\"zeta\", ss=False, gen=None, root_index=-1):\r\n        if ss:\r\n            alias += str(n)\r\n        return self.alg_field_from_poly(cyclotomic_poly(n, gen), alias=alias, root_index=root_index)\r\n\r\n```\r\n\r\nThen the above examples become\r\n\r\n```python\r\nK = QQ.alg_field_from_poly(f, \"alpha\")\r\n```\r\n\r\nand\r\n\r\n```python\r\nK = QQ.cyclotomic_field(19)\r\n```\r\n\r\n## Rationale\r\n\r\nFor a \"canonical\" default root, `-1` is the best index to pass to `CRootOf`. In common cases like quadratic or cyclotomic fields, this will select the root you tend to think of: the positive square root, or the square root with positive imaginary part, or `exp(2*I*pi/n)` for the `n`th cyclotomic field.\r\n",
      "issue_closed_at": "2022-02-01T23:03:24Z",
      "base_commit": "c1bfbd69158c4a97ce605a14ee5a00287469e01a",
      "changes": [
        {
          "file": "sympy/polys/domains/domain.py",
          "type": "line",
          "name": "line 3",
          "code": "\nfrom typing import Any, Optional, Type\n\nfrom sympy.core import Basic, sympify\nfrom sympy.core.sorting import default_sort_key, ordered\nfrom sympy.external.gmpy import HAS_GMPY"
        },
        {
          "file": "sympy/polys/domains/domain.py",
          "type": "function",
          "name": "algebraic_field",
          "class_name": "Domain",
          "code": "def algebraic_field(self, *extension):\n        r\"\"\"Returns an algebraic field, i.e. `K(\\alpha, \\ldots)`. \"\"\"\n        raise DomainError(\"Cannot create algebraic field over %s\" % self)"
        },
        {
          "file": "sympy/polys/numberfields/basis.py",
          "type": "function",
          "name": "round_two",
          "class_name": null,
          "code": "def round_two(T, radicals=None):\n    r\"\"\"\n    Zassenhaus's \"Round 2\" algorithm.\n\n    Explanation\n    ===========\n\n    Carry out Zassenhaus's \"Round 2\" algorithm on a monic irreducible\n    polynomial *T* over :ref:`ZZ`. This computes an integral basis and the\n    discriminant for the field $K = \\mathbb{Q}[x]/(T(x))$.\n\n    Ordinarily this function need not be called directly, as one can instead\n    access the :py:meth:`~.AlgebraicField.maximal_order`,\n    :py:meth:`~.AlgebraicField.integral_basis`, and\n    :py:meth:`~.AlgebraicField.discriminant` methods of an\n    :py:class:`~.AlgebraicField`.\n\n    Examples\n    ========\n\n    Working through an AlgebraicField:\n\n    >>> from sympy import Poly, QQ\n    >>> from sympy.abc import x, theta\n    >>> T = Poly(x ** 3 + x ** 2 - 2 * x + 8)\n    >>> K = QQ.algebraic_field((T, theta))\n    >>> print(K.maximal_order())\n    Submodule[[2, 0, 0], [0, 2, 0], [0, 1, 1]]/2\n    >>> print(K.discriminant())\n    -503\n    >>> print(K.integral_basis(fmt='sympy'))\n    [1, theta, theta**2/2 + theta/2]\n\n    Calling directly:\n\n    >>> from sympy import Poly\n    >>> from sympy.abc import x\n    >>> from sympy.polys.numberfields.basis import round_two\n    >>> T = Poly(x ** 3 + x ** 2 - 2 * x + 8)\n    >>> print(round_two(T))\n    (Submodule[[2, 0, 0], [0, 2, 0], [0, 1, 1]]/2, -503)\n\n    The nilradicals mod $p$ that are sometimes computed during the Round Two\n    algorithm may be useful in further calculations. Pass a dictionary under\n    `radicals` to receive these:\n\n    >>> T = Poly(x**3 + 3*x**2 + 5)\n    >>> rad = {}\n    >>> ZK, dK = round_two(T, radicals=rad)\n    >>> print(rad)\n    {3: Submodule[[-1, 1, 0], [-1, 0, 1]]}\n\n    Parameters\n    ==========\n\n    T : :py:class:`~.Poly`\n        The irreducible monic polynomial over :ref:`ZZ` defining the number\n        field.\n\n    radicals : dict, optional\n        This is a way for any $p$-radicals (if computed) to be returned by\n        reference. If desired, pass an empty dictionary. If the algorithm\n        reaches the point where it computes the nilradical mod $p$ of the ring\n        of integers $Z_K$, then an $\\mathbb{F}_p$-basis for this ideal will be\n        stored in this dictionary under the key ``p``. This can be useful for\n        other algorithms, such as prime decomposition.\n\n    Returns\n    =======\n\n    Pair ``(ZK, dK)``, where:\n\n        ``ZK`` is a :py:class:`~sympy.polys.numberfields.modules.Submodule`\n        representing the maximal order.\n\n        ``dK`` is the discriminant of the field $K = \\mathbb{Q}[x]/(T(x))$.\n\n    See Also\n    ========\n\n    .AlgebraicField.maximal_order\n    .AlgebraicField.integral_basis\n    .AlgebraicField.discriminant\n\n    References\n    ==========\n\n    .. [1] Cohen, H. *A Course in Computational Algebraic Number Theory.*\n\n    \"\"\"\n    if T.domain == QQ:\n        try:\n            T = Poly(T, domain=ZZ)\n        except CoercionFailed:\n            pass  # Let the error be raised by the next clause.\n    if (   not T.is_univariate\n        or not T.is_irreducible\n        or not T.is_monic\n        or not T.domain == ZZ):\n        raise ValueError('Round 2 requires a monic irreducible univariate polynomial over ZZ.')\n    n = T.degree()\n    D = T.discriminant()\n    D_modulus = ZZ.from_sympy(abs(D))\n    # D must be 0 or 1 mod 4 (see Cohen Sec 4.4), which ensures we can write\n    # it in the form D = D_0 * F**2, where D_0 is 1 or a fundamental discriminant.\n    _, F = extract_fundamental_discriminant(D)\n    Ztheta = PowerBasis(T)\n    H = Ztheta.whole_submodule()\n    nilrad = None\n    while F:\n        # Next prime:\n        p, e = F.popitem()\n        U_bar, m = _apply_Dedekind_criterion(T, p)\n        if m == 0:\n            continue\n        # For a given prime p, the first enlargement of the order spanned by\n        # the current basis can be done in a simple way:\n        U = Ztheta.element_from_poly(Poly(U_bar, domain=ZZ))\n        # TODO:\n        #  Theory says only first m columns of the U//p*H term below are needed.\n        #  Could be slightly more efficient to use only those. Maybe `Submodule`\n        #  class should support a slice operator?\n        H = H.add(U // p * H, hnf_modulus=D_modulus)\n        if e <= m:\n            continue\n        # A second, and possibly more, enlargements for p will be needed.\n        # These enlargements require a more involved procedure.\n        q = p\n        while q < n:\n            q *= p\n        H1, nilrad = _second_enlargement(H, p, q)\n        while H1 != H:\n            H = H1\n            H1, nilrad = _second_enlargement(H, p, q)\n    # Note: We do not store all nilradicals mod p, only the very last. This is\n    # because, unless computed against the entire integral basis, it might not\n    # be accurate. (In other words, if H was not already equal to ZK when we\n    # passed it to `_second_enlargement`, then we can't trust the nilradical\n    # so computed.) Example: if T(x) = x ** 3 + 15 * x ** 2 - 9 * x + 13, then\n    # F is divisible by 2, 3, and 7, and the nilradical mod 2 as computed above\n    # will not be accurate for the full, maximal order ZK.\n    if nilrad is not None and isinstance(radicals, dict):\n        radicals[p] = nilrad\n    ZK = H\n    # Pre-set expensive boolean properties which we already know to be true:\n    ZK._starts_with_unity = True\n    ZK._is_sq_maxrank_HNF = True\n    dK = (D * ZK.matrix.det() ** 2) // ZK.denom ** (2 * n)\n    return ZK, dK"
        },
        {
          "file": "sympy/polys/numberfields/primes.py",
          "type": "function",
          "name": "prime_valuation",
          "class_name": null,
          "code": "def prime_valuation(I, P):\n    r\"\"\"\n    Compute the *P*-adic valuation for an integral ideal *I*.\n\n    Examples\n    ========\n\n    >>> from sympy import QQ\n    >>> from sympy.abc import theta\n    >>> from sympy.polys import cyclotomic_poly\n    >>> from sympy.polys.numberfields import prime_valuation\n    >>> T = cyclotomic_poly(5)\n    >>> K = QQ.algebraic_field((T, theta))\n    >>> P = K.primes_above(5)\n    >>> ZK = K.maximal_order()\n    >>> print(prime_valuation(25*ZK, P[0]))\n    8\n\n    Parameters\n    ==========\n\n    I : :py:class:`~.Submodule`\n        An integral ideal whose valuation is desired.\n\n    P : :py:class:`~.PrimeIdeal`\n        The prime at which to compute the valuation.\n\n    Returns\n    =======\n\n    int\n\n    See Also\n    ========\n\n    .PrimeIdeal.valuation\n\n    References\n    ==========\n\n    .. [1] Cohen, H. *A Course in Computational Algebraic Number Theory.*\n       (See Algorithm 4.8.17.)\n\n    \"\"\"\n    p, ZK = P.p, P.ZK\n    n, W, d = ZK.n, ZK.matrix, ZK.denom\n\n    A = W.convert_to(QQ).inv() * I.matrix * d / I.denom\n    # Although A must have integer entries, given that I is an integral ideal,\n    # as a DomainMatrix it will still be over QQ, so we convert back:\n    A = A.convert_to(ZZ)\n    D = A.det()\n    if D % p != 0:\n        return 0\n\n    beta = P.test_factor()\n\n    f = d ** n // W.det()\n    need_complete_test = (f % p == 0)\n    v = 0\n    while True:\n        # Entering the loop, the cols of A represent lin combs of omegas.\n        # Turn them into lin combs of thetas:\n        A = W * A\n        # And then one column at a time...\n        for j in range(n):\n            c = ZK.parent(A[:, j], denom=d)\n            c *= beta\n            # ...turn back into lin combs of omegas, after multiplying by beta:\n            c = ZK.represent(c).flat()\n            for i in range(n):\n                A[i, j] = c[i]\n        if A[n - 1, n - 1].element % p != 0:\n            break\n        A = A / p\n        # As noted above, domain converts to QQ even when division goes evenly.\n        # So must convert back, even when we don't \"need_complete_test\".\n        if need_complete_test:\n            # In this case, having a non-integer entry is actually just our\n            # halting condition.\n            try:\n                A = A.convert_to(ZZ)\n            except CoercionFailed:\n                break\n        else:\n            # In this case theory says we should not have any non-integer entries.\n            A = A.convert_to(ZZ)\n        v += 1\n    return v"
        }
      ]
    },
    {
      "pr_number": 13441,
      "pr_title": "address issues with _commutative_match and _combine_inverse",
      "pr_body": "fixes #10979 (as best as I can see how); I suspect the slowdown is coming from the `_should_evalf` procedure rather than `count_ops`.",
      "issue_id": 10979,
      "issue_title": "count_ops is slow for large expressions",
      "issue_body": "It seems that this script was hanging inside `count_ops`:\n\n```\nmoorepants@garuda:pydy.wiki(master)$ SYMPY_CACHE_SIZE=10000 ipython\nPython 3.5.1 |Continuum Analytics, Inc.| (default, Dec  7 2015, 11:16:01) \nType \"copyright\", \"credits\" or \"license\" for more information.\n\nIPython 4.1.2 -- An enhanced Interactive Python.\n?         -> Introduction and overview of IPython's features.\n%quickref -> Quick reference.\nhelp      -> Python's own help system.\nobject?   -> Details about 'object', use 'object??' for extra details.\n\nIn [1]: %paste\n   In [1]: from pydy.models import n_link_pendulum_on_cart\n\n   In [2]: sys = n_link_pendulum_on_cart(3)\n\n   In [3]: x_dot = sys.eom_method.rhs()\n\n   In [4]: %time jac = x_dot.jacobian(sys.states)\n## -- End pasted text --\nCPU times: user 2.2 s, sys: 4 ms, total: 2.21 s\nWall time: 2.2 s\n\nIn [2]: %paste\n   In [5]: sys = n_link_pendulum_on_cart(4)\n\n   In [6]: x_dot = sys.eom_method.rhs()\n\n   In [7]: %time jac = x_dot.jacobian(sys.states)\n## -- End pasted text --\n^C---------------------------------------------------------------------------\nKeyboardInterrupt                         Traceback (most recent call last)\n<ipython-input-2-1039ec729c05> in <module>()\n      3 x_dot = sys.eom_method.rhs()\n      4 \n----> 5 get_ipython().magic('time jac = x_dot.jacobian(sys.states)')\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/interactiveshell.py in magic(self, arg_s)\n   2161         magic_name, _, magic_arg_s = arg_s.partition(' ')\n   2162         magic_name = magic_name.lstrip(prefilter.ESC_MAGIC)\n-> 2163         return self.run_line_magic(magic_name, magic_arg_s)\n   2164 \n   2165     #-------------------------------------------------------------------------\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/interactiveshell.py in run_line_magic(self, magic_name, line)\n   2082                 kwargs['local_ns'] = sys._getframe(stack_depth).f_locals\n   2083             with self.builtin_trap:\n-> 2084                 result = fn(*args,**kwargs)\n   2085             return result\n   2086 \n\n<decorator-gen-60> in time(self, line, cell, local_ns)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/magic.py in <lambda>(f, *a, **k)\n    191     # but it's overkill for just that one bit of state.\n    192     def magic_deco(arg):\n--> 193         call = lambda f, *a, **k: f(*a, **k)\n    194 \n    195         if callable(arg):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/IPython/core/magics/execution.py in time(self, line, cell, local_ns)\n   1175         else:\n   1176             st = clock2()\n-> 1177             exec(code, glob, local_ns)\n   1178             end = clock2()\n   1179             out = None\n\n<timed exec> in <module>()\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in jacobian(self, X)\n   1551         # m is the number of functions and n is the number of variables\n   1552         # computing the Jacobian is now easy:\n-> 1553         return self._new(m, n, lambda j, i: self[j].diff(X[i]))\n   1554 \n   1555     def QRdecomposition(self):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/dense.py in _new(cls, *args, **kwargs)\n    601     @classmethod\n    602     def _new(cls, *args, **kwargs):\n--> 603         rows, cols, flat_list = cls._handle_creation_inputs(*args, **kwargs)\n    604         self = object.__new__(cls)\n    605         self.rows = rows\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in _handle_creation_inputs(cls, *args, **kwargs)\n    207                     flat_list.extend(\n    208                         [cls._sympify(op(cls._sympify(i), cls._sympify(j)))\n--> 209                         for j in range(cols)])\n    210 \n    211             # Matrix(2, 2, [1, 2, 3, 4])\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in <listcomp>(.0)\n    207                     flat_list.extend(\n    208                         [cls._sympify(op(cls._sympify(i), cls._sympify(j)))\n--> 209                         for j in range(cols)])\n    210 \n    211             # Matrix(2, 2, [1, 2, 3, 4])\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/matrices/matrices.py in <lambda>(j, i)\n   1551         # m is the number of functions and n is the number of variables\n   1552         # computing the Jacobian is now easy:\n-> 1553         return self._new(m, n, lambda j, i: self[j].diff(X[i]))\n   1554 \n   1555     def QRdecomposition(self):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/power.py in _eval_derivative(self, s)\n    982     def _eval_derivative(self, s):\n    983         from sympy import log\n--> 984         dbase = self.base.diff(s)\n    985         dexp = self.exp.diff(s)\n    986         return self * (dexp * log(self.base) + dbase * self.exp/self.base)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in _eval_derivative(self, s)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in <listcomp>(.0)\n    351     @cacheit\n    352     def _eval_derivative(self, s):\n--> 353         return self.func(*[a.diff(s) for a in self.args])\n    354 \n    355     def _eval_nseries(self, x, n, logx):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/mul.py in _eval_derivative(self, s)\n    832         terms = []\n    833         for i in range(len(args)):\n--> 834             d = args[i].diff(s)\n    835             if d:\n    836                 terms.append(self.func(*(args[:i] + [d] + args[i + 1:])))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in diff(self, *symbols, **assumptions)\n   2864         new_symbols = list(map(sympify, symbols))  # e.g. x, 2, y, z\n   2865         assumptions.setdefault(\"evaluate\", True)\n-> 2866         return Derivative(self, *new_symbols, **assumptions)\n   2867 \n   2868     ###########################################################################\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, expr, *variables, **assumptions)\n   1141                     old_v = v\n   1142                     v = new_v\n-> 1143                 obj = expr._eval_derivative(v)\n   1144                 nderivs += 1\n   1145                 if not is_symbol:\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/power.py in _eval_derivative(self, s)\n    984         dbase = self.base.diff(s)\n    985         dexp = self.exp.diff(s)\n--> 986         return self * (dexp * log(self.base) + dbase * self.exp/self.base)\n    987 \n    988     def _eval_evalf(self, prec):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in __new__(cls, *args, **options)\n    388 \n    389         pr = max(cls._should_evalf(a) for a in result.args)\n--> 390         pr2 = min(cls._should_evalf(a) for a in result.args)\n    391         if pr2 > 0:\n    392             return result.evalf(mlib.libmpf.prec_to_dps(pr))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in <genexpr>(.0)\n    388 \n    389         pr = max(cls._should_evalf(a) for a in result.args)\n--> 390         pr2 = min(cls._should_evalf(a) for a in result.args)\n    391         if pr2 > 0:\n    392             return result.evalf(mlib.libmpf.prec_to_dps(pr))\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in _should_evalf(cls, arg)\n    411         # Don't use as_real_imag() here, that's too much work\n    412         a, b = Wild('a'), Wild('b')\n--> 413         m = arg.match(a + b*S.ImaginaryUnit)\n    414         if not m or not (m[a].is_Float or m[b].is_Float):\n    415             return -1\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/basic.py in match(self, pattern, old)\n   1489         \"\"\"\n   1490         pattern = sympify(pattern)\n-> 1491         return pattern.matches(self, old=old)\n   1492 \n   1493     def count_ops(self, visual=None):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in matches(self, expr, repl_dict, old)\n    365 \n    366     def matches(self, expr, repl_dict={}, old=False):\n--> 367         return AssocOp._matches_commutative(self, expr, repl_dict, old)\n    368 \n    369     @staticmethod\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/operations.py in _matches_commutative(self, expr, repl_dict, old)\n    215                     d1 = w.matches(last_op, repl_dict)\n    216                     if d1 is not None:\n--> 217                         d2 = self.xreplace(d1).matches(expr, d1)\n    218                         if d2 is not None:\n    219                             return d2\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/add.py in matches(self, expr, repl_dict, old)\n    365 \n    366     def matches(self, expr, repl_dict={}, old=False):\n--> 367         return AssocOp._matches_commutative(self, expr, repl_dict, old)\n    368 \n    369     @staticmethod\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/operations.py in _matches_commutative(self, expr, repl_dict, old)\n    201             newexpr = self._combine_inverse(expr, exact)\n    202             if not old and (expr.is_Add or expr.is_Mul):\n--> 203                 if newexpr.count_ops() > expr.count_ops():\n    204                     return None\n    205             return newpattern.matches(newexpr, repl_dict)\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/expr.py in count_ops(self, visual)\n   1017         \"\"\"wrapper for count_ops that returns the operation count.\"\"\"\n   1018         from .function import count_ops\n-> 1019         return count_ops(self, visual)\n   1020 \n   1021     def args_cnc(self, cset=False, warn=True, split_1=True):\n\n/home/moorepants/miniconda3/lib/python3.5/site-packages/sympy/core/function.py in count_ops(expr, visual)\n   2378                 a.is_Pow or\n   2379                 a.is_Function or\n-> 2380                 isinstance(a, Derivative) or\n   2381                     isinstance(a, Integral)):\n   2382 \n\nKeyboardInterrupt: \n\nIn [3]: \n```\n",
      "issue_closed_at": "2017-10-13T23:48:13Z",
      "base_commit": "e0cd7d65857a90376a9b49529840f96908dd774f",
      "changes": [
        {
          "file": "sympy/core/add.py",
          "type": "function",
          "name": "matches",
          "class_name": "Add",
          "code": "def matches(self, expr, repl_dict={}, old=False):\n        return AssocOp._matches_commutative(self, expr, repl_dict, old)"
        },
        {
          "file": "sympy/core/function.py",
          "type": "function",
          "name": "_should_evalf",
          "class_name": "Function",
          "code": "def _should_evalf(cls, arg):\n        \"\"\"\n        Decide if the function should automatically evalf().\n\n        By default (in this implementation), this happens if (and only if) the\n        ARG is a floating point number.\n        This function is used by __new__.\n\n        Returns the precision to evalf to, or -1 if it shouldn't evalf.\n        \"\"\"\n        from sympy.core.symbol import Wild\n        if arg.is_Float:\n            return arg._prec\n        if not arg.is_Add:\n            return -1\n        # Don't use as_real_imag() here, that's too much work\n        a, b = Wild('a'), Wild('b')\n        m = arg.match(a + b*S.ImaginaryUnit)\n        if not m or not (m[a].is_Float or m[b].is_Float):\n            return -1\n        l = [m[i]._prec for i in m if m[i].is_Float]\n        l.append(-1)\n        return max(l)"
        },
        {
          "file": "sympy/core/operations.py",
          "type": "function",
          "name": "_new_rawargs",
          "class_name": "AssocOp",
          "code": "def _new_rawargs(self, *args, **kwargs):\n        \"\"\"Create new instance of own class with args exactly as provided by\n        caller but returning the self class identity if args is empty.\n\n           This is handy when we want to optimize things, e.g.\n\n               >>> from sympy import Mul, S\n               >>> from sympy.abc import x, y\n               >>> e = Mul(3, x, y)\n               >>> e.args\n               (3, x, y)\n               >>> Mul(*e.args[1:])\n               x*y\n               >>> e._new_rawargs(*e.args[1:])  # the same as above, but faster\n               x*y\n\n           Note: use this with caution. There is no checking of arguments at\n           all. This is best used when you are rebuilding an Add or Mul after\n           simply removing one or more terms. If modification which result,\n           for example, in extra 1s being inserted (as when collecting an\n           expression's numerators and denominators) they will not show up in\n           the result but a Mul will be returned nonetheless:\n\n               >>> m = (x*y)._new_rawargs(S.One, x); m\n               x\n               >>> m == x\n               False\n               >>> m.is_Mul\n               True\n\n           Another issue to be aware of is that the commutativity of the result\n           is based on the commutativity of self. If you are rebuilding the\n           terms that came from a commutative object then there will be no\n           problem, but if self was non-commutative then what you are\n           rebuilding may now be commutative.\n\n           Although this routine tries to do as little as possible with the\n           input, getting the commutativity right is important, so this level\n           of safety is enforced: commutativity will always be recomputed if\n           self is non-commutative and kwarg `reeval=False` has not been\n           passed.\n        \"\"\"\n        if kwargs.pop('reeval', True) and self.is_commutative is False:\n            is_commutative = None\n        else:\n            is_commutative = self.is_commutative\n        return self._from_args(args, is_commutative)"
        },
        {
          "file": "sympy/core/operations.py",
          "type": "function",
          "name": "_matches_commutative",
          "class_name": "AssocOp",
          "code": "def _matches_commutative(self, expr, repl_dict={}, old=False):\n        \"\"\"\n        Matches Add/Mul \"pattern\" to an expression \"expr\".\n\n        repl_dict ... a dictionary of (wild: expression) pairs, that get\n                      returned with the results\n\n        This function is the main workhorse for Add/Mul.\n\n        For instance:\n\n        >>> from sympy import symbols, Wild, sin\n        >>> a = Wild(\"a\")\n        >>> b = Wild(\"b\")\n        >>> c = Wild(\"c\")\n        >>> x, y, z = symbols(\"x y z\")\n        >>> (a+sin(b)*c)._matches_commutative(x+sin(y)*z)\n        {a_: x, b_: y, c_: z}\n\n        In the example above, \"a+sin(b)*c\" is the pattern, and \"x+sin(y)*z\" is\n        the expression.\n\n        The repl_dict contains parts that were already matched. For example\n        here:\n\n        >>> (x+sin(b)*c)._matches_commutative(x+sin(y)*z, repl_dict={a: x})\n        {a_: x, b_: y, c_: z}\n\n        the only function of the repl_dict is to return it in the\n        result, e.g. if you omit it:\n\n        >>> (x+sin(b)*c)._matches_commutative(x+sin(y)*z)\n        {b_: y, c_: z}\n\n        the \"a: x\" is not returned in the result, but otherwise it is\n        equivalent.\n\n        \"\"\"\n        # make sure expr is Expr if pattern is Expr\n        from .expr import Add, Expr\n        from sympy import Mul\n        if isinstance(self, Expr) and not isinstance(expr, Expr):\n            return None\n\n        # handle simple patterns\n        if self == expr:\n            return repl_dict\n\n        d = self._matches_simple(expr, repl_dict)\n        if d is not None:\n            return d\n\n        # eliminate exact part from pattern: (2+a+w1+w2).matches(expr) -> (w1+w2).matches(expr-a-2)\n        from .function import WildFunction\n        from .symbol import Wild\n        wild_part = []\n        exact_part = []\n        for p in ordered(self.args):\n            if p.has(Wild, WildFunction) and (not expr.has(p)):\n                # not all Wild should stay Wilds, for example:\n                # (w2+w3).matches(w1) -> (w1+w3).matches(w1) -> w3.matches(0)\n                wild_part.append(p)\n            else:\n                exact_part.append(p)\n\n        if exact_part:\n            exact = self.func(*exact_part)\n            free = expr.free_symbols\n            if free and (exact.free_symbols - free):\n                # there are symbols in the exact part that are not\n                # in the expr; but if there are no free symbols, let\n                # the matching continue\n                return None\n            newpattern = self.func(*wild_part)\n            newexpr = self._combine_inverse(expr, exact)\n            if not old and (expr.is_Add or expr.is_Mul):\n                if newexpr.count_ops() > expr.count_ops():\n                    return None\n            return newpattern.matches(newexpr, repl_dict)\n\n        # now to real work ;)\n        i = 0\n        saw = set()\n        while expr not in saw:\n            saw.add(expr)\n            expr_list = (self.identity,) + tuple(ordered(self.make_args(expr)))\n            for last_op in reversed(expr_list):\n                for w in reversed(wild_part):\n                    d1 = w.matches(last_op, repl_dict)\n                    if d1 is not None:\n                        d2 = self.xreplace(d1).matches(expr, d1)\n                        if d2 is not None:\n                            return d2\n\n            if i == 0:\n                if self.is_Mul:\n                    # make e**i look like Mul\n                    if expr.is_Pow and expr.exp.is_Integer:\n                        if expr.exp > 0:\n                            expr = Mul(*[expr.base, expr.base**(expr.exp - 1)], evaluate=False)\n                        else:\n                            expr = Mul(*[1/expr.base, expr.base**(expr.exp + 1)], evaluate=False)\n                        i += 1\n                        continue\n\n                elif self.is_Add:\n                    # make i*e look like Add\n                    c, e = expr.as_coeff_Mul()\n                    if abs(c) > 1:\n                        if c > 0:\n                            expr = Add(*[e, (c - 1)*e], evaluate=False)\n                        else:\n                            expr = Add(*[-e, (c + 1)*e], evaluate=False)\n                        i += 1\n                        continue\n\n                    # try collection on non-Wild symbols\n                    from sympy.simplify.radsimp import collect\n                    was = expr\n                    did = set()\n                    for w in reversed(wild_part):\n                        c, w = w.as_coeff_mul(Wild)\n                        free = c.free_symbols - did\n                        if free:\n                            did.update(free)\n                            expr = collect(expr, free)\n                    if expr != was:\n                        i += 0\n                        continue\n\n                break  # if we didn't continue, there is nothing more to do\n\n        return"
        }
      ]
    },
    {
      "pr_number": 16668,
      "pr_title": "speed up Line/Segment intersections",
      "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\n\r\ncloses #16628\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nIntersections will return more quickly for Line/Segment interactions in 2D because a redundant check that the point is in the line has been eliminated.\r\n\r\nPoint containment in a 2D Segment now uses a quick check before using the triangle inequality.\r\n\r\n#### Other comments\r\n\r\nSome tests which were quite fast already were moved out of the slow test.\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- geometry\r\n  * improved response time of 2D Line/Segement interactions and Segment containment of Point\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 16628,
      "issue_title": "Geometry: Line-Segment-Intersection can run into memory problems",
      "issue_body": "Hi,\r\n\r\nwhen determining a line-segment-intersection, depending on the geometry, sympy can take up som much memory that the script will crash.\r\n\r\nI'm on Windows, using CPython 3.7.3 on sympy commit b97cbe99, although the issue exists in version 1.3 too.\r\n\r\nExample code:\r\n<pre>\r\n#!Python -u\r\n\r\nimport sys, os, time, math\r\n\r\nsys.stderr.write(str(os.getpid())+time.strftime(': starting @ %d%b%y, %H:%M\\n'))\r\n\t\r\nfrom sympy.geometry import Point, Point2D, Line, Segment, intersection\r\nfrom sympy import *\r\nimport sympy\r\n\r\nsys.stderr.write('sympy version: '+sympy.__version__+'\\n')\r\n\r\np0=Point2D(249/5, 497999/10000)\r\np1x=3*(-19659028262*sqrt(405639795226) + 676896692394731 + 6704069269*sqrt(630547164901) + 33200*sqrt(255775022850776494562626))/(2000*(sqrt(255775022850776494562626) + 995999*sqrt(405639795226) + 995999*sqrt(630547164901) + 811280586451))\r\np1y=(-498000*sqrt(255775022850776494562626) - 995999*sqrt(630547164901) + 90004251917891999 + 496005510002*sqrt(405639795226))/(10000*(sqrt(255775022850776494562626) + 995999*sqrt(405639795226) + 995999*sqrt(630547164901) + 811280586451))\r\np1=Point2D(p1x, p1y)\r\n\r\n\r\nsys.stderr.write('p1: '+str(p1)+'\\n')\r\n\r\np2=Point2D(497/10, -497/10)\r\np3=Point2D(-497/10, -497/10)\r\n\r\nl=Line(p0,p1)\r\ns=Segment(p2,p3)\r\nints=intersection(l,s)\r\n\r\n\r\nsys.stderr.write(str(os.getpid())+time.strftime(': done @ %d%b%y, %H:%M\\n'))\r\n</pre>",
      "issue_closed_at": "2019-04-17T22:08:24Z",
      "base_commit": "96925f549e10ee2d95e592d1fe84cbdd39cec3f4",
      "changes": [
        {
          "file": "sympy/geometry/line.py",
          "type": "function",
          "name": "intersect_parallel_segments",
          "class_name": "LinearEntity",
          "code": "def intersect_parallel_segments(seg1, seg2):\n            if seg1.contains(seg2):\n                return [seg2]\n            if seg2.contains(seg1):\n                return [seg1]\n\n            # direct the segments so they're oriented the same way\n            if seg1.direction.dot(seg2.direction) < 0:\n                seg2 = Segment(seg2.p2, seg2.p1)\n            # order the segments so seg1 is \"behind\" seg2\n            if seg1._span_test(seg2.p1) < 0:\n                seg1, seg2 = seg2, seg1\n            if seg2._span_test(seg1.p2) < 0:\n                return []\n            return [Segment(seg2.p1, seg1.p2)]"
        },
        {
          "file": "sympy/geometry/line.py",
          "type": "function",
          "name": "contains",
          "class_name": "Segment",
          "code": "def contains(self, other):\n        \"\"\"\n        Is the other GeometryEntity contained within this Segment?\n\n        Examples\n        ========\n\n        >>> from sympy import Point, Segment\n        >>> p1, p2 = Point(0, 1), Point(3, 4)\n        >>> s = Segment(p1, p2)\n        >>> s2 = Segment(p2, p1)\n        >>> s.contains(s2)\n        True\n        >>> from sympy import Point3D, Segment3D\n        >>> p1, p2 = Point3D(0, 1, 1), Point3D(3, 4, 5)\n        >>> s = Segment3D(p1, p2)\n        >>> s2 = Segment3D(p2, p1)\n        >>> s.contains(s2)\n        True\n        >>> s.contains((p1 + p2) / 2)\n        True\n        \"\"\"\n\n        if not isinstance(other, GeometryEntity):\n            other = Point(other, dim=self.ambient_dimension)\n        if isinstance(other, Point):\n            if Point.is_collinear(other, self.p1, self.p2):\n                d1, d2 = other - self.p1, other - self.p2\n                d = self.p2 - self.p1\n                # without the call to simplify, sympy cannot tell that an expression\n                # like (a+b)*(a/2+b/2) is always non-negative.  If it cannot be\n                # determined, raise an Undecidable error\n                try:\n                    # the triangle inequality says that |d1|+|d2| >= |d| and is strict\n                    # only if other lies in the line segment\n                    return bool(simplify(Eq(abs(d1) + abs(d2) - abs(d), 0)))\n                except TypeError:\n                    raise Undecidable(\"Cannot determine if {} is in {}\".format(other, self))\n        if isinstance(other, Segment):\n            return other.p1 in self and other.p2 in self\n\n        return False"
        },
        {
          "file": "sympy/geometry/polygon.py",
          "type": "function",
          "name": "bisectors",
          "class_name": "Triangle",
          "code": "def bisectors(self):\n        \"\"\"The angle bisectors of the triangle.\n\n        An angle bisector of a triangle is a straight line through a vertex\n        which cuts the corresponding angle in half.\n\n        Returns\n        =======\n\n        bisectors : dict\n            Each key is a vertex (Point) and each value is the corresponding\n            bisector (Segment).\n\n        See Also\n        ========\n\n        sympy.geometry.point.Point, sympy.geometry.line.Segment\n\n        Examples\n        ========\n\n        >>> from sympy.geometry import Point, Triangle, Segment\n        >>> p1, p2, p3 = Point(0, 0), Point(1, 0), Point(0, 1)\n        >>> t = Triangle(p1, p2, p3)\n        >>> from sympy import sqrt\n        >>> t.bisectors()[p2] == Segment(Point(1, 0), Point(0, sqrt(2) - 1))\n        True\n\n        \"\"\"\n        # use lines containing sides so containment check during\n        # intersection calculation can be avoided\n        # This would reduce the processing time for calculating the bisectors\n        s = [Line(l) for l in self.sides]\n        v = self.vertices\n        c = self.incenter\n        l1 = Segment(v[0], Line(v[0], c).intersection(s[1])[0])\n        l2 = Segment(v[1], Line(v[1], c).intersection(s[2])[0])\n        l3 = Segment(v[2], Line(v[2], c).intersection(s[0])[0])\n        return {v[0]: l1, v[1]: l2, v[2]: l3}"
        }
      ]
    },
    {
      "pr_number": 16331,
      "pr_title": "Make var names in generated Cython code unique",
      "pr_body": "Previously, the following call to `unfuncify` would result in non-compiling\r\nCython code due to a variable clash:\r\n\r\n    >>> unfuncify((x, y), x + y, backend='Cython')\r\n\r\nThis was because the resulting Cython code looked something like:\r\n\r\n    def autofunc_c(np.ndarray[np.double_t, ndim=1] _x, np.ndarray[np.double_t, ndim=1] _y):\r\n        cdef int _m = _y.shape[0]\r\n        cdef np.ndarray[np.double_t, ndim=1] _y = np.empty((_m))\r\n\r\nwhere the argument `_y` is declared in the function body even though it is\r\nalready passed as an argument. This commit leverages `doprint` within the C code\r\ngenerator to make sure that generated variable names are unique.\r\n\r\n<!-- 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 #15628\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nAdd a new method to `CythonCodeWrapper` that converts unique dummy variables into distinct strings and pass every expression that emits variable names through this expression. \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\n* utilities\r\n  * make variable names in generated cython code unique\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 15628,
      "issue_title": "Issue ufuncifying a two argument function with the Cython backend",
      "issue_body": "```\r\n>>> ufuncify((x, y), x + y, backend='Cython')\r\nTraceback (most recent call last):\r\n  File \"./sympy/utilities/autowrap.py\", line 168, in _process_files\r\n    retoutput = check_output(command, stderr=STDOUT)\r\n  File \"/Users/aaronmeurer/anaconda3/lib/python3.5/subprocess.py\", line 316, in check_output\r\n    **kwargs).stdout\r\n  File \"/Users/aaronmeurer/anaconda3/lib/python3.5/subprocess.py\", line 398, in run\r\n    output=stdout, stderr=stderr)\r\nsubprocess.CalledProcessError: Command '['/Users/aaronmeurer/anaconda3/bin/python', 'setup.py', 'build_ext', '--inplace']' returned non-zero exit status 1\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"<stdin>\", line 1, in <module>\r\n  File \"./sympy/utilities/autowrap.py\", line 1105, in ufuncify\r\n    tempdir, args, flags, verbose, helpers, **kwargs)\r\n  File \"./sympy/utilities/autowrap.py\", line 640, in autowrap\r\n    return code_wrapper.wrap_code(routine, helpers=helps)\r\n  File \"./sympy/utilities/autowrap.py\", line 149, in wrap_code\r\n    self._process_files(routine)\r\n  File \"./sympy/utilities/autowrap.py\", line 172, in _process_files\r\n    \" \".join(command), e.output.decode('utf-8')))\r\nsympy.utilities.autowrap.CodeWrapError: Error while executing command: /Users/aaronmeurer/anaconda3/bin/python setup.py build_ext --inplace. Command output is:\r\n\r\nError compiling Cython file:\r\n------------------------------------------------------------\r\n...\r\n    void autofunc(double *y_4785968, double *x_4785972, double *y_4785973, int m_4785969)\r\n\r\ndef autofunc_c(np.ndarray[np.double_t, ndim=1] _x, np.ndarray[np.double_t, ndim=1] _y):\r\n\r\n    cdef int _m = _y.shape[0]\r\n    cdef np.ndarray[np.double_t, ndim=1] _y = np.empty((_m))\r\n                                        ^\r\n------------------------------------------------------------\r\n\r\nwrapper_module_3.pyx:10:41: '_y' redeclared\r\n\r\nError compiling Cython file:\r\n------------------------------------------------------------\r\n...\r\ncimport numpy as np\r\n\r\ncdef extern from 'wrapped_code_3.h':\r\n    void autofunc(double *y_4785968, double *x_4785972, double *y_4785973, int m_4785969)\r\n\r\ndef autofunc_c(np.ndarray[np.double_t, ndim=1] _x, np.ndarray[np.double_t, ndim=1] _y):\r\n                                                  ^\r\n------------------------------------------------------------\r\n\r\nwrapper_module_3.pyx:7:51: Previous declaration is here\r\nwarning: wrapper_module_3.pyx:10:41: cdef variable '_y' declared after it is used\r\nCompiling wrapper_module_3.pyx because it changed.\r\n[1/1] Cythonizing wrapper_module_3.pyx\r\nTraceback (most recent call last):\r\n  File \"setup.py\", line 19, in <module>\r\n    setup(ext_modules=cythonize(ext_mods, **cy_opts))\r\n  File \"/Users/aaronmeurer/anaconda3/lib/python3.5/site-packages/Cython/Build/Dependencies.py\", line 1026, in cythonize\r\n    cythonize_one(*args)\r\n  File \"/Users/aaronmeurer/anaconda3/lib/python3.5/site-packages/Cython/Build/Dependencies.py\", line 1146, in cythonize_one\r\n    raise CompileError(None, pyx_file)\r\nCython.Compiler.Errors.CompileError: wrapper_module_3.pyx\r\n```\r\n\r\nIt works if the function just has one argument, or if you use a different backend. ",
      "issue_closed_at": "2019-03-20T15:46:53Z",
      "base_commit": "356a73cd676e0c3f1a1c3057a6895db0d82a1be7",
      "changes": [
        {
          "file": "sympy/utilities/autowrap.py",
          "type": "function",
          "name": "dump_pyx",
          "class_name": "CythonCodeWrapper",
          "code": "def dump_pyx(self, routines, f, prefix):\n        \"\"\"Write a Cython file with python wrappers\n\n        This file contains all the definitions of the routines in c code and\n        refers to the header file.\n\n        Arguments\n        ---------\n        routines\n            List of Routine instances\n        f\n            File-like object to write the file to\n        prefix\n            The filename prefix, used to refer to the proper header file.\n            Only the basename of the prefix is used.\n        \"\"\"\n        headers = []\n        functions = []\n        for routine in routines:\n            prototype = self.generator.get_prototype(routine)\n\n            # C Function Header Import\n            headers.append(self.pyx_header.format(header_file=prefix,\n                                                  prototype=prototype))\n\n            # Partition the C function arguments into categories\n            py_rets, py_args, py_loc, py_inf = self._partition_args(routine.arguments)\n\n            # Function prototype\n            name = routine.name\n            arg_string = \", \".join(self._prototype_arg(arg) for arg in py_args)\n\n            # Local Declarations\n            local_decs = []\n            for arg, val in py_inf.items():\n                proto = self._prototype_arg(arg)\n                mat, ind = val\n                local_decs.append(\"    cdef {0} = {1}.shape[{2}]\".format(proto, mat, ind))\n            local_decs.extend([\"    cdef {0}\".format(self._declare_arg(a)) for a in py_loc])\n            declarations = \"\\n\".join(local_decs)\n            if declarations:\n                declarations = declarations + \"\\n\"\n\n            # Function Body\n            args_c = \", \".join([self._call_arg(a) for a in routine.arguments])\n            rets = \", \".join([str(r.name) for r in py_rets])\n            if routine.results:\n                body = '    return %s(%s)' % (routine.name, args_c)\n                if rets:\n                    body = body + ', ' + rets\n            else:\n                body = '    %s(%s)\\n' % (routine.name, args_c)\n                body = body + '    return ' + rets\n\n            functions.append(self.pyx_func.format(name=name, arg_string=arg_string,\n                    declarations=declarations, body=body))\n\n        # Write text to file\n        if self._need_numpy:\n            # Only import numpy if required\n            f.write(self.pyx_imports)\n        f.write('\\n'.join(headers))\n        f.write('\\n'.join(functions))"
        },
        {
          "file": "sympy/utilities/autowrap.py",
          "type": "function",
          "name": "dump_pyx",
          "class_name": "CythonCodeWrapper",
          "code": "def dump_pyx(self, routines, f, prefix):\n        \"\"\"Write a Cython file with python wrappers\n\n        This file contains all the definitions of the routines in c code and\n        refers to the header file.\n\n        Arguments\n        ---------\n        routines\n            List of Routine instances\n        f\n            File-like object to write the file to\n        prefix\n            The filename prefix, used to refer to the proper header file.\n            Only the basename of the prefix is used.\n        \"\"\"\n        headers = []\n        functions = []\n        for routine in routines:\n            prototype = self.generator.get_prototype(routine)\n\n            # C Function Header Import\n            headers.append(self.pyx_header.format(header_file=prefix,\n                                                  prototype=prototype))\n\n            # Partition the C function arguments into categories\n            py_rets, py_args, py_loc, py_inf = self._partition_args(routine.arguments)\n\n            # Function prototype\n            name = routine.name\n            arg_string = \", \".join(self._prototype_arg(arg) for arg in py_args)\n\n            # Local Declarations\n            local_decs = []\n            for arg, val in py_inf.items():\n                proto = self._prototype_arg(arg)\n                mat, ind = val\n                local_decs.append(\"    cdef {0} = {1}.shape[{2}]\".format(proto, mat, ind))\n            local_decs.extend([\"    cdef {0}\".format(self._declare_arg(a)) for a in py_loc])\n            declarations = \"\\n\".join(local_decs)\n            if declarations:\n                declarations = declarations + \"\\n\"\n\n            # Function Body\n            args_c = \", \".join([self._call_arg(a) for a in routine.arguments])\n            rets = \", \".join([str(r.name) for r in py_rets])\n            if routine.results:\n                body = '    return %s(%s)' % (routine.name, args_c)\n                if rets:\n                    body = body + ', ' + rets\n            else:\n                body = '    %s(%s)\\n' % (routine.name, args_c)\n                body = body + '    return ' + rets\n\n            functions.append(self.pyx_func.format(name=name, arg_string=arg_string,\n                    declarations=declarations, body=body))\n\n        # Write text to file\n        if self._need_numpy:\n            # Only import numpy if required\n            f.write(self.pyx_imports)\n        f.write('\\n'.join(headers))\n        f.write('\\n'.join(functions))"
        },
        {
          "file": "sympy/utilities/autowrap.py",
          "type": "function",
          "name": "_prototype_arg",
          "class_name": "CythonCodeWrapper",
          "code": "def _prototype_arg(self, arg):\n        mat_dec = \"np.ndarray[{mtype}, ndim={ndim}] {name}\"\n        np_types = {'double': 'np.double_t',\n                    'int': 'np.int_t'}\n        t = arg.get_datatype('c')\n        if arg.dimensions:\n            self._need_numpy = True\n            ndim = len(arg.dimensions)\n            mtype = np_types[t]\n            return mat_dec.format(mtype=mtype, ndim=ndim, name=arg.name)\n        else:\n            return \"%s %s\" % (t, str(arg.name))"
        },
        {
          "file": "sympy/utilities/autowrap.py",
          "type": "function",
          "name": "_declare_arg",
          "class_name": "CythonCodeWrapper",
          "code": "def _declare_arg(self, arg):\n        proto = self._prototype_arg(arg)\n        if arg.dimensions:\n            shape = '(' + ','.join(str(i[1] + 1) for i in arg.dimensions) + ')'\n            return proto + \" = np.empty({shape})\".format(shape=shape)\n        else:\n            return proto + \" = 0\""
        }
      ]
    }
  ]
}