Metadata-Version: 2.1
Name: partialwrap
Version: 2.0
Summary: partialwrap: a small Python library providing wrappers for external executables and Python functions to be used easily with Python’s functools.partial.
Home-page: https://github.com/mcuntz/partialwrap
Author: Matthias Cuntz
Author-email: mc@macu.de
Maintainer: Matthias Cuntz
Maintainer-email: mc@macu.de
License: MIT
Project-URL: Documentation, https://mcuntz.github.io/partialwrap/
Project-URL: Source, https://github.com/mcuntz/partialwrap
Project-URL: Tracker, https://github.com/mcuntz/partialwrap/issues
Project-URL: Changelog, https://github.com/mcuntz/partialwrap/blob/main/CHANGELOG.rst
Project-URL: Conda-Forge, https://anaconda.org/conda-forge/partialwrap
Platform: any
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: End Users/Desktop
Classifier: Intended Audience :: Science/Research
Classifier: License :: OSI Approved :: MIT License
Classifier: Natural Language :: English
Classifier: Operating System :: MacOS
Classifier: Operating System :: MacOS :: MacOS X
Classifier: Operating System :: Microsoft
Classifier: Operating System :: Microsoft :: Windows
Classifier: Operating System :: POSIX
Classifier: Operating System :: Unix
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3 :: Only
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Topic :: Scientific/Engineering
Classifier: Topic :: Scientific/Engineering :: Mathematics
Classifier: Topic :: Software Development
Classifier: Topic :: Utilities
Requires-Python: >=3.8
Description-Content-Type: text/x-rst
License-File: LICENSE
License-File: AUTHORS.rst
Requires-Dist: numpy
Provides-Extra: doc
Requires-Dist: numpy; extra == "doc"
Requires-Dist: scipy; extra == "doc"
Requires-Dist: emcee; extra == "doc"
Requires-Dist: numpydoc<2,>=1.1; extra == "doc"
Requires-Dist: jinja2>=2; extra == "doc"
Requires-Dist: sphinx<7,>=4; extra == "doc"
Requires-Dist: sphinx_book_theme>=1.0.1; extra == "doc"
Provides-Extra: test
Requires-Dist: coverage[toml]<6,>=5.2.1; extra == "test"
Requires-Dist: pytest<7,>=6.0; extra == "test"
Requires-Dist: pytest-cov<3,>=2.11.0; extra == "test"

partialwrap – Wrappers of external executables and Python functions for functools.partial
=========================================================================================

A a small Python library providing wrappers for external executables
to be used easily with Python’s functools.partial.

|DOI| |PyPI version| |Conda version| |License| |Build Status| |Coverage Status|


About partialwrap
-----------------

**partialwrap** is a Python library providing easy wrapper functions
to use with Python’s `functools.partial`_. Partial's ingenious
mechanism allows to use even very complex functions with many
arguments and keyword arguments with routines that need functions in
the simple form ``func(x)``. This includes Python's `map` function,
the minimizers in `scipy.optimize`_, and plenty of third-party modules
such as `emcee`_ or `pyeee`_. **partialwrap** allows to use any
external executable as well as any Python function with arbitrary
arguments and keywords to be used with `functools.partial`_. It also
allows to use the executables or functions with easy parallelization
of code with, for example, the parallel `map` function of the
`multiprocessing`_ module or via the Message Passing Interface
(`MPI`_).


Documentation
-------------

The complete documentation for **partialwrap** is available at Github
pages:

   https://mcuntz.github.io/partialwrap/


Quick usage guide
-----------------

**partialwrap** provides two wrapper functions to work with external
executables: ``partialwrap.exe_wrapper`` and
``partialwrap.exe_mask_wrapper``.

**partialwrap** writes the input arguments to the wrapper functions
into files that can be read by the external executable. The executable
should write its result to a file that will then be read by
**partialwrap** in return. This means **partialwrap** needs to have a
function ``parameterwriter`` that writes the parameter file
``parameterfile`` needed by the executable ``exe``.  It then also
needs to have a function ``outputreader`` for reading the output file
``outputfile`` of the external executable, perhaps calculating an
objective function value.

Take the `Rastrigin function`_, which is a popular function for
performance testing of optimization algorithms:
``y = a*n + sum_i^n(x_i^2 - a*cos(b*x_i))``. It has a global minimum
of ``0`` at all ``x_i = 0``. ``a`` influences mainly the depth of the
(local and global) minima, whereas ``b`` influences mainly the size of
the minima. A common form uses ``a = 10`` and ``b = 2*pi``. The
parameters ``x_i`` should then be in the interval [-5.12, 5.12].

Consider for simplicity an external Python program
(e.g. ``rastrigin1.py``) that calculates the Rastrigin function with
``a = 10`` and ``b = 2*pi``, reading in an arbitrary number of
parameters ``x_i`` from a ``parameterfile = params.txt`` and writing
its output into ``outputfile = out.txt``:

.. code:: python

   # File: rastrigin1.py
   import numpy as np
   from partialwrap import standard_parameter_reader

   # Rastrigin function a=10, b=2*pi
   def rastrigin1(x):
       return 10. * len(x) + np.sum(x**2 - 10. * np.cos(2. * np.pi * x))

   # read parameters
   x = standard_parameter_reader('params.txt')

   # calc function
   y = rastrigin1(x)

   # write output file
   with open('out.txt', 'w') as ff:
       print(y, file=ff)

This program can be called on the command line (if `params.txt` is
present) with:

.. code:: bash

   python rastrigin1.py

The external program calculating the Rastrigin function could, of
course, also be written in any compiled language such as C or
Fortran. See the `userguide`_ for details. The external program, here
the Python version, can be used with Python's `functools.partial`_ and
the wrapper function ``partialwrap.exe_wrapper``:

.. code:: python

   import scipy.optimize as opt
   from functools import partial
   from partialwrap import exe_wrapper
   from partialwrap import standard_parameter_writer, standard_output_reader

   rastrigin_exe   = ['python3', 'rastrigin1.py']
   parameterfile   = 'params.txt'
   parameterwriter = standard_parameter_writer
   outputfile      = 'out.txt'
   outputreader    = standard_output_reader
   rastrigin_wrap  = partial(exe_wrapper, rastrigin_exe,
                             parameterfile, parameterwriter,
                             outputfile, outputreader, {})

   x0  = [0.1, 0.2, 0.3]
   res = opt.minimize(rastrigin_wrap, x0, method='BFGS')

The `scipy.optimize`_ function ``minimize()`` passes its sampled
parameters to ``exe_wrapper``, which writes it to the file
``parameterfile = 'params.txt'``. It then calls ``rastrigin_exe =
'python3 rastrigin1.py'`` and reads its ``outputfile = 'out.txt'``.
``partialwrap.standard_parameter_reader`` and
``partialwrap.standard_parameter_writer`` are convenience functions
that read and write one parameter per line in a file without a
header. The empty dictionary at the end is explained in the
`userguide`_.

More elaborate input/output of the external program can simply be done
by replacing ``standard_parameter_reader`` and
``standard_parameter_writer`` with appropriate functions, while the
rest stays pretty much the same.


Installation
------------

The easiest way to install is via `pip`:

.. code-block:: bash

   pip install partialwrap

or via `conda`:

.. code-block:: bash

   conda install -c conda-forge partialwrap


Requirements:
-------------

-  `NumPy <https://www.numpy.org>`__


License
-------

**partialwrap** is distributed under the MIT License. See the
`LICENSE`_ file for details.

Copyright (c) 2016-2023 Matthias Cuntz

The project structure is based on a `template`_ provided by `Sebastian Müller`_.


.. |DOI| image:: https://zenodo.org/badge/DOI/10.5281/zenodo.3893705.svg
   :target: https://doi.org/10.5281/zenodo.3893705
.. |PyPI version| image:: https://badge.fury.io/py/partialwrap.svg
   :target: https://badge.fury.io/py/partialwrap
.. |Conda version| image:: https://anaconda.org/conda-forge/partialwrap/badges/version.svg
   :target: https://anaconda.org/conda-forge/partialwrap
.. |License| image:: http://img.shields.io/badge/license-MIT-blue.svg?style=flat
   :target: https://github.com/mcuntz/partialwrap/blob/master/LICENSE
.. |Build Status| image:: https://github.com/mcuntz/partialwrap/workflows/Continuous%20Integration/badge.svg?branch=main
   :target: https://github.com/mcuntz/partialwrap/actions
.. |Coverage Status| image:: https://coveralls.io/repos/github/mcuntz/partialwrap/badge.svg?branch=master
   :target: https://coveralls.io/github/mcuntz/partialwrap?branch=master

.. _functools.partial: https://docs.python.org/3/library/functools.html#functools.partial
.. _scipy.optimize: https://docs.scipy.org/doc/scipy/reference/tutorial/optimize.html
.. _emcee: https://github.com/dfm/emcee
.. _pyeee: https://github.com/mcuntz/pyeee
.. _multiprocessing: https://docs.python.org/3/library/multiprocessing.html
.. _MPI: https://bitbucket.org/mpi4py/mpi4py
.. _Rastrigin function: https://en.wikipedia.org/wiki/Rastrigin_function
.. _partial: https://docs.python.org/3/library/functools.html#functools.partial
.. _functools: https://docs.python.org/3/library/functools.html
.. _userguide: https://mcuntz.github.io/partialwrap/html/userguide.html
.. _LICENSE: https://github.com/mcuntz/partialwrap/LICENSE
.. _template: https://github.com/MuellerSeb/template
.. _Sebastian Müller: https://github.com/MuellerSeb
