{
  "instance_id": "sympy__sympy-24102",
  "repo": "sympy/sympy",
  "created_at": "2022-10-01T18:41:32Z",
  "problem_statement": "Cannot parse Greek characters (and possibly others) in parse_mathematica\nThe old Mathematica parser `mathematica` in the package `sympy.parsing.mathematica` was able to parse e.g. Greek characters. Hence the following example works fine:\r\n```\r\nfrom sympy.parsing.mathematica import mathematica\r\nmathematica('\u03bb')\r\nOut[]: \r\n\u03bb\r\n```\r\n\r\nAs of SymPy v. 1.11, the `mathematica` function is deprecated, and is replaced by `parse_mathematica`. This function, however, seems unable to handle the simple example above:\r\n```\r\nfrom sympy.parsing.mathematica import parse_mathematica\r\nparse_mathematica('\u03bb')\r\nTraceback (most recent call last):\r\n...\r\nFile \"<string>\", line unknown\r\nSyntaxError: unable to create a single AST for the expression\r\n```\r\n\r\nThis appears to be due to a bug in `parse_mathematica`, which is why I have opened this issue.\r\n\r\nThanks in advance!\nCannot parse Greek characters (and possibly others) in parse_mathematica\nThe old Mathematica parser `mathematica` in the package `sympy.parsing.mathematica` was able to parse e.g. Greek characters. Hence the following example works fine:\r\n```\r\nfrom sympy.parsing.mathematica import mathematica\r\nmathematica('\u03bb')\r\nOut[]: \r\n\u03bb\r\n```\r\n\r\nAs of SymPy v. 1.11, the `mathematica` function is deprecated, and is replaced by `parse_mathematica`. This function, however, seems unable to handle the simple example above:\r\n```\r\nfrom sympy.parsing.mathematica import parse_mathematica\r\nparse_mathematica('\u03bb')\r\nTraceback (most recent call last):\r\n...\r\nFile \"<string>\", line unknown\r\nSyntaxError: unable to create a single AST for the expression\r\n```\r\n\r\nThis appears to be due to a bug in `parse_mathematica`, which is why I have opened this issue.\r\n\r\nThanks in advance!\n",
  "patch": "diff --git a/sympy/parsing/mathematica.py b/sympy/parsing/mathematica.py\n--- a/sympy/parsing/mathematica.py\n+++ b/sympy/parsing/mathematica.py\n@@ -654,7 +654,7 @@ def _from_mathematica_to_tokens(self, code: str):\n             code_splits[i] = code_split\n \n         # Tokenize the input strings with a regular expression:\n-        token_lists = [tokenizer.findall(i) if isinstance(i, str) else [i] for i in code_splits]\n+        token_lists = [tokenizer.findall(i) if isinstance(i, str) and i.isascii() else [i] for i in code_splits]\n         tokens = [j for i in token_lists for j in i]\n \n         # Remove newlines at the beginning\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": 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": 24078,
      "pr_title": "Replace deprecated .isSet() method with .is_set()",
      "pr_body": "\r\n\r\n#### References to other Issues or PRs\r\nFixes #24074\r\n\r\n#### Brief description of what is fixed or changed\r\n\r\nChange pyglet threading code to use `Event.is_set()` which is the approved replacement for `Event.isSet()`\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\r\n<!-- BEGIN RELEASE NOTES -->\r\n\r\n* plotting\r\n  * Replace use of deprecated `Event.isSet()` method\r\n \r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 24074,
      "issue_title": "pygletplot is using deprecated threading.event.isSet()",
      "issue_body": "`isSet()` is deprecated in favour of `is_set()`\r\n\r\nhttps://docs.python.org/3/library/threading.html#threading.Event.is_set\r\n\r\nWhen I try to run `sympy.test(\"sympy/plotting/pygletplot\")`, this causes errors because the threading library interprets a `DeprecationWarning` inside a thread as an exception and `raise`s it outside the thread.\r\n\r\nIt seems like the only place using this function is plot_mode_base.py\r\n\r\nhttps://github.com/sympy/sympy/search?q=isSet\r\n\r\nThe `is_set` alternative was added in Python 2.6, so I don't think we need to use `isSet` for backwards compatibility any more: https://docs.python.org/2.6/library/threading.html#threading.Event.is_set",
      "issue_closed_at": "2022-09-23T09:45:57Z",
      "base_commit": "514579c655bf22e2af14f0743376ae1d7befe345",
      "changes": [
        {
          "file": "sympy/plotting/pygletplot/plot_mode_base.py",
          "type": "function",
          "name": "_calculate_all",
          "class_name": "PlotModeBase",
          "code": "def _calculate_all(self):\n        self._calculate_verts()\n        self._calculate_cverts()"
        },
        {
          "file": "sympy/plotting/pygletplot/plot_mode_base.py",
          "type": "function",
          "name": "_calculate_verts",
          "class_name": "PlotModeBase",
          "code": "def _calculate_verts(self):\n        if self._calculating_verts.isSet():\n            return\n        self._calculating_verts.set()\n        try:\n            self._on_calculate_verts()\n        finally:\n            self._calculating_verts.clear()\n        if callable(self.bounds_callback):\n            self.bounds_callback()"
        },
        {
          "file": "sympy/plotting/pygletplot/plot_mode_base.py",
          "type": "function",
          "name": "_calculate_cverts",
          "class_name": "PlotModeBase",
          "code": "def _calculate_cverts(self):\n        if self._calculating_verts.isSet():\n            return\n        while self._calculating_cverts.isSet():\n            sleep(0)  # wait for previous calculation\n        self._calculating_cverts.set()\n        try:\n            self._on_calculate_cverts()\n        finally:\n            self._calculating_cverts.clear()"
        },
        {
          "file": "sympy/plotting/pygletplot/plot_mode_base.py",
          "type": "function",
          "name": "_get_calculating_verts_len",
          "class_name": "PlotModeBase",
          "code": "def _get_calculating_verts_len(self):\n        return self._calculating_verts_len"
        }
      ]
    },
    {
      "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": 11384,
      "pr_title": "Update formal power series printing",
      "pr_body": "fixes #11102 \nping @asmeurer \n",
      "issue_id": 11102,
      "issue_title": "fps should print as a formal power series",
      "issue_body": "When I first used `fps`, I didn't realize it really was a formal power series as it claims to be, because it prints like a normal series (same as `series`)\n\n```\nIn [21]: fps(sin(x))\nOut[21]:\n     3     5\n    x     x     \u239b 6\u239e\nx - \u2500\u2500 + \u2500\u2500\u2500 + O\u239dx \u23a0\n    6    120\n```\n\nBut if you look at the string form, you see\n\n```\nIn [22]: print(fps(sin(x)))\nFormalPowerSeries(sin(x), x, 0, 1, (SeqFormula(Piecewise(((-1/4)**(_k/2 - 1/2)/(RisingFactorial(3/2, _k/2 - 1/2)*factorial(_k/2 - 1/2)), Eq(Mod(_k, 2), 1)), (0, True)), (_k, 2, oo)), SeqFormula(x**_k, (_k, 0, oo)), x))\n```\n\nThat is, it really does represent it as the formula `Sum((-1)**n/factorial(2*n + 1)*x**n, (n, 0, oo))` (albiet, not simplified). It out to print it like this, so you can see that that's what it's working with.\n\nSide question: if you enter something it can't compute, it just returns the function\n\n```\nIn [25]: fps(tan(x))\nOut[25]: tan(x)\n```\n\nIs that intentional? It seems like it ought to raise an exception in that case. \n\n@leosartaj \n",
      "issue_closed_at": "2016-07-14T22:47:58Z",
      "base_commit": "496e776108957d8c049cbef49522cef4c1955e2f",
      "changes": [
        {
          "file": "sympy/printing/latex.py",
          "type": "function",
          "name": "_print_FourierSeries",
          "class_name": "LatexPrinter",
          "code": "def _print_FourierSeries(self, s):\n        return self._print_Add(s.truncate()) + self._print(' + \\ldots')"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_FourierSeries",
          "class_name": "PrettyPrinter",
          "code": "def _print_FourierSeries(self, s):\n        if self._use_unicode:\n            dots = u(\"\\N{HORIZONTAL ELLIPSIS}\")\n        else:\n            dots = '...'\n        return self._print_Add(s.truncate()) + self._print(dots)"
        }
      ]
    }
  ]
}