diff --git a/.github/workflows/pythonpackage.yml b/.github/workflows/pythonpackage.yml index 084f9b57..29025de5 100644 --- a/.github/workflows/pythonpackage.yml +++ b/.github/workflows/pythonpackage.yml @@ -8,7 +8,7 @@ jobs: strategy: max-parallel: 21 matrix: - python-version: ['3.7', '3.8', '3.9', '3.10', '3.11'] + python-version: ['3.7', '3.8', '3.9', '3.10', '3.11', '3.12'] os: [ubuntu-latest, macOS-latest, windows-latest] steps: @@ -27,13 +27,13 @@ jobs: sudo apt-get update -qq sudo apt-get install -qq glpk-utils - name: install xpress - if: matrix.os != 'macOS-latest' || matrix.python-version < '3.11' + if: (matrix.os != 'macOS-latest' || matrix.python-version < '3.11') && matrix.python-version < '3.12' run: pip install xpress - name: Install gurobipy run: | python -m pip install gurobipy - name: Install highspy - if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest' + if: (matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest') && matrix.python-version < '3.12' run: | pip install highspy numpy - name: Install coptpy diff --git a/doc/source/CaseStudies/a_blending_problem.rst b/doc/source/CaseStudies/a_blending_problem.rst index 7bda6926..9b4371e4 100644 --- a/doc/source/CaseStudies/a_blending_problem.rst +++ b/doc/source/CaseStudies/a_blending_problem.rst @@ -303,7 +303,7 @@ the data. :lines: 10-61 The ``prob`` variable is created to contain the formulation, and the -usual parameters are passed into :obj:`~pulp.LpProblem`. +usual parameters are passed into :class:`~pulp.LpProblem`. .. literalinclude:: ../../../examples/WhiskasModel2.py :lines: 63-64 @@ -330,7 +330,7 @@ Further list comprehensions are used to define the other 5 constraints, which ar .. literalinclude:: ../../../examples/WhiskasModel2.py :lines: 75-92 -Following this, the :ref:`writeLP` line etc follow exactly the same as +Following this, the :meth:`~pulp.LpProblem.writeLP` line etc follow exactly the same as in the simplified example. The optimal solution is 60% Beef and 40% Gel leading to a objective diff --git a/doc/source/conf.py b/doc/source/conf.py index 10285028..0104eee6 100644 --- a/doc/source/conf.py +++ b/doc/source/conf.py @@ -223,4 +223,4 @@ # Example configuration for intersphinx: refer to the Python standard library. -intersphinx_mapping = {"http://docs.python.org/3": None} +intersphinx_mapping = {"python": ("https://docs.python.org/3", None)} diff --git a/doc/source/guides/how_to_configure_solvers.rst b/doc/source/guides/how_to_configure_solvers.rst index d4283d17..26cf797f 100644 --- a/doc/source/guides/how_to_configure_solvers.rst +++ b/doc/source/guides/how_to_configure_solvers.rst @@ -12,12 +12,14 @@ PuLP has some helper functions that permit a user to query which solvers are ava import pulp as pl solver_list = pl.listSolvers() + print(solver_list) # ['GLPK_CMD', 'PYGLPK', 'CPLEX_CMD', 'CPLEX_PY', 'CPLEX_DLL', 'GUROBI', 'GUROBI_CMD', 'MOSEK', 'XPRESS', 'PULP_CBC_CMD', 'COIN_CMD', 'COINMP_DLL', 'CHOCO_CMD', 'MIPCL_CMD', 'SCIP_CMD'] If passed the `onlyAvailable=True` argument, PuLP lists the solvers that are currently available:: import pulp as pl solver_list = pl.listSolvers(onlyAvailable=True) + print(solver_list) # ['GLPK_CMD', 'CPLEX_CMD', 'CPLEX_PY', 'GUROBI', 'GUROBI_CMD', 'PULP_CBC_CMD', 'COIN_CMD'] Also, it's possible to get a solver object by using the name of the solver. Any arguments passed to this function are passed to the constructor: @@ -33,7 +35,7 @@ In the next sections, we will explain how to configure a solver to be accessible What is an environment variable -------------------------------------- -An environment variable is probably better explained `somewhere else `_. For the sake of this document, it is a text value stored during your session that allows you to configure some applications that make use of them. For example, when you write: +An environment variable is probably better explained `somewhere else `_. For the sake of this document, it is a text value stored during your session that allows you to configure some applications that make use of them. For example, when you write:: python @@ -76,11 +78,11 @@ Imagine using the ``CPLEX_CMD`` solver, the first one is really simple: model += _var + _var2 == 1 result = model.solve(solver) -The only to do was to look for the 'cplex.exe' file (in Windows, although in Linux and Mac is something similar but with 'cplex') and pass the absolute path to the solver. +The only thing to do was to look for the 'cplex.exe' file (if you're in Windows, for Linux and Mac you look for the 'cplex' file) and pass the absolute path to the solver. The second one is a little more cumbersome but you only do it once per machine. You need to configure the ``PATH`` environment variable to include the path to the ``C:\Program Files\IBM\ILOG\CPLEX_Studio128\cplex\bin\x64_win64`` directory. -Here is one random guide to editing environment variables in: `Windows `_ or `Linux or Mac `_. The idea is that once it is correctly configured you can forget about it (until you change pc or solver version). +Here is one random guide to editing environment variables in: `Windows `_ and `Linux or Mac `_. The idea is that once it is correctly configured you can forget about it (until you change pc or solver version). Once we have done that, we just do something very similar to the previous example: @@ -107,7 +109,7 @@ Whatever the reason, it's better to be safe than sorry and this means knowing wh CPLEX ******* -**Linux / Mac: add the following lines to the ~.bashrc (or ~.profile or /etc/profile or /etc/bash.bashrc) file**:: +**Linux / Mac**: add the following lines to the ~.bashrc (or ~.profile or /etc/profile or /etc/bash.bashrc) file:: export CPLEX_HOME="/opt/ibm/ILOG/CPLEX_Studio128/cplex" export CPO_HOME="/opt/ibm/ILOG/CPLEX_Studio128/cpoptimizer" @@ -115,7 +117,7 @@ CPLEX export LD_LIBRARY_PATH="${LD_LIBRARY_PATH}:${CPLEX_HOME}/bin/x86-64_linux:${CPO_HOME}/bin/x86-64_linux" export PYTHONPATH="${PYTHONPATH}:/opt/ibm/ILOG/CPLEX_Studio128/cplex/python/3.5/x86-64_linux" -**Windows: add the following environment variables (via the command line or the graphical user interface)**:: +**Windows**: add the following environment variables (via the command line or the graphical user interface):: set CPLEX_HOME=C:/Program Files/IBM/ILOG/CPLEX_Studio128/cplex set CPO_HOME=C:/Program Files/IBM/ILOG/CPLEX_Studio128/cpoptimizer @@ -126,13 +128,13 @@ CPLEX GUROBI ******* -**Linux / Mac: add the following lines to the ~.bashrc (or ~.profile or /etc/profile or /etc/bash.bashrc) file**:: +**Linux / Mac**: add the following lines to the ~.bashrc (or ~.profile or /etc/profile or /etc/bash.bashrc) file:: export GUROBI_HOME="/opt/gurobi801/linux64" export PATH="${PATH}:${GUROBI_HOME}/bin" - export LD_LIBRARY_PATH="${GUROBI_HOME}/lib" + export LD_LIBRARY_PATH="${LD_LIBRARY_PATH}:${GUROBI_HOME}/lib" -**Windows: add the following environment variables (via the command line or graphical user interface)**:: +**Windows**: add the following environment variables (via the command line or graphical user interface):: set GUROBI_HOME=/opt/gurobi801/linux64 set PATH=%PATH%;%GUROBI_HOME%/bin @@ -142,7 +144,7 @@ GUROBI Configuring where the CMD solvers write their temporary files --------------------------------------------------------------------------- -In the case of solver APIs that use the command line (again, those that end in ``CMD``, sometimes a user wants to control where the files are written. There are plenty of options. +In the case of solver APIs that use the command line (again, those that end in ``CMD``), sometimes a user wants to control where the files are written. There are plenty of options. By default, PuLP does not keep the intermediary files (the \*.mps, \*.lp, \*.mst, \*.sol) and they are written in a temporary directory of the operating system. PuLP looks for the TEMP, TMP and TMPDIR environment variables to write the file (in that order). After using them, PuLP deletes them. If you change any of these environment variables before solving, you should be able to choose where you want PuLP to write the results. @@ -191,6 +193,10 @@ PuLP has the integrations with the official python API solvers for the following * Gurobi (GUROBI) * Cplex (CPLEX_PY) * Xpress (XPRESS_PY) +* HiGHS (HiGHS) +* SCIP (SCIP_PY) +* XPRESS (XPRESS_PY) +* COPT (COPT) These API offer a series of advantages over using the command line option: @@ -212,7 +218,7 @@ Installing GUROBI For this solver to work, the only option is to install the python package that comes with the gurobi installation. -Following my installation paths it would be (Linux): +Following my installation paths it would be (Linux):: cd /opt/gurobi801/linux64/ sudo python3 setup.py install @@ -220,6 +226,7 @@ Following my installation paths it would be (Linux): As you can see, it is necessary to have admin rights to install it. .. _solver-specific-config: + Using solver-specific functionality ********************************************** diff --git a/pulp/apis/xpress_api.py b/pulp/apis/xpress_api.py index 94d4dba5..ae45c96f 100644 --- a/pulp/apis/xpress_api.py +++ b/pulp/apis/xpress_api.py @@ -537,7 +537,7 @@ def buildSolverModel(self, lp): # passed explicitly into the constructor. for option in self.options: if isinstance(option, tuple): - name = optione[0] + name = option[0] value = option[1] else: fields = option.split("=", 1) diff --git a/pulp/pulp.py b/pulp/pulp.py index aa384327..d19a889c 100644 --- a/pulp/pulp.py +++ b/pulp/pulp.py @@ -1562,7 +1562,7 @@ def addVariable(self, variable): """ Adds a variable to the problem before a constraint is added - @param variable: the variable to be added + :param variable: the variable to be added """ if variable.hash not in self._variable_ids: self._variables.append(variable) @@ -1572,7 +1572,7 @@ def addVariables(self, variables): """ Adds variables to the problem before a constraint is added - @param variables: the variables to be added + :param variables: the variables to be added """ for v in variables: self.addVariable(v) @@ -1686,7 +1686,7 @@ def extend(self, other, use_objective=True): extends an LpProblem by adding constraints either from a dictionary a tuple or another LpProblem object. - @param use_objective: determines whether the objective is imported from + :param bool use_objective: determines whether the objective is imported from the other problem For dictionaries the constraints will be named with the keys @@ -1742,6 +1742,7 @@ def writeMPS( :param bool rename: if True, normalized names are used for variables and constraints :param mip: variables and variable renames :return: + Side Effects: - The file is created """ @@ -1763,6 +1764,7 @@ def writeLP(self, filename, writeSOS=1, mip=1, max_length=100): :param str filename: the name of the file to be created. :return: variables + Side Effects: - The file is created """ diff --git a/pulp/utilities.py b/pulp/utilities.py index 407d3247..cbb73ee6 100644 --- a/pulp/utilities.py +++ b/pulp/utilities.py @@ -175,7 +175,7 @@ def splitDict(data): """ Split a dictionary with lists as the data, into smaller dictionaries - :param data: A dictionary with lists as the values + :param dict data: A dictionary with lists as the values :return: A tuple of dictionaries each containing the data separately, with the same dictionary keys @@ -198,9 +198,9 @@ def read_table(data, coerce_type, transpose=False): simple script ::return: a dictionary of with the keys being a tuple of the strings in the first row and colum of the table - ::param data: the multiline string containing the table data - ::param coerce_type: the type that the table data is converted to - ::param transpose: reverses the data if needed + :param str data: the multiline string containing the table data + :param coerce_type: the type that the table data is converted to + :param bool transpose: reverses the data if needed Example: >>> table_data = '''