Interface to PHC.#

PHC computes numerical information about systems of polynomials over the complex numbers.

PHC implements polynomial homotopy methods to exploit structure in order to better approximate all isolated solutions. The package also includes extra tools to handle positive dimensional solution components.

AUTHORS:

  • PHC was written by J. Verschelde, R. Cools, and many others (?)

  • William Stein and Kelly ?? – first version of interface to PHC

  • Marshall Hampton – second version of interface to PHC

  • Marshall Hampton and Alex Jokela – third version, path tracking

class sage.interfaces.phc.PHC[source]#

Bases: object

A class to interface with PHCpack, for computing numerical homotopies and root counts.

EXAMPLES:

sage: from sage.interfaces.phc import phc
sage: R.<x,y> = PolynomialRing(CDF,2)
sage: testsys = [x^2 + 1, x*y - 1]
sage: phc.mixed_volume(testsys)        # optional -- phc
2
sage: v = phc.blackbox(testsys, R)     # optional -- phc
sage: sols = v.solutions()             # optional -- phc
sage: sols.sort()                      # optional -- phc
sage: sols                             # optional -- phc
[[-1.00000000000000*I, 1.00000000000000*I], [1.00000000000000*I, -1.00000000000000*I]]
sage: sol_dict = v.solution_dicts()    # optional -- phc
sage: x_sols_from_dict = [d[x] for d in sol_dict]    # optional -- phc
sage: x_sols_from_dict.sort(); x_sols_from_dict      # optional -- phc
[-1.00000000000000*I, 1.00000000000000*I]
sage: residuals = [[test_equation.change_ring(CDF).subs(sol) for test_equation in testsys] for sol in v.solution_dicts()]  # optional -- phc
sage: residuals                        # optional -- phc
[[0, 0], [0, 0]]
>>> from sage.all import *
>>> from sage.interfaces.phc import phc
>>> R = PolynomialRing(CDF,Integer(2), names=('x', 'y',)); (x, y,) = R._first_ngens(2)
>>> testsys = [x**Integer(2) + Integer(1), x*y - Integer(1)]
>>> phc.mixed_volume(testsys)        # optional -- phc
2
>>> v = phc.blackbox(testsys, R)     # optional -- phc
>>> sols = v.solutions()             # optional -- phc
>>> sols.sort()                      # optional -- phc
>>> sols                             # optional -- phc
[[-1.00000000000000*I, 1.00000000000000*I], [1.00000000000000*I, -1.00000000000000*I]]
>>> sol_dict = v.solution_dicts()    # optional -- phc
>>> x_sols_from_dict = [d[x] for d in sol_dict]    # optional -- phc
>>> x_sols_from_dict.sort(); x_sols_from_dict      # optional -- phc
[-1.00000000000000*I, 1.00000000000000*I]
>>> residuals = [[test_equation.change_ring(CDF).subs(sol) for test_equation in testsys] for sol in v.solution_dicts()]  # optional -- phc
>>> residuals                        # optional -- phc
[[0, 0], [0, 0]]
blackbox(polys, input_ring, verbose=False)[source]#

Return as a string the result of running PHC with the given polynomials under blackbox mode (the ‘-b’ option).

INPUT:

  • polys – a list of multivariate polynomials (elements of a multivariate polynomial ring).

  • input_ring – for coercion of the variables into the desired ring.

  • verbose – print lots of verbose information about what this function does.

OUTPUT:

  • a PHC_Object object containing the phcpack output string.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [x^6-y^2,y^5-1]
sage: sol = phc.blackbox(start_sys, R2)        # optional -- phc
sage: len(sol.solutions())                     # optional -- phc
30
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [x**Integer(6)-y**Integer(2),y**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)        # optional -- phc
>>> len(sol.solutions())                     # optional -- phc
30
mixed_volume(polys, verbose=False)[source]#

Compute the mixed volume of the polynomial system given by the input polys.

INPUT:

  • polys – a list of multivariate polynomials (elements of a multivariate polynomial ring).

  • verbose – print lots of verbose information about what this function does.

OUTPUT:

  • The mixed volume.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x,y,z> = PolynomialRing(QQ,3)
sage: test_sys = [(x+y+z)^2-1,x^2-x,y^2-1]
sage: phc.mixed_volume(test_sys)                # optional -- phc
4
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(3), names=('x', 'y', 'z',)); (x, y, z,) = R2._first_ngens(3)
>>> test_sys = [(x+y+z)**Integer(2)-Integer(1),x**Integer(2)-x,y**Integer(2)-Integer(1)]
>>> phc.mixed_volume(test_sys)                # optional -- phc
4
path_track(start_sys, end_sys, input_ring, c_skew=0.001, saved_start=None)[source]#

This function computes homotopy paths between the solutions of start_sys and end_sys.

INPUT:

  • start_sys – a square polynomial system, given as a list of polynomials

  • end_sys – same type as start_sys

  • input_ring – for coercion of the variables into the desired ring.

  • c_skew – optional. the imaginary part of homotopy multiplier; nonzero values are often necessary to avoid intermediate path collisions

  • saved_start – optional. A phc output file. If not given, start system solutions are computed via the phc.blackbox function.

OUTPUT:

  • a list of paths as dictionaries, with the keys variables and t-values on the path.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [x^6-y^2,y^5-1]
sage: sol = phc.blackbox(start_sys, R2)        # optional -- phc
sage: start_save = sol.save_as_start()         # optional -- phc
sage: end_sys = [x^7-2,y^5-x^2]                # optional -- phc
sage: sol_paths = phc.path_track(start_sys, end_sys, R2, saved_start = start_save)  # optional -- phc
sage: len(sol_paths)        # optional -- phc
30
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [x**Integer(6)-y**Integer(2),y**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)        # optional -- phc
>>> start_save = sol.save_as_start()         # optional -- phc
>>> end_sys = [x**Integer(7)-Integer(2),y**Integer(5)-x**Integer(2)]                # optional -- phc
>>> sol_paths = phc.path_track(start_sys, end_sys, R2, saved_start = start_save)  # optional -- phc
>>> len(sol_paths)        # optional -- phc
30
plot_paths_2d(start_sys, end_sys, input_ring, c_skew=0.001, endpoints=True, saved_start=None, rand_colors=False)[source]#

Return a graphics object of solution paths in the complex plane.

INPUT:

  • start_sys – a square polynomial system, given as a list of polynomials

  • end_sys – same type as start_sys

  • input_ring – for coercion of the variables into the desired ring.

  • c_skew – optional. the imaginary part of homotopy multiplier; nonzero values are often necessary to avoid intermediate path collisions

  • endpoints – optional. Whether to draw in the ends of paths as points.

  • saved_start – optional. A phc output file. If not given, start system solutions are computed via the phc.blackbox function.

OUTPUT:

  • lines and points of solution paths

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: from sage.structure.sage_object import SageObject
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [x^5-y^2,y^5-1]
sage: sol = phc.blackbox(start_sys, R2)    # optional -- phc
sage: start_save = sol.save_as_start()     # optional -- phc
sage: end_sys = [x^5-25,y^5-x^2]           # optional -- phc
sage: testing = phc.plot_paths_2d(start_sys, end_sys, R2)  # optional -- phc
sage: type(testing)                        # optional -- phc (normally use plot here)
<class 'sage.plot.graphics.Graphics'>
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> from sage.structure.sage_object import SageObject
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [x**Integer(5)-y**Integer(2),y**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)    # optional -- phc
>>> start_save = sol.save_as_start()     # optional -- phc
>>> end_sys = [x**Integer(5)-Integer(25),y**Integer(5)-x**Integer(2)]           # optional -- phc
>>> testing = phc.plot_paths_2d(start_sys, end_sys, R2)  # optional -- phc
>>> type(testing)                        # optional -- phc (normally use plot here)
<class 'sage.plot.graphics.Graphics'>
start_from(start_filename_or_string, polys, input_ring, path_track_file=None, verbose=False)[source]#

This computes solutions starting from a phcpack solution file.

INPUT:

  • start_filename_or_string – the filename for a phcpack start system, or the contents of such a file as a string. Variable names must match the inputring variables. The value of the homotopy variable t should be 1, not 0.

  • polys – a list of multivariate polynomials (elements of a multivariate polynomial ring).

  • input_ring: for coercion of the variables into the desired ring.

  • path_track_file: whether to save path-tracking information

  • verbose – print lots of verbose information about what this function does.

OUTPUT:

  • A solution in the form of a PHCObject.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [x^6-y^2,y^5-1]
sage: sol = phc.blackbox(start_sys, R2)        # optional -- phc
sage: start_save = sol.save_as_start()         # optional -- phc
sage: end_sys = [x^7-2,y^5-x^2]                # optional -- phc
sage: sol = phc.start_from(start_save, end_sys, R2)  # optional -- phc
sage: len(sol.solutions())                     # optional -- phc
30
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [x**Integer(6)-y**Integer(2),y**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)        # optional -- phc
>>> start_save = sol.save_as_start()         # optional -- phc
>>> end_sys = [x**Integer(7)-Integer(2),y**Integer(5)-x**Integer(2)]                # optional -- phc
>>> sol = phc.start_from(start_save, end_sys, R2)  # optional -- phc
>>> len(sol.solutions())                     # optional -- phc
30
class sage.interfaces.phc.PHC_Object(output_file_contents, input_ring)[source]#

Bases: object

A container for data from the PHCpack program - lists of float solutions, etc. Currently the file contents are kept as a string; for really large outputs this would be bad.

INPUT:

  • output_file_contents: the string output of PHCpack

  • input_ring: for coercion of the variables into the desired ring.

EXAMPLES:

sage: from sage.interfaces.phc import phc
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [(x-1)^2+(y-1)-1, x^2+y^2-1]
sage: sol = phc.blackbox(start_sys, R2)  # optional -- phc
sage: str(sum([x[0] for x in sol.solutions()]).real())[0:3]  # optional -- phc
'2.0'
>>> from sage.all import *
>>> from sage.interfaces.phc import phc
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [(x-Integer(1))**Integer(2)+(y-Integer(1))-Integer(1), x**Integer(2)+y**Integer(2)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)  # optional -- phc
>>> str(sum([x[Integer(0)] for x in sol.solutions()]).real())[Integer(0):Integer(3)]  # optional -- phc
'2.0'
classified_solution_dicts()[source]#

Return a dictionary of lists of dictionaries of solutions.

Its not as crazy as it sounds; the keys are the types of solutions as classified by phcpack: regular vs. singular, complex vs. real

INPUT:

  • None

OUTPUT:

  • A dictionary of lists of dictionaries of solutions

EXAMPLES:

sage: from sage.interfaces.phc import phc
sage: R.<x,y> = PolynomialRing(CC,2)
sage: p_sys = [x^10-y,y^2-1]
sage: sol = phc.blackbox(p_sys,R)         # optional -- phc
sage: classifieds = sol.classified_solution_dicts()          # optional -- phc
sage: str(sum([q[y] for q in classifieds['real']]))[0:3]     # optional -- phc
'2.0'
>>> from sage.all import *
>>> from sage.interfaces.phc import phc
>>> R = PolynomialRing(CC,Integer(2), names=('x', 'y',)); (x, y,) = R._first_ngens(2)
>>> p_sys = [x**Integer(10)-y,y**Integer(2)-Integer(1)]
>>> sol = phc.blackbox(p_sys,R)         # optional -- phc
>>> classifieds = sol.classified_solution_dicts()          # optional -- phc
>>> str(sum([q[y] for q in classifieds['real']]))[Integer(0):Integer(3)]     # optional -- phc
'2.0'
save_as_start(start_filename=None, sol_filter='')[source]#

Save a solution as a phcpack start file. The usual output is just as a string, but it can be saved to a file as well. Even if saved to a file, it still returns the output string.

EXAMPLES:

sage: from sage.interfaces.phc import phc
sage: R2.<x,y> = PolynomialRing(QQ,2)
sage: start_sys = [x^3-y^2,y^5-1]
sage: sol = phc.blackbox(start_sys, R2)  # optional -- phc
sage: start_save = sol.save_as_start()   # optional -- phc
sage: end_sys = [x^7-2,y^5-x^2]          # optional -- phc
sage: sol = phc.start_from(start_save, end_sys, R2)  # optional -- phc
sage: len(sol.solutions())               # optional -- phc
15
>>> from sage.all import *
>>> from sage.interfaces.phc import phc
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x', 'y',)); (x, y,) = R2._first_ngens(2)
>>> start_sys = [x**Integer(3)-y**Integer(2),y**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(start_sys, R2)  # optional -- phc
>>> start_save = sol.save_as_start()   # optional -- phc
>>> end_sys = [x**Integer(7)-Integer(2),y**Integer(5)-x**Integer(2)]          # optional -- phc
>>> sol = phc.start_from(start_save, end_sys, R2)  # optional -- phc
>>> len(sol.solutions())               # optional -- phc
15
solution_dicts(get_failures=False)[source]#

Return a list of solutions in dictionary form: variable:value.

INPUT:

  • self – for access to self_out_file_contents, the string of raw PHCpack output.

  • get_failures (optional) – a boolean. The default (False) is to not process failed homotopies. These either lie on positive-dimensional components or at infinity.

OUTPUT:

  • solution_dicts: a list of dictionaries. Each dictionary element is of the form variable:value, where the variable is an element of the input_ring, and the value is in ComplexField.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R.<x,y,z> = PolynomialRing(QQ,3)
sage: fs = [x^2-1,y^2-x,z^2-y]
sage: sol = phc.blackbox(fs,R)          # optional -- phc
sage: s_list = sol.solution_dicts()     # optional -- phc
sage: s_list.sort()                     # optional -- phc
sage: s_list[0]                         # optional -- phc
{y: 1.00000000000000, z: -1.00000000000000, x: 1.00000000000000}
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R = PolynomialRing(QQ,Integer(3), names=('x', 'y', 'z',)); (x, y, z,) = R._first_ngens(3)
>>> fs = [x**Integer(2)-Integer(1),y**Integer(2)-x,z**Integer(2)-y]
>>> sol = phc.blackbox(fs,R)          # optional -- phc
>>> s_list = sol.solution_dicts()     # optional -- phc
>>> s_list.sort()                     # optional -- phc
>>> s_list[Integer(0)]                         # optional -- phc
{y: 1.00000000000000, z: -1.00000000000000, x: 1.00000000000000}
solutions(get_failures=False)[source]#

Return a list of solutions in the ComplexField.

Use the variable_list function to get the order of variables used by PHCpack, which is usually different than the term order of the input_ring.

INPUT:

  • self – for access to self_out_file_contents, the string of raw PHCpack output.

  • get_failures (optional) – a boolean. The default (False) is to not process failed homotopies. These either lie on positive-dimensional components or at infinity.

OUTPUT:

  • solutions: a list of lists of ComplexField-valued solutions.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x1,x2> = PolynomialRing(QQ,2)
sage: test_sys = [x1^5-x1*x2^2-1, x2^5-x1*x2-1]
sage: sol = phc.blackbox(test_sys, R2)          # optional -- phc
sage: len(sol.solutions())                      # optional -- phc
25
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x1', 'x2',)); (x1, x2,) = R2._first_ngens(2)
>>> test_sys = [x1**Integer(5)-x1*x2**Integer(2)-Integer(1), x2**Integer(5)-x1*x2-Integer(1)]
>>> sol = phc.blackbox(test_sys, R2)          # optional -- phc
>>> len(sol.solutions())                      # optional -- phc
25
variable_list()[source]#

Return the variables, as strings, in the order in which PHCpack has processed them.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x1,x2> = PolynomialRing(QQ,2)
sage: test_sys = [x1^5-x1*x2^2-1, x2^5-x1*x2-1]
sage: sol = phc.blackbox(test_sys, R2)          # optional -- phc
sage: sol.variable_list()                       # optional -- phc
['x1', 'x2']
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x1', 'x2',)); (x1, x2,) = R2._first_ngens(2)
>>> test_sys = [x1**Integer(5)-x1*x2**Integer(2)-Integer(1), x2**Integer(5)-x1*x2-Integer(1)]
>>> sol = phc.blackbox(test_sys, R2)          # optional -- phc
>>> sol.variable_list()                       # optional -- phc
['x1', 'x2']
sage.interfaces.phc.get_classified_solution_dicts(output_file_contents, input_ring, get_failures=True)[source]#

Return a dictionary of lists of dictionaries of variable:value (key:value) pairs. Only used internally; see the classified_solution_dict function in the PHC_Object class definition for details.

INPUT:

  • output_file_contents – phc solution output as a string

  • input_ring – a PolynomialRing that variable names can be coerced into

OUTPUT:

  • a dictionary of lists if dictionaries of solutions, classifies by type

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x1,x2> = PolynomialRing(QQ,2)
sage: test_sys = [(x1-2)^5-x2, (x2-1)^5-1]
sage: sol = phc.blackbox(test_sys, R2)          # optional -- phc
sage: sol_classes = get_classified_solution_dicts(sol.output_file_contents,R2)  # optional -- phc
sage: len(sol_classes['real'])            # optional -- phc
1
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x1', 'x2',)); (x1, x2,) = R2._first_ngens(2)
>>> test_sys = [(x1-Integer(2))**Integer(5)-x2, (x2-Integer(1))**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(test_sys, R2)          # optional -- phc
>>> sol_classes = get_classified_solution_dicts(sol.output_file_contents,R2)  # optional -- phc
>>> len(sol_classes['real'])            # optional -- phc
1
sage.interfaces.phc.get_solution_dicts(output_file_contents, input_ring, get_failures=True)[source]#

Return a list of dictionaries of variable:value (key:value) pairs. Only used internally; see the solution_dict function in the PHC_Object class definition for details.

INPUT:

  • output_file_contents – phc solution output as a string

  • input_ring – a PolynomialRing that variable names can be coerced into

OUTPUT:

a list of dictionaries of solutions

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x1,x2> = PolynomialRing(QQ,2)
sage: test_sys = [(x1-1)^5-x2, (x2-1)^5-1]
sage: sol = phc.blackbox(test_sys, R2)             # optional -- phc
sage: test = get_solution_dicts(sol.output_file_contents,R2)  # optional -- phc
sage: str(sum([q[x1].real() for q in test]))[0:4]  # optional -- phc
'25.0'
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x1', 'x2',)); (x1, x2,) = R2._first_ngens(2)
>>> test_sys = [(x1-Integer(1))**Integer(5)-x2, (x2-Integer(1))**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(test_sys, R2)             # optional -- phc
>>> test = get_solution_dicts(sol.output_file_contents,R2)  # optional -- phc
>>> str(sum([q[x1].real() for q in test]))[Integer(0):Integer(4)]  # optional -- phc
'25.0'
sage.interfaces.phc.get_variable_list(output_file_contents)[source]#

Return the variables, as strings, in the order in which PHCpack has processed them.

EXAMPLES:

sage: from sage.interfaces.phc import *
sage: R2.<x1,x2> = PolynomialRing(QQ,2)
sage: test_sys = [(x1-2)^5-x2, (x2-1)^5-1]
sage: sol = phc.blackbox(test_sys, R2)             # optional -- phc
sage: get_variable_list(sol.output_file_contents)  # optional -- phc
['x1', 'x2']
>>> from sage.all import *
>>> from sage.interfaces.phc import *
>>> R2 = PolynomialRing(QQ,Integer(2), names=('x1', 'x2',)); (x1, x2,) = R2._first_ngens(2)
>>> test_sys = [(x1-Integer(2))**Integer(5)-x2, (x2-Integer(1))**Integer(5)-Integer(1)]
>>> sol = phc.blackbox(test_sys, R2)             # optional -- phc
>>> get_variable_list(sol.output_file_contents)  # optional -- phc
['x1', 'x2']
sage.interfaces.phc.random() x in the interval [0, 1).#